xref: /reactos/dll/win32/ieframe/dochost.c (revision 7eead935)
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