1 /* 2 * Copyright 2005-2006 Jacek Caban for CodeWeavers 3 * 4 * This library is free software; you can redistribute it and/or 5 * modify it under the terms of the GNU Lesser General Public 6 * License as published by the Free Software Foundation; either 7 * version 2.1 of the License, or (at your option) any later version. 8 * 9 * This library is distributed in the hope that it will be useful, 10 * but WITHOUT ANY WARRANTY; without even the implied warranty of 11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 12 * Lesser General Public License for more details. 13 * 14 * You should have received a copy of the GNU Lesser General Public 15 * License along with this library; if not, write to the Free Software 16 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA 17 */ 18 19 #include "ieframe.h" 20 21 #include "exdispid.h" 22 #include "mshtml.h" 23 #include "perhist.h" 24 #include "initguid.h" 25 26 #include "wine/debug.h" 27 28 WINE_DEFAULT_DEBUG_CHANNEL(ieframe); 29 30 DEFINE_OLEGUID(CGID_DocHostCmdPriv, 0x000214D4L, 0, 0); 31 32 #define DOCHOST_DOCCANNAVIGATE 0 33 34 /* Undocumented notification, see mshtml tests */ 35 #define CMDID_EXPLORER_UPDATEHISTORY 38 36 37 static ATOM doc_view_atom = 0; 38 39 void push_dochost_task(DocHost *This, task_header_t *task, task_proc_t proc, task_destr_t destr, BOOL send) 40 { 41 BOOL is_empty; 42 43 task->proc = proc; 44 task->destr = destr; 45 46 is_empty = list_empty(&This->task_queue); 47 list_add_tail(&This->task_queue, &task->entry); 48 49 if(send) 50 SendMessageW(This->frame_hwnd, WM_DOCHOSTTASK, 0, 0); 51 else if(is_empty) 52 PostMessageW(This->frame_hwnd, WM_DOCHOSTTASK, 0, 0); 53 } 54 55 LRESULT process_dochost_tasks(DocHost *This) 56 { 57 task_header_t *task; 58 59 while(!list_empty(&This->task_queue)) { 60 task = LIST_ENTRY(This->task_queue.next, task_header_t, entry); 61 list_remove(&task->entry); 62 63 task->proc(This, task); 64 task->destr(task); 65 } 66 67 return 0; 68 } 69 70 void abort_dochost_tasks(DocHost *This, task_proc_t proc) 71 { 72 task_header_t *task, *cursor; 73 74 LIST_FOR_EACH_ENTRY_SAFE(task, cursor, &This->task_queue, task_header_t, entry) { 75 if(proc && proc != task->proc) 76 continue; 77 78 list_remove(&task->entry); 79 task->destr(task); 80 } 81 } 82 83 void on_commandstate_change(DocHost *doc_host, LONG command, BOOL enable) 84 { 85 DISPPARAMS dispparams; 86 VARIANTARG params[2]; 87 88 TRACE("command=%d enable=%d\n", command, enable); 89 90 dispparams.cArgs = 2; 91 dispparams.cNamedArgs = 0; 92 dispparams.rgdispidNamedArgs = NULL; 93 dispparams.rgvarg = params; 94 95 V_VT(params) = VT_BOOL; 96 V_BOOL(params) = enable ? VARIANT_TRUE : VARIANT_FALSE; 97 98 V_VT(params+1) = VT_I4; 99 V_I4(params+1) = command; 100 101 call_sink(doc_host->cps.wbe2, DISPID_COMMANDSTATECHANGE, &dispparams); 102 103 doc_host->container_vtbl->on_command_state_change(doc_host, command, enable); 104 } 105 106 void update_navigation_commands(DocHost *dochost) 107 { 108 unsigned pos = dochost->travellog.loading_pos == -1 ? dochost->travellog.position : dochost->travellog.loading_pos; 109 110 on_commandstate_change(dochost, CSC_NAVIGATEBACK, pos > 0); 111 on_commandstate_change(dochost, CSC_NAVIGATEFORWARD, pos < dochost->travellog.length); 112 } 113 114 static void notif_complete(DocHost *This, DISPID dispid) 115 { 116 DISPPARAMS dispparams; 117 VARIANTARG params[2]; 118 VARIANT url; 119 120 dispparams.cArgs = 2; 121 dispparams.cNamedArgs = 0; 122 dispparams.rgdispidNamedArgs = NULL; 123 dispparams.rgvarg = params; 124 125 V_VT(params) = (VT_BYREF|VT_VARIANT); 126 V_BYREF(params) = &url; 127 128 V_VT(params+1) = VT_DISPATCH; 129 V_DISPATCH(params+1) = (IDispatch*)This->wb; 130 131 V_VT(&url) = VT_BSTR; 132 V_BSTR(&url) = SysAllocString(This->url); 133 134 TRACE("%d >>>\n", dispid); 135 call_sink(This->cps.wbe2, dispid, &dispparams); 136 TRACE("%d <<<\n", dispid); 137 138 SysFreeString(V_BSTR(&url)); 139 This->busy = VARIANT_FALSE; 140 } 141 142 static void object_available(DocHost *This) 143 { 144 IHlinkTarget *hlink; 145 HRESULT hres; 146 147 TRACE("(%p)\n", This); 148 149 if(!This->document) { 150 WARN("document == NULL\n"); 151 return; 152 } 153 154 hres = IUnknown_QueryInterface(This->document, &IID_IHlinkTarget, (void**)&hlink); 155 if(SUCCEEDED(hres)) { 156 hres = IHlinkTarget_Navigate(hlink, 0, NULL); 157 IHlinkTarget_Release(hlink); 158 if(FAILED(hres)) 159 FIXME("Navigate failed\n"); 160 }else { 161 IOleObject *ole_object; 162 RECT rect; 163 164 TRACE("No IHlink iface\n"); 165 166 hres = IUnknown_QueryInterface(This->document, &IID_IOleObject, (void**)&ole_object); 167 if(FAILED(hres)) { 168 FIXME("Could not get IOleObject iface: %08x\n", hres); 169 return; 170 } 171 172 GetClientRect(This->hwnd, &rect); 173 hres = IOleObject_DoVerb(ole_object, OLEIVERB_SHOW, NULL, &This->IOleClientSite_iface, -1, This->hwnd, &rect); 174 IOleObject_Release(ole_object); 175 if(FAILED(hres)) 176 FIXME("DoVerb failed: %08x\n", hres); 177 } 178 } 179 180 static HRESULT get_doc_ready_state(DocHost *This, READYSTATE *ret) 181 { 182 DISPPARAMS dp = {NULL,NULL,0,0}; 183 IDispatch *disp; 184 EXCEPINFO ei; 185 VARIANT var; 186 HRESULT hres; 187 188 hres = IUnknown_QueryInterface(This->document, &IID_IDispatch, (void**)&disp); 189 if(FAILED(hres)) 190 return hres; 191 192 hres = IDispatch_Invoke(disp, DISPID_READYSTATE, &IID_NULL, LOCALE_SYSTEM_DEFAULT, DISPATCH_PROPERTYGET, 193 &dp, &var, &ei, NULL); 194 IDispatch_Release(disp); 195 if(FAILED(hres)) { 196 WARN("Invoke(DISPID_READYSTATE failed: %08x\n", hres); 197 return hres; 198 } 199 200 if(V_VT(&var) != VT_I4) { 201 WARN("V_VT(var) = %d\n", V_VT(&var)); 202 VariantClear(&var); 203 return E_FAIL; 204 } 205 206 *ret = V_I4(&var); 207 return S_OK; 208 } 209 210 static void advise_prop_notif(DocHost *This, BOOL set) 211 { 212 IConnectionPointContainer *cp_container; 213 IConnectionPoint *cp; 214 HRESULT hres; 215 216 hres = IUnknown_QueryInterface(This->document, &IID_IConnectionPointContainer, (void**)&cp_container); 217 if(FAILED(hres)) 218 return; 219 220 hres = IConnectionPointContainer_FindConnectionPoint(cp_container, &IID_IPropertyNotifySink, &cp); 221 IConnectionPointContainer_Release(cp_container); 222 if(FAILED(hres)) 223 return; 224 225 if(set) 226 hres = IConnectionPoint_Advise(cp, (IUnknown*)&This->IPropertyNotifySink_iface, &This->prop_notif_cookie); 227 else 228 hres = IConnectionPoint_Unadvise(cp, This->prop_notif_cookie); 229 IConnectionPoint_Release(cp); 230 231 if(SUCCEEDED(hres)) 232 This->is_prop_notif = set; 233 } 234 235 void set_doc_state(DocHost *This, READYSTATE doc_state) 236 { 237 This->doc_state = doc_state; 238 if(doc_state > This->ready_state) 239 This->ready_state = doc_state; 240 } 241 242 static void update_ready_state(DocHost *This, READYSTATE ready_state) 243 { 244 if(ready_state > READYSTATE_LOADING && This->travellog.loading_pos != -1) { 245 WARN("histupdate not notified\n"); 246 This->travellog.position = This->travellog.loading_pos; 247 This->travellog.loading_pos = -1; 248 } 249 250 if(ready_state > READYSTATE_LOADING && This->doc_state <= READYSTATE_LOADING && !This->browser_service /* FIXME */) 251 notif_complete(This, DISPID_NAVIGATECOMPLETE2); 252 253 if(ready_state == READYSTATE_COMPLETE && This->doc_state < READYSTATE_COMPLETE) { 254 set_doc_state(This, READYSTATE_COMPLETE); 255 if(!This->browser_service) /* FIXME: Not fully correct */ 256 notif_complete(This, DISPID_DOCUMENTCOMPLETE); 257 }else { 258 set_doc_state(This, ready_state); 259 } 260 } 261 262 typedef struct { 263 task_header_t header; 264 IUnknown *doc; 265 READYSTATE ready_state; 266 } ready_state_task_t; 267 268 static void ready_state_task_destr(task_header_t *_task) 269 { 270 ready_state_task_t *task = (ready_state_task_t*)_task; 271 272 IUnknown_Release(task->doc); 273 heap_free(task); 274 } 275 276 static void ready_state_proc(DocHost *This, task_header_t *_task) 277 { 278 ready_state_task_t *task = (ready_state_task_t*)_task; 279 280 if(task->doc == This->document) 281 update_ready_state(This, task->ready_state); 282 } 283 284 static void push_ready_state_task(DocHost *This, READYSTATE ready_state) 285 { 286 ready_state_task_t *task = heap_alloc(sizeof(ready_state_task_t)); 287 288 IUnknown_AddRef(This->document); 289 task->doc = This->document; 290 task->ready_state = ready_state; 291 292 push_dochost_task(This, &task->header, ready_state_proc, ready_state_task_destr, FALSE); 293 } 294 295 static void object_available_task_destr(task_header_t *task) 296 { 297 heap_free(task); 298 } 299 300 static void object_available_proc(DocHost *This, task_header_t *task) 301 { 302 object_available(This); 303 } 304 305 HRESULT dochost_object_available(DocHost *This, IUnknown *doc) 306 { 307 READYSTATE ready_state; 308 task_header_t *task; 309 IOleObject *oleobj; 310 HRESULT hres; 311 312 IUnknown_AddRef(doc); 313 This->document = doc; 314 315 hres = IUnknown_QueryInterface(doc, &IID_IOleObject, (void**)&oleobj); 316 if(SUCCEEDED(hres)) { 317 CLSID clsid; 318 319 hres = IOleObject_GetUserClassID(oleobj, &clsid); 320 if(SUCCEEDED(hres)) 321 TRACE("Got clsid %s\n", 322 IsEqualGUID(&clsid, &CLSID_HTMLDocument) ? "CLSID_HTMLDocument" : debugstr_guid(&clsid)); 323 324 hres = IOleObject_SetClientSite(oleobj, &This->IOleClientSite_iface); 325 if(FAILED(hres)) 326 FIXME("SetClientSite failed: %08x\n", hres); 327 328 IOleObject_Release(oleobj); 329 }else { 330 FIXME("Could not get IOleObject iface: %08x\n", hres); 331 } 332 333 /* FIXME: Call SetAdvise */ 334 335 task = heap_alloc(sizeof(*task)); 336 push_dochost_task(This, task, object_available_proc, object_available_task_destr, FALSE); 337 338 hres = get_doc_ready_state(This, &ready_state); 339 if(SUCCEEDED(hres)) { 340 if(ready_state == READYSTATE_COMPLETE) 341 push_ready_state_task(This, READYSTATE_COMPLETE); 342 if(ready_state != READYSTATE_COMPLETE || This->doc_navigate) 343 advise_prop_notif(This, TRUE); 344 }else if(!This->doc_navigate) { 345 /* If we can't get document's ready state, there is not much we can do. 346 * Assume that document is complete at this point. */ 347 push_ready_state_task(This, READYSTATE_COMPLETE); 348 } 349 350 return S_OK; 351 } 352 353 static LRESULT resize_document(DocHost *This, LONG width, LONG height) 354 { 355 RECT rect = {0, 0, width, height}; 356 357 TRACE("(%p)->(%d %d)\n", This, width, height); 358 359 if(This->view) 360 IOleDocumentView_SetRect(This->view, &rect); 361 362 return 0; 363 } 364 365 static LRESULT WINAPI doc_view_proc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam) 366 { 367 DocHost *This; 368 369 static const WCHAR wszTHIS[] = {'T','H','I','S',0}; 370 371 if(msg == WM_CREATE) { 372 This = *(DocHost**)lParam; 373 SetPropW(hwnd, wszTHIS, This); 374 }else { 375 This = GetPropW(hwnd, wszTHIS); 376 } 377 378 switch(msg) { 379 case WM_SIZE: 380 return resize_document(This, LOWORD(lParam), HIWORD(lParam)); 381 } 382 383 return DefWindowProcW(hwnd, msg, wParam, lParam); 384 } 385 386 static void free_travellog_entry(travellog_entry_t *entry) 387 { 388 if(entry->stream) 389 IStream_Release(entry->stream); 390 heap_free(entry->url); 391 } 392 393 static IStream *get_travellog_stream(DocHost *This) 394 { 395 IPersistHistory *persist_history; 396 IStream *stream; 397 HRESULT hres; 398 399 hres = IUnknown_QueryInterface(This->document, &IID_IPersistHistory, (void**)&persist_history); 400 if(FAILED(hres)) 401 return NULL; 402 403 hres = CreateStreamOnHGlobal(NULL, TRUE, &stream); 404 if(SUCCEEDED(hres)) 405 hres = IPersistHistory_SaveHistory(persist_history, stream); 406 IPersistHistory_Release(persist_history); 407 if(FAILED(hres)) { 408 IStream_Release(stream); 409 return NULL; 410 } 411 412 return stream; 413 } 414 415 static void dump_travellog(DocHost *This) 416 { 417 unsigned i; 418 419 for(i=0; i < This->travellog.length; i++) 420 TRACE("%d: %s %s\n", i, i == This->travellog.position ? "=>" : " ", debugstr_w(This->travellog.log[i].url)); 421 if(i == This->travellog.position) 422 TRACE("%d: =>\n", i); 423 } 424 425 static void update_travellog(DocHost *This) 426 { 427 travellog_entry_t *new_entry; 428 429 static const WCHAR about_schemeW[] = {'a','b','o','u','t',':'}; 430 431 if(This->url && !strncmpiW(This->url, about_schemeW, ARRAY_SIZE(about_schemeW))) { 432 TRACE("Skipping about URL\n"); 433 return; 434 } 435 436 if(!This->travellog.log) { 437 This->travellog.log = heap_alloc(4 * sizeof(*This->travellog.log)); 438 if(!This->travellog.log) 439 return; 440 441 This->travellog.size = 4; 442 }else if(This->travellog.size < This->travellog.position+1) { 443 travellog_entry_t *new_travellog; 444 445 new_travellog = heap_realloc(This->travellog.log, This->travellog.size*2*sizeof(*This->travellog.log)); 446 if(!new_travellog) 447 return; 448 449 This->travellog.log = new_travellog; 450 This->travellog.size *= 2; 451 } 452 453 if(This->travellog.loading_pos == -1) { 454 /* Clear forward history. */ 455 while(This->travellog.length > This->travellog.position) 456 free_travellog_entry(This->travellog.log + --This->travellog.length); 457 } 458 459 new_entry = This->travellog.log + This->travellog.position; 460 461 new_entry->url = heap_strdupW(This->url); 462 TRACE("Adding %s at %d\n", debugstr_w(This->url), This->travellog.position); 463 if(!new_entry->url) 464 return; 465 466 new_entry->stream = get_travellog_stream(This); 467 468 if(This->travellog.loading_pos == -1) { 469 This->travellog.position++; 470 }else { 471 This->travellog.position = This->travellog.loading_pos; 472 This->travellog.loading_pos = -1; 473 } 474 if(This->travellog.position > This->travellog.length) 475 This->travellog.length = This->travellog.position; 476 477 dump_travellog(This); 478 } 479 480 void create_doc_view_hwnd(DocHost *This) 481 { 482 RECT rect; 483 484 static const WCHAR wszShell_DocObject_View[] = 485 {'S','h','e','l','l',' ','D','o','c','O','b','j','e','c','t',' ','V','i','e','w',0}; 486 487 if(!doc_view_atom) { 488 static WNDCLASSEXW wndclass = { 489 sizeof(wndclass), 490 CS_PARENTDC, 491 doc_view_proc, 492 0, 0 /* native uses 4*/, NULL, NULL, NULL, 493 (HBRUSH)(COLOR_WINDOW + 1), NULL, 494 wszShell_DocObject_View, 495 NULL 496 }; 497 498 wndclass.hInstance = ieframe_instance; 499 500 doc_view_atom = RegisterClassExW(&wndclass); 501 } 502 503 This->container_vtbl->get_docobj_rect(This, &rect); 504 This->hwnd = CreateWindowExW(0, wszShell_DocObject_View, 505 wszShell_DocObject_View, 506 WS_CHILD | WS_VISIBLE | WS_CLIPSIBLINGS | WS_CLIPCHILDREN | WS_TABSTOP, 507 rect.left, rect.top, rect.right, rect.bottom, This->frame_hwnd, 508 NULL, ieframe_instance, This); 509 } 510 511 void deactivate_document(DocHost *This) 512 { 513 IOleInPlaceObjectWindowless *winobj; 514 IOleObject *oleobj = NULL; 515 IHlinkTarget *hlink = NULL; 516 HRESULT hres; 517 518 if(!This->document) return; 519 520 if(This->doc_navigate) { 521 IUnknown_Release(This->doc_navigate); 522 This->doc_navigate = NULL; 523 } 524 525 if(This->is_prop_notif) 526 advise_prop_notif(This, FALSE); 527 528 if(This->view) 529 IOleDocumentView_UIActivate(This->view, FALSE); 530 531 hres = IUnknown_QueryInterface(This->document, &IID_IOleInPlaceObjectWindowless, 532 (void**)&winobj); 533 if(SUCCEEDED(hres)) { 534 IOleInPlaceObjectWindowless_InPlaceDeactivate(winobj); 535 IOleInPlaceObjectWindowless_Release(winobj); 536 } 537 538 if(This->view) { 539 IOleDocumentView_Show(This->view, FALSE); 540 IOleDocumentView_CloseView(This->view, 0); 541 IOleDocumentView_SetInPlaceSite(This->view, NULL); 542 IOleDocumentView_Release(This->view); 543 This->view = NULL; 544 } 545 546 hres = IUnknown_QueryInterface(This->document, &IID_IOleObject, (void**)&oleobj); 547 if(SUCCEEDED(hres)) 548 IOleObject_Close(oleobj, OLECLOSE_NOSAVE); 549 550 hres = IUnknown_QueryInterface(This->document, &IID_IHlinkTarget, (void**)&hlink); 551 if(SUCCEEDED(hres)) { 552 IHlinkTarget_SetBrowseContext(hlink, NULL); 553 IHlinkTarget_Release(hlink); 554 } 555 556 if(oleobj) { 557 IOleClientSite *client_site = NULL; 558 559 IOleObject_GetClientSite(oleobj, &client_site); 560 if(client_site) { 561 if(client_site == &This->IOleClientSite_iface) 562 IOleObject_SetClientSite(oleobj, NULL); 563 IOleClientSite_Release(client_site); 564 } 565 566 IOleObject_Release(oleobj); 567 } 568 569 IUnknown_Release(This->document); 570 This->document = NULL; 571 } 572 573 HRESULT refresh_document(DocHost *This, const VARIANT *level) 574 { 575 IOleCommandTarget *cmdtrg; 576 VARIANT vin, vout; 577 HRESULT hres; 578 579 if(level && (V_VT(level) != VT_I4 || V_I4(level) != REFRESH_NORMAL)) 580 FIXME("Unsupported refresh level %s\n", debugstr_variant(level)); 581 582 if(!This->document) { 583 FIXME("no document\n"); 584 return E_FAIL; 585 } 586 587 hres = IUnknown_QueryInterface(This->document, &IID_IOleCommandTarget, (void**)&cmdtrg); 588 if(FAILED(hres)) 589 return hres; 590 591 V_VT(&vin) = VT_EMPTY; 592 V_VT(&vout) = VT_EMPTY; 593 hres = IOleCommandTarget_Exec(cmdtrg, NULL, OLECMDID_REFRESH, OLECMDEXECOPT_PROMPTUSER, &vin, &vout); 594 IOleCommandTarget_Release(cmdtrg); 595 if(FAILED(hres)) 596 return hres; 597 598 VariantClear(&vout); 599 return S_OK; 600 } 601 602 void release_dochost_client(DocHost *This) 603 { 604 if(This->hwnd) { 605 DestroyWindow(This->hwnd); 606 This->hwnd = NULL; 607 } 608 609 if(This->hostui) { 610 IDocHostUIHandler_Release(This->hostui); 611 This->hostui = NULL; 612 } 613 614 if(This->client_disp) { 615 IDispatch_Release(This->client_disp); 616 This->client_disp = NULL; 617 } 618 619 if(This->frame) { 620 IOleInPlaceFrame_Release(This->frame); 621 This->frame = NULL; 622 } 623 624 if(This->olecmd) { 625 IOleCommandTarget_Release(This->olecmd); 626 This->olecmd = NULL; 627 } 628 } 629 630 static inline DocHost *impl_from_IOleCommandTarget(IOleCommandTarget *iface) 631 { 632 return CONTAINING_RECORD(iface, DocHost, IOleCommandTarget_iface); 633 } 634 635 static HRESULT WINAPI ClOleCommandTarget_QueryInterface(IOleCommandTarget *iface, 636 REFIID riid, void **ppv) 637 { 638 DocHost *This = impl_from_IOleCommandTarget(iface); 639 return IOleClientSite_QueryInterface(&This->IOleClientSite_iface, riid, ppv); 640 } 641 642 static ULONG WINAPI ClOleCommandTarget_AddRef(IOleCommandTarget *iface) 643 { 644 DocHost *This = impl_from_IOleCommandTarget(iface); 645 return IOleClientSite_AddRef(&This->IOleClientSite_iface); 646 } 647 648 static ULONG WINAPI ClOleCommandTarget_Release(IOleCommandTarget *iface) 649 { 650 DocHost *This = impl_from_IOleCommandTarget(iface); 651 return IOleClientSite_Release(&This->IOleClientSite_iface); 652 } 653 654 static HRESULT WINAPI ClOleCommandTarget_QueryStatus(IOleCommandTarget *iface, 655 const GUID *pguidCmdGroup, ULONG cCmds, OLECMD prgCmds[], OLECMDTEXT *pCmdText) 656 { 657 DocHost *This = impl_from_IOleCommandTarget(iface); 658 ULONG i; 659 660 TRACE("(%p)->(%s %u %p %p)\n", This, debugstr_guid(pguidCmdGroup), cCmds, prgCmds, 661 pCmdText); 662 for(i=0; prgCmds && i < cCmds; i++) 663 TRACE("unsupported command %u (%x)\n", prgCmds[i].cmdID, prgCmds[i].cmdf); 664 665 return E_NOTIMPL; 666 } 667 668 static HRESULT WINAPI ClOleCommandTarget_Exec(IOleCommandTarget *iface, 669 const GUID *pguidCmdGroup, DWORD nCmdID, DWORD nCmdexecopt, VARIANT *pvaIn, 670 VARIANT *pvaOut) 671 { 672 DocHost *This = impl_from_IOleCommandTarget(iface); 673 674 TRACE("(%p)->(%s %d %d %s %s)\n", This, debugstr_guid(pguidCmdGroup), nCmdID, nCmdexecopt, 675 debugstr_variant(pvaIn), debugstr_variant(pvaOut)); 676 677 if(!pguidCmdGroup) { 678 switch(nCmdID) { 679 case OLECMDID_UPDATECOMMANDS: 680 if(!This->olecmd) 681 return E_NOTIMPL; 682 return IOleCommandTarget_Exec(This->olecmd, pguidCmdGroup, nCmdID, nCmdexecopt, pvaIn, pvaOut); 683 case OLECMDID_SETDOWNLOADSTATE: 684 if(This->olecmd) 685 return IOleCommandTarget_Exec(This->olecmd, pguidCmdGroup, nCmdID, nCmdexecopt, pvaIn, pvaOut); 686 687 if(!pvaIn || V_VT(pvaIn) != VT_I4) 688 return E_INVALIDARG; 689 690 notify_download_state(This, V_I4(pvaIn)); 691 return S_OK; 692 default: 693 TRACE("Unimplemented cmdid %d\n", nCmdID); 694 return E_NOTIMPL; 695 } 696 } 697 698 if(IsEqualGUID(pguidCmdGroup, &CGID_DocHostCmdPriv)) { 699 switch(nCmdID) { 700 case DOCHOST_DOCCANNAVIGATE: 701 if(!pvaIn || V_VT(pvaIn) != VT_UNKNOWN) 702 return E_INVALIDARG; 703 704 if(This->doc_navigate) 705 IUnknown_Release(This->doc_navigate); 706 IUnknown_AddRef(V_UNKNOWN(pvaIn)); 707 This->doc_navigate = V_UNKNOWN(pvaIn); 708 return S_OK; 709 710 case 1: { 711 IHTMLWindow2 *win2; 712 SAFEARRAY *sa = V_ARRAY(pvaIn); 713 VARIANT status_code, url, htmlwindow; 714 LONG ind; 715 HRESULT hres; 716 717 if(V_VT(pvaIn) != VT_ARRAY || !sa || (SafeArrayGetDim(sa) != 1)) 718 return E_INVALIDARG; 719 720 ind = 0; 721 hres = SafeArrayGetElement(sa, &ind, &status_code); 722 if(FAILED(hres) || V_VT(&status_code)!=VT_I4) 723 return E_INVALIDARG; 724 725 ind = 1; 726 hres = SafeArrayGetElement(sa, &ind, &url); 727 if(FAILED(hres) || V_VT(&url)!=VT_BSTR) 728 return E_INVALIDARG; 729 730 ind = 3; 731 hres = SafeArrayGetElement(sa, &ind, &htmlwindow); 732 if(FAILED(hres) || V_VT(&htmlwindow)!=VT_UNKNOWN || !V_UNKNOWN(&htmlwindow)) 733 return E_INVALIDARG; 734 735 hres = IUnknown_QueryInterface(V_UNKNOWN(&htmlwindow), &IID_IHTMLWindow2, (void**)&win2); 736 if(FAILED(hres)) 737 return E_INVALIDARG; 738 739 handle_navigation_error(This, V_I4(&status_code), V_BSTR(&url), win2); 740 IHTMLWindow2_Release(win2); 741 return S_OK; 742 } 743 744 default: 745 TRACE("unsupported command %d of CGID_DocHostCmdPriv\n", nCmdID); 746 return E_NOTIMPL; 747 } 748 } 749 750 if(IsEqualGUID(pguidCmdGroup, &CGID_Explorer)) { 751 switch(nCmdID) { 752 case CMDID_EXPLORER_UPDATEHISTORY: 753 update_travellog(This); 754 update_navigation_commands(This); 755 return S_OK; 756 757 default: 758 TRACE("Unimplemented cmdid %d of CGID_Explorer\n", nCmdID); 759 return E_NOTIMPL; 760 } 761 } 762 763 if(IsEqualGUID(pguidCmdGroup, &CGID_ShellDocView)) { 764 switch(nCmdID) { 765 default: 766 TRACE("Unimplemented cmdid %d of CGID_ShellDocView\n", nCmdID); 767 return E_NOTIMPL; 768 } 769 } 770 771 if(IsEqualGUID(&CGID_DocHostCommandHandler, pguidCmdGroup)) { 772 if(!This->olecmd) 773 return E_NOTIMPL; 774 return IOleCommandTarget_Exec(This->olecmd, pguidCmdGroup, nCmdID, nCmdexecopt, pvaIn, pvaOut); 775 } 776 777 TRACE("Unimplemented cmdid %d of group %s\n", nCmdID, debugstr_guid(pguidCmdGroup)); 778 return E_NOTIMPL; 779 } 780 781 static const IOleCommandTargetVtbl OleCommandTargetVtbl = { 782 ClOleCommandTarget_QueryInterface, 783 ClOleCommandTarget_AddRef, 784 ClOleCommandTarget_Release, 785 ClOleCommandTarget_QueryStatus, 786 ClOleCommandTarget_Exec 787 }; 788 789 static inline DocHost *impl_from_IDocHostUIHandler2(IDocHostUIHandler2 *iface) 790 { 791 return CONTAINING_RECORD(iface, DocHost, IDocHostUIHandler2_iface); 792 } 793 794 static HRESULT WINAPI DocHostUIHandler_QueryInterface(IDocHostUIHandler2 *iface, 795 REFIID riid, void **ppv) 796 { 797 DocHost *This = impl_from_IDocHostUIHandler2(iface); 798 return IOleClientSite_QueryInterface(&This->IOleClientSite_iface, riid, ppv); 799 } 800 801 static ULONG WINAPI DocHostUIHandler_AddRef(IDocHostUIHandler2 *iface) 802 { 803 DocHost *This = impl_from_IDocHostUIHandler2(iface); 804 return IOleClientSite_AddRef(&This->IOleClientSite_iface); 805 } 806 807 static ULONG WINAPI DocHostUIHandler_Release(IDocHostUIHandler2 *iface) 808 { 809 DocHost *This = impl_from_IDocHostUIHandler2(iface); 810 return IOleClientSite_Release(&This->IOleClientSite_iface); 811 } 812 813 static HRESULT WINAPI DocHostUIHandler_ShowContextMenu(IDocHostUIHandler2 *iface, 814 DWORD dwID, POINT *ppt, IUnknown *pcmdtReserved, IDispatch *pdispReserved) 815 { 816 DocHost *This = impl_from_IDocHostUIHandler2(iface); 817 HRESULT hres; 818 819 TRACE("(%p)->(%d %p %p %p)\n", This, dwID, ppt, pcmdtReserved, pdispReserved); 820 821 if(This->hostui) { 822 hres = IDocHostUIHandler_ShowContextMenu(This->hostui, dwID, ppt, pcmdtReserved, 823 pdispReserved); 824 if(hres == S_OK) 825 return S_OK; 826 } 827 828 FIXME("default action not implemented\n"); 829 return E_NOTIMPL; 830 } 831 832 static HRESULT WINAPI DocHostUIHandler_GetHostInfo(IDocHostUIHandler2 *iface, 833 DOCHOSTUIINFO *pInfo) 834 { 835 DocHost *This = impl_from_IDocHostUIHandler2(iface); 836 HRESULT hres; 837 838 TRACE("(%p)->(%p)\n", This, pInfo); 839 840 if(This->hostui) { 841 hres = IDocHostUIHandler_GetHostInfo(This->hostui, pInfo); 842 if(SUCCEEDED(hres)) 843 return hres; 844 } 845 846 pInfo->dwFlags = DOCHOSTUIFLAG_DISABLE_HELP_MENU | DOCHOSTUIFLAG_OPENNEWWIN 847 | DOCHOSTUIFLAG_URL_ENCODING_ENABLE_UTF8 | DOCHOSTUIFLAG_ENABLE_INPLACE_NAVIGATION 848 | DOCHOSTUIFLAG_IME_ENABLE_RECONVERSION; 849 return S_OK; 850 } 851 852 static HRESULT WINAPI DocHostUIHandler_ShowUI(IDocHostUIHandler2 *iface, DWORD dwID, 853 IOleInPlaceActiveObject *pActiveObject, IOleCommandTarget *pCommandTarget, 854 IOleInPlaceFrame *pFrame, IOleInPlaceUIWindow *pDoc) 855 { 856 DocHost *This = impl_from_IDocHostUIHandler2(iface); 857 FIXME("(%p)->(%d %p %p %p %p)\n", This, dwID, pActiveObject, pCommandTarget, 858 pFrame, pDoc); 859 return E_NOTIMPL; 860 } 861 862 static HRESULT WINAPI DocHostUIHandler_HideUI(IDocHostUIHandler2 *iface) 863 { 864 DocHost *This = impl_from_IDocHostUIHandler2(iface); 865 FIXME("(%p)\n", This); 866 return E_NOTIMPL; 867 } 868 869 static HRESULT WINAPI DocHostUIHandler_UpdateUI(IDocHostUIHandler2 *iface) 870 { 871 DocHost *This = impl_from_IDocHostUIHandler2(iface); 872 873 TRACE("(%p)\n", This); 874 875 if(!This->hostui) 876 return S_FALSE; 877 878 return IDocHostUIHandler_UpdateUI(This->hostui); 879 } 880 881 static HRESULT WINAPI DocHostUIHandler_EnableModeless(IDocHostUIHandler2 *iface, 882 BOOL fEnable) 883 { 884 DocHost *This = impl_from_IDocHostUIHandler2(iface); 885 FIXME("(%p)->(%x)\n", This, fEnable); 886 return E_NOTIMPL; 887 } 888 889 static HRESULT WINAPI DocHostUIHandler_OnDocWindowActivate(IDocHostUIHandler2 *iface, 890 BOOL fActivate) 891 { 892 DocHost *This = impl_from_IDocHostUIHandler2(iface); 893 FIXME("(%p)->(%x)\n", This, fActivate); 894 return E_NOTIMPL; 895 } 896 897 static HRESULT WINAPI DocHostUIHandler_OnFrameWindowActivate(IDocHostUIHandler2 *iface, 898 BOOL fActivate) 899 { 900 DocHost *This = impl_from_IDocHostUIHandler2(iface); 901 FIXME("(%p)->(%x)\n", This, fActivate); 902 return E_NOTIMPL; 903 } 904 905 static HRESULT WINAPI DocHostUIHandler_ResizeBorder(IDocHostUIHandler2 *iface, 906 LPCRECT prcBorder, IOleInPlaceUIWindow *pUIWindow, BOOL fRameWindow) 907 { 908 DocHost *This = impl_from_IDocHostUIHandler2(iface); 909 FIXME("(%p)->(%p %p %X)\n", This, prcBorder, pUIWindow, fRameWindow); 910 return E_NOTIMPL; 911 } 912 913 static HRESULT WINAPI DocHostUIHandler_TranslateAccelerator(IDocHostUIHandler2 *iface, 914 LPMSG lpMsg, const GUID *pguidCmdGroup, DWORD nCmdID) 915 { 916 DocHost *This = impl_from_IDocHostUIHandler2(iface); 917 HRESULT hr = S_FALSE; 918 TRACE("(%p)->(%p %p %d)\n", This, lpMsg, pguidCmdGroup, nCmdID); 919 920 if(This->hostui) 921 hr = IDocHostUIHandler_TranslateAccelerator(This->hostui, lpMsg, pguidCmdGroup, nCmdID); 922 923 return hr; 924 } 925 926 static HRESULT WINAPI DocHostUIHandler_GetOptionKeyPath(IDocHostUIHandler2 *iface, 927 LPOLESTR *pchKey, DWORD dw) 928 { 929 DocHost *This = impl_from_IDocHostUIHandler2(iface); 930 931 TRACE("(%p)->(%p %d)\n", This, pchKey, dw); 932 933 if(This->hostui) 934 return IDocHostUIHandler_GetOptionKeyPath(This->hostui, pchKey, dw); 935 936 return S_OK; 937 } 938 939 static HRESULT WINAPI DocHostUIHandler_GetDropTarget(IDocHostUIHandler2 *iface, 940 IDropTarget *pDropTarget, IDropTarget **ppDropTarget) 941 { 942 DocHost *This = impl_from_IDocHostUIHandler2(iface); 943 FIXME("(%p)\n", This); 944 return E_NOTIMPL; 945 } 946 947 static HRESULT WINAPI DocHostUIHandler_GetExternal(IDocHostUIHandler2 *iface, 948 IDispatch **ppDispatch) 949 { 950 DocHost *This = impl_from_IDocHostUIHandler2(iface); 951 952 TRACE("(%p)->(%p)\n", This, ppDispatch); 953 954 if(This->hostui) 955 return IDocHostUIHandler_GetExternal(This->hostui, ppDispatch); 956 957 if(!This->shell_ui_helper) { 958 HRESULT hres; 959 960 hres = create_shell_ui_helper(&This->shell_ui_helper); 961 if(FAILED(hres)) 962 return hres; 963 } 964 965 *ppDispatch = (IDispatch*)This->shell_ui_helper; 966 IDispatch_AddRef(*ppDispatch); 967 return S_OK; 968 } 969 970 static HRESULT WINAPI DocHostUIHandler_TranslateUrl(IDocHostUIHandler2 *iface, 971 DWORD dwTranslate, OLECHAR *pchURLIn, OLECHAR **ppchURLOut) 972 { 973 DocHost *This = impl_from_IDocHostUIHandler2(iface); 974 975 TRACE("(%p)->(%d %s %p)\n", This, dwTranslate, debugstr_w(pchURLIn), ppchURLOut); 976 977 if(This->hostui) 978 return IDocHostUIHandler_TranslateUrl(This->hostui, dwTranslate, 979 pchURLIn, ppchURLOut); 980 981 return S_FALSE; 982 } 983 984 static HRESULT WINAPI DocHostUIHandler_FilterDataObject(IDocHostUIHandler2 *iface, 985 IDataObject *pDO, IDataObject **ppDORet) 986 { 987 DocHost *This = impl_from_IDocHostUIHandler2(iface); 988 FIXME("(%p)->(%p %p)\n", This, pDO, ppDORet); 989 return E_NOTIMPL; 990 } 991 992 static HRESULT WINAPI DocHostUIHandler_GetOverrideKeyPath(IDocHostUIHandler2 *iface, 993 LPOLESTR *pchKey, DWORD dw) 994 { 995 DocHost *This = impl_from_IDocHostUIHandler2(iface); 996 IDocHostUIHandler2 *handler; 997 HRESULT hres; 998 999 TRACE("(%p)->(%p %d)\n", This, pchKey, dw); 1000 1001 if(!This->hostui) 1002 return S_OK; 1003 1004 hres = IDocHostUIHandler_QueryInterface(This->hostui, &IID_IDocHostUIHandler2, 1005 (void**)&handler); 1006 if(SUCCEEDED(hres)) { 1007 hres = IDocHostUIHandler2_GetOverrideKeyPath(handler, pchKey, dw); 1008 IDocHostUIHandler2_Release(handler); 1009 return hres; 1010 } 1011 1012 return S_OK; 1013 } 1014 1015 static const IDocHostUIHandler2Vtbl DocHostUIHandler2Vtbl = { 1016 DocHostUIHandler_QueryInterface, 1017 DocHostUIHandler_AddRef, 1018 DocHostUIHandler_Release, 1019 DocHostUIHandler_ShowContextMenu, 1020 DocHostUIHandler_GetHostInfo, 1021 DocHostUIHandler_ShowUI, 1022 DocHostUIHandler_HideUI, 1023 DocHostUIHandler_UpdateUI, 1024 DocHostUIHandler_EnableModeless, 1025 DocHostUIHandler_OnDocWindowActivate, 1026 DocHostUIHandler_OnFrameWindowActivate, 1027 DocHostUIHandler_ResizeBorder, 1028 DocHostUIHandler_TranslateAccelerator, 1029 DocHostUIHandler_GetOptionKeyPath, 1030 DocHostUIHandler_GetDropTarget, 1031 DocHostUIHandler_GetExternal, 1032 DocHostUIHandler_TranslateUrl, 1033 DocHostUIHandler_FilterDataObject, 1034 DocHostUIHandler_GetOverrideKeyPath 1035 }; 1036 1037 static inline DocHost *impl_from_IPropertyNotifySink(IPropertyNotifySink *iface) 1038 { 1039 return CONTAINING_RECORD(iface, DocHost, IPropertyNotifySink_iface); 1040 } 1041 1042 static HRESULT WINAPI PropertyNotifySink_QueryInterface(IPropertyNotifySink *iface, 1043 REFIID riid, void **ppv) 1044 { 1045 DocHost *This = impl_from_IPropertyNotifySink(iface); 1046 return IOleClientSite_QueryInterface(&This->IOleClientSite_iface, riid, ppv); 1047 } 1048 1049 static ULONG WINAPI PropertyNotifySink_AddRef(IPropertyNotifySink *iface) 1050 { 1051 DocHost *This = impl_from_IPropertyNotifySink(iface); 1052 return IOleClientSite_AddRef(&This->IOleClientSite_iface); 1053 } 1054 1055 static ULONG WINAPI PropertyNotifySink_Release(IPropertyNotifySink *iface) 1056 { 1057 DocHost *This = impl_from_IPropertyNotifySink(iface); 1058 return IOleClientSite_Release(&This->IOleClientSite_iface); 1059 } 1060 1061 static HRESULT WINAPI PropertyNotifySink_OnChanged(IPropertyNotifySink *iface, DISPID dispID) 1062 { 1063 DocHost *This = impl_from_IPropertyNotifySink(iface); 1064 1065 TRACE("(%p)->(%d)\n", This, dispID); 1066 1067 switch(dispID) { 1068 case DISPID_READYSTATE: { 1069 READYSTATE ready_state; 1070 HRESULT hres; 1071 1072 hres = get_doc_ready_state(This, &ready_state); 1073 if(FAILED(hres)) 1074 return hres; 1075 1076 if(ready_state == READYSTATE_COMPLETE && !This->doc_navigate) 1077 advise_prop_notif(This, FALSE); 1078 1079 update_ready_state(This, ready_state); 1080 break; 1081 } 1082 default: 1083 FIXME("unimplemented dispid %d\n", dispID); 1084 return E_NOTIMPL; 1085 } 1086 1087 return S_OK; 1088 } 1089 1090 static HRESULT WINAPI PropertyNotifySink_OnRequestEdit(IPropertyNotifySink *iface, DISPID dispID) 1091 { 1092 DocHost *This = impl_from_IPropertyNotifySink(iface); 1093 FIXME("(%p)->(%d)\n", This, dispID); 1094 return E_NOTIMPL; 1095 } 1096 1097 static const IPropertyNotifySinkVtbl PropertyNotifySinkVtbl = { 1098 PropertyNotifySink_QueryInterface, 1099 PropertyNotifySink_AddRef, 1100 PropertyNotifySink_Release, 1101 PropertyNotifySink_OnChanged, 1102 PropertyNotifySink_OnRequestEdit 1103 }; 1104 1105 void DocHost_Init(DocHost *This, IWebBrowser2 *wb, const IDocHostContainerVtbl* container) 1106 { 1107 This->IDocHostUIHandler2_iface.lpVtbl = &DocHostUIHandler2Vtbl; 1108 This->IOleCommandTarget_iface.lpVtbl = &OleCommandTargetVtbl; 1109 This->IPropertyNotifySink_iface.lpVtbl = &PropertyNotifySinkVtbl; 1110 1111 This->wb = wb; 1112 This->container_vtbl = container; 1113 1114 This->ready_state = READYSTATE_UNINITIALIZED; 1115 list_init(&This->task_queue); 1116 1117 This->travellog.loading_pos = -1; 1118 1119 DocHost_ClientSite_Init(This); 1120 DocHost_Frame_Init(This); 1121 1122 ConnectionPointContainer_Init(&This->cps, (IUnknown*)wb); 1123 IEHTMLWindow_Init(This); 1124 NewWindowManager_Init(This); 1125 } 1126 1127 void DocHost_Release(DocHost *This) 1128 { 1129 if(This->shell_ui_helper) 1130 IShellUIHelper2_Release(This->shell_ui_helper); 1131 1132 abort_dochost_tasks(This, NULL); 1133 release_dochost_client(This); 1134 DocHost_ClientSite_Release(This); 1135 1136 ConnectionPointContainer_Destroy(&This->cps); 1137 1138 while(This->travellog.length) 1139 free_travellog_entry(This->travellog.log + --This->travellog.length); 1140 heap_free(This->travellog.log); 1141 1142 heap_free(This->url); 1143 } 1144