msvfw32/tests: Add ICDecompressQuery format tests.
[wine] / dlls / shdocvw / dochost.c
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 "wine/debug.h"
20 #include "shdocvw.h"
21 #include "hlink.h"
22 #include "exdispid.h"
23 #include "mshtml.h"
24 #include "initguid.h"
25
26 WINE_DEFAULT_DEBUG_CHANNEL(shdocvw);
27
28 DEFINE_OLEGUID(CGID_DocHostCmdPriv, 0x000214D4L, 0, 0);
29
30 #define DOCHOST_DOCCANNAVIGATE  0
31
32 static ATOM doc_view_atom = 0;
33
34 void push_dochost_task(DocHost *This, task_header_t *task, task_proc_t proc, BOOL send)
35 {
36     task->proc = proc;
37
38     /* FIXME: Don't use lParam */
39     if(send)
40         SendMessageW(This->frame_hwnd, WM_DOCHOSTTASK, 0, (LPARAM)task);
41     else
42         PostMessageW(This->frame_hwnd, WM_DOCHOSTTASK, 0, (LPARAM)task);
43 }
44
45 LRESULT process_dochost_task(DocHost *This, LPARAM lparam)
46 {
47     task_header_t *task = (task_header_t*)lparam;
48
49     task->proc(This, task);
50
51     heap_free(task);
52     return 0;
53 }
54
55 static void notif_complete(DocHost *This, DISPID dispid)
56 {
57     DISPPARAMS dispparams;
58     VARIANTARG params[2];
59     VARIANT url;
60
61     dispparams.cArgs = 2;
62     dispparams.cNamedArgs = 0;
63     dispparams.rgdispidNamedArgs = NULL;
64     dispparams.rgvarg = params;
65
66     V_VT(params) = (VT_BYREF|VT_VARIANT);
67     V_BYREF(params) = &url;
68
69     V_VT(params+1) = VT_DISPATCH;
70     V_DISPATCH(params+1) = This->disp;
71
72     V_VT(&url) = VT_BSTR;
73     V_BSTR(&url) = SysAllocString(This->url);
74
75     TRACE("%d >>>\n", dispid);
76     call_sink(This->cps.wbe2, dispid, &dispparams);
77     TRACE("%d <<<\n", dispid);
78
79     SysFreeString(V_BSTR(&url));
80     This->busy = VARIANT_FALSE;
81 }
82
83 static void object_available(DocHost *This)
84 {
85     IHlinkTarget *hlink;
86     HRESULT hres;
87
88     TRACE("(%p)\n", This);
89
90     if(!This->document) {
91         WARN("document == NULL\n");
92         return;
93     }
94
95     hres = IUnknown_QueryInterface(This->document, &IID_IHlinkTarget, (void**)&hlink);
96     if(FAILED(hres)) {
97         FIXME("Could not get IHlinkTarget interface\n");
98         return;
99     }
100
101     hres = IHlinkTarget_Navigate(hlink, 0, NULL);
102     IHlinkTarget_Release(hlink);
103     if(FAILED(hres))
104         FIXME("Navigate failed\n");
105 }
106
107 static HRESULT get_doc_ready_state(DocHost *This, READYSTATE *ret)
108 {
109     DISPPARAMS dp = {NULL,NULL,0,0};
110     IDispatch *disp;
111     EXCEPINFO ei;
112     VARIANT var;
113     HRESULT hres;
114
115     hres = IUnknown_QueryInterface(This->document, &IID_IDispatch, (void**)&disp);
116     if(FAILED(hres))
117         return hres;
118
119     hres = IDispatch_Invoke(disp, DISPID_READYSTATE, &IID_NULL, LOCALE_SYSTEM_DEFAULT, DISPATCH_PROPERTYGET,
120             &dp, &var, &ei, NULL);
121     IDispatch_Release(disp);
122     if(FAILED(hres)) {
123         WARN("Invoke(DISPID_READYSTATE failed: %08x\n", hres);
124         return hres;
125     }
126
127     if(V_VT(&var) != VT_I4) {
128         WARN("V_VT(var) = %d\n", V_VT(&var));
129         VariantClear(&var);
130         return E_FAIL;
131     }
132
133     *ret = V_I4(&var);
134     return S_OK;
135 }
136
137 static void advise_prop_notif(DocHost *This, BOOL set)
138 {
139     IConnectionPointContainer *cp_container;
140     IConnectionPoint *cp;
141     HRESULT hres;
142
143     hres = IUnknown_QueryInterface(This->document, &IID_IConnectionPointContainer, (void**)&cp_container);
144     if(FAILED(hres))
145         return;
146
147     hres = IConnectionPointContainer_FindConnectionPoint(cp_container, &IID_IPropertyNotifySink, &cp);
148     IConnectionPointContainer_Release(cp_container);
149     if(FAILED(hres))
150         return;
151
152     if(set)
153         hres = IConnectionPoint_Advise(cp, (IUnknown*)PROPNOTIF(This), &This->prop_notif_cookie);
154     else
155         hres = IConnectionPoint_Unadvise(cp, This->prop_notif_cookie);
156     IConnectionPoint_Release(cp);
157
158     if(SUCCEEDED(hres))
159         This->is_prop_notif = set;
160 }
161
162 void set_doc_state(DocHost *This, READYSTATE doc_state)
163 {
164     This->doc_state = doc_state;
165     if(doc_state > This->ready_state)
166         This->ready_state = doc_state;
167 }
168
169 static void update_ready_state(DocHost *This, READYSTATE ready_state)
170 {
171     if(ready_state > READYSTATE_LOADING && This->doc_state <= READYSTATE_LOADING)
172         notif_complete(This, DISPID_NAVIGATECOMPLETE2);
173
174     if(ready_state == READYSTATE_COMPLETE && This->doc_state < READYSTATE_COMPLETE) {
175         set_doc_state(This, READYSTATE_COMPLETE);
176         notif_complete(This, DISPID_DOCUMENTCOMPLETE);
177     }else {
178         set_doc_state(This, ready_state);
179     }
180 }
181
182 typedef struct {
183     task_header_t header;
184     IUnknown *doc;
185     READYSTATE ready_state;
186 } ready_state_task_t;
187
188 static void ready_state_proc(DocHost *This, task_header_t *_task)
189 {
190     ready_state_task_t *task = (ready_state_task_t*)_task;
191
192     if(task->doc == This->document)
193         update_ready_state(This, task->ready_state);
194
195     IUnknown_Release(task->doc);
196 }
197
198 static void push_ready_state_task(DocHost *This, READYSTATE ready_state)
199 {
200     ready_state_task_t *task = heap_alloc(sizeof(ready_state_task_t));
201
202     IUnknown_AddRef(This->document);
203     task->doc = This->document;
204     task->ready_state = ready_state;
205
206     push_dochost_task(This, &task->header, ready_state_proc, FALSE);
207 }
208
209 static void object_available_proc(DocHost *This, task_header_t *task)
210 {
211     object_available(This);
212 }
213
214 HRESULT dochost_object_available(DocHost *This, IUnknown *doc)
215 {
216     READYSTATE ready_state;
217     task_header_t *task;
218     IOleObject *oleobj;
219     HRESULT hres;
220
221     IUnknown_AddRef(doc);
222     This->document = doc;
223
224     hres = IUnknown_QueryInterface(doc, &IID_IOleObject, (void**)&oleobj);
225     if(SUCCEEDED(hres)) {
226         CLSID clsid;
227
228         hres = IOleObject_GetUserClassID(oleobj, &clsid);
229         if(SUCCEEDED(hres))
230             TRACE("Got clsid %s\n",
231                   IsEqualGUID(&clsid, &CLSID_HTMLDocument) ? "CLSID_HTMLDocument" : debugstr_guid(&clsid));
232
233         hres = IOleObject_SetClientSite(oleobj, CLIENTSITE(This));
234         if(FAILED(hres))
235             FIXME("SetClientSite failed: %08x\n", hres);
236
237         IOleObject_Release(oleobj);
238     }else {
239         FIXME("Could not get IOleObject iface: %08x\n", hres);
240     }
241
242     /* FIXME: Call SetAdvise */
243
244     task = heap_alloc(sizeof(*task));
245     push_dochost_task(This, task, object_available_proc, FALSE);
246
247     hres = get_doc_ready_state(This, &ready_state);
248     if(SUCCEEDED(hres)) {
249         if(ready_state == READYSTATE_COMPLETE)
250             push_ready_state_task(This, READYSTATE_COMPLETE);
251         if(ready_state != READYSTATE_COMPLETE || This->doc_navigate)
252             advise_prop_notif(This, TRUE);
253     }
254
255     return S_OK;
256 }
257
258 static LRESULT resize_document(DocHost *This, LONG width, LONG height)
259 {
260     RECT rect = {0, 0, width, height};
261
262     TRACE("(%p)->(%d %d)\n", This, width, height);
263
264     if(This->view)
265         IOleDocumentView_SetRect(This->view, &rect);
266
267     return 0;
268 }
269
270 static LRESULT WINAPI doc_view_proc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam)
271 {
272     DocHost *This;
273
274     static const WCHAR wszTHIS[] = {'T','H','I','S',0};
275
276     if(msg == WM_CREATE) {
277         This = *(DocHost**)lParam;
278         SetPropW(hwnd, wszTHIS, This);
279     }else {
280         This = GetPropW(hwnd, wszTHIS);
281     }
282
283     switch(msg) {
284     case WM_SIZE:
285         return resize_document(This, LOWORD(lParam), HIWORD(lParam));
286     }
287
288     return DefWindowProcW(hwnd, msg, wParam, lParam);
289 }
290
291 void create_doc_view_hwnd(DocHost *This)
292 {
293     RECT rect;
294
295     static const WCHAR wszShell_DocObject_View[] =
296         {'S','h','e','l','l',' ','D','o','c','O','b','j','e','c','t',' ','V','i','e','w',0};
297
298     if(!doc_view_atom) {
299         static WNDCLASSEXW wndclass = {
300             sizeof(wndclass),
301             CS_PARENTDC,
302             doc_view_proc,
303             0, 0 /* native uses 4*/, NULL, NULL, NULL,
304             (HBRUSH)(COLOR_WINDOW + 1), NULL,
305             wszShell_DocObject_View,
306             NULL
307         };
308
309         wndclass.hInstance = shdocvw_hinstance;
310
311         doc_view_atom = RegisterClassExW(&wndclass);
312     }
313
314     GetClientRect(This->frame_hwnd, &rect);
315     adjust_ie_docobj_rect(This->frame_hwnd, &rect);
316     This->hwnd = CreateWindowExW(0, wszShell_DocObject_View,
317          wszShell_DocObject_View,
318          WS_CHILD | WS_VISIBLE | WS_CLIPSIBLINGS | WS_CLIPCHILDREN | WS_TABSTOP,
319          rect.left, rect.top, rect.right, rect.bottom, This->frame_hwnd,
320          NULL, shdocvw_hinstance, This);
321 }
322
323 void deactivate_document(DocHost *This)
324 {
325     IOleInPlaceObjectWindowless *winobj;
326     IOleObject *oleobj = NULL;
327     IHlinkTarget *hlink = NULL;
328     HRESULT hres;
329
330     if(This->doc_navigate) {
331         IUnknown_Release(This->doc_navigate);
332         This->doc_navigate = NULL;
333     }
334
335     if(This->is_prop_notif)
336         advise_prop_notif(This, FALSE);
337
338     if(This->view)
339         IOleDocumentView_UIActivate(This->view, FALSE);
340
341     hres = IUnknown_QueryInterface(This->document, &IID_IOleInPlaceObjectWindowless,
342                                    (void**)&winobj);
343     if(SUCCEEDED(hres)) {
344         IOleInPlaceObjectWindowless_InPlaceDeactivate(winobj);
345         IOleInPlaceObjectWindowless_Release(winobj);
346     }
347
348     if(This->view) {
349         IOleDocumentView_Show(This->view, FALSE);
350         IOleDocumentView_CloseView(This->view, 0);
351         IOleDocumentView_SetInPlaceSite(This->view, NULL);
352         IOleDocumentView_Release(This->view);
353         This->view = NULL;
354     }
355
356     hres = IUnknown_QueryInterface(This->document, &IID_IOleObject, (void**)&oleobj);
357     if(SUCCEEDED(hres))
358         IOleObject_Close(oleobj, OLECLOSE_NOSAVE);
359
360     hres = IUnknown_QueryInterface(This->document, &IID_IHlinkTarget, (void**)&hlink);
361     if(SUCCEEDED(hres)) {
362         IHlinkTarget_SetBrowseContext(hlink, NULL);
363         IHlinkTarget_Release(hlink);
364     }
365
366     if(oleobj) {
367         IOleClientSite *client_site = NULL;
368
369         IOleObject_GetClientSite(oleobj, &client_site);
370         if(client_site) {
371             if(client_site == CLIENTSITE(This))
372                 IOleObject_SetClientSite(oleobj, NULL);
373             IOleClientSite_Release(client_site);
374         }
375
376         IOleObject_Release(oleobj);
377     }
378
379     IUnknown_Release(This->document);
380     This->document = NULL;
381 }
382
383 void release_dochost_client(DocHost *This)
384 {
385     if(This->hwnd) {
386         DestroyWindow(This->hwnd);
387         This->hwnd = NULL;
388     }
389
390     if(This->hostui) {
391         IDocHostUIHandler_Release(This->hostui);
392         This->hostui = NULL;
393     }
394
395     if(This->client_disp) {
396         IDispatch_Release(This->client_disp);
397         This->client_disp = NULL;
398     }
399
400     if(This->frame) {
401         IOleInPlaceFrame_Release(This->frame);
402         This->frame = NULL;
403     }
404 }
405
406 #define OLECMD_THIS(iface) DEFINE_THIS(DocHost, OleCommandTarget, iface)
407
408 static HRESULT WINAPI ClOleCommandTarget_QueryInterface(IOleCommandTarget *iface,
409         REFIID riid, void **ppv)
410 {
411     DocHost *This = OLECMD_THIS(iface);
412     return IOleClientSite_QueryInterface(CLIENTSITE(This), riid, ppv);
413 }
414
415 static ULONG WINAPI ClOleCommandTarget_AddRef(IOleCommandTarget *iface)
416 {
417     DocHost *This = OLECMD_THIS(iface);
418     return IOleClientSite_AddRef(CLIENTSITE(This));
419 }
420
421 static ULONG WINAPI ClOleCommandTarget_Release(IOleCommandTarget *iface)
422 {
423     DocHost *This = OLECMD_THIS(iface);
424     return IOleClientSite_Release(CLIENTSITE(This));
425 }
426
427 static HRESULT WINAPI ClOleCommandTarget_QueryStatus(IOleCommandTarget *iface,
428         const GUID *pguidCmdGroup, ULONG cCmds, OLECMD prgCmds[], OLECMDTEXT *pCmdText)
429 {
430     DocHost *This = OLECMD_THIS(iface);
431     ULONG i= 0;
432     FIXME("(%p)->(%s %u %p %p)\n", This, debugstr_guid(pguidCmdGroup), cCmds, prgCmds,
433           pCmdText);
434     while (prgCmds && (cCmds > i)) {
435         FIXME("command_%u: %u, 0x%x\n", i, prgCmds[i].cmdID, prgCmds[i].cmdf);
436         i++;
437     }
438     return E_NOTIMPL;
439 }
440
441 static HRESULT WINAPI ClOleCommandTarget_Exec(IOleCommandTarget *iface,
442         const GUID *pguidCmdGroup, DWORD nCmdID, DWORD nCmdexecopt, VARIANT *pvaIn,
443         VARIANT *pvaOut)
444 {
445     DocHost *This = OLECMD_THIS(iface);
446
447     TRACE("(%p)->(%s %d %d %p %p)\n", This, debugstr_guid(pguidCmdGroup), nCmdID,
448           nCmdexecopt, debugstr_variant(pvaIn), debugstr_variant(pvaOut));
449
450     if(!pguidCmdGroup) {
451         FIXME("Unimplemented cmdid %d\n", nCmdID);
452         return E_NOTIMPL;
453     }
454
455     if(IsEqualGUID(pguidCmdGroup, &CGID_DocHostCmdPriv)) {
456         switch(nCmdID) {
457         case DOCHOST_DOCCANNAVIGATE:
458             if(!pvaIn || V_VT(pvaIn) != VT_UNKNOWN)
459                 return E_INVALIDARG;
460
461             if(This->doc_navigate)
462                 IUnknown_Release(This->doc_navigate);
463             IUnknown_AddRef(V_UNKNOWN(pvaIn));
464             This->doc_navigate = V_UNKNOWN(pvaIn);
465             return S_OK;
466
467         default:
468             FIXME("unsupported command %d of CGID_DocHostCmdPriv\n", nCmdID);
469             return E_NOTIMPL;
470         }
471     }
472
473     FIXME("Unimplemented group %s\n", debugstr_guid(pguidCmdGroup));
474     return E_NOTIMPL;
475 }
476
477 #undef OLECMD_THIS
478
479 static const IOleCommandTargetVtbl OleCommandTargetVtbl = {
480     ClOleCommandTarget_QueryInterface,
481     ClOleCommandTarget_AddRef,
482     ClOleCommandTarget_Release,
483     ClOleCommandTarget_QueryStatus,
484     ClOleCommandTarget_Exec
485 };
486
487 #define DOCHOSTUI_THIS(iface) DEFINE_THIS(DocHost, DocHostUIHandler, iface)
488
489 static HRESULT WINAPI DocHostUIHandler_QueryInterface(IDocHostUIHandler2 *iface,
490                                                       REFIID riid, void **ppv)
491 {
492     DocHost *This = DOCHOSTUI_THIS(iface);
493     return IOleClientSite_QueryInterface(CLIENTSITE(This), riid, ppv);
494 }
495
496 static ULONG WINAPI DocHostUIHandler_AddRef(IDocHostUIHandler2 *iface)
497 {
498     DocHost *This = DOCHOSTUI_THIS(iface);
499     return IOleClientSite_AddRef(CLIENTSITE(This));
500 }
501
502 static ULONG WINAPI DocHostUIHandler_Release(IDocHostUIHandler2 *iface)
503 {
504     DocHost *This = DOCHOSTUI_THIS(iface);
505     return IOleClientSite_Release(CLIENTSITE(This));
506 }
507
508 static HRESULT WINAPI DocHostUIHandler_ShowContextMenu(IDocHostUIHandler2 *iface,
509          DWORD dwID, POINT *ppt, IUnknown *pcmdtReserved, IDispatch *pdispReserved)
510 {
511     DocHost *This = DOCHOSTUI_THIS(iface);
512     HRESULT hres;
513
514     TRACE("(%p)->(%d %p %p %p)\n", This, dwID, ppt, pcmdtReserved, pdispReserved);
515
516     if(This->hostui) {
517         hres = IDocHostUIHandler_ShowContextMenu(This->hostui, dwID, ppt, pcmdtReserved,
518                                                  pdispReserved);
519         if(hres == S_OK)
520             return S_OK;
521     }
522
523     FIXME("default action not implemented\n");
524     return E_NOTIMPL;
525 }
526
527 static HRESULT WINAPI DocHostUIHandler_GetHostInfo(IDocHostUIHandler2 *iface,
528         DOCHOSTUIINFO *pInfo)
529 {
530     DocHost *This = DOCHOSTUI_THIS(iface);
531     HRESULT hres;
532
533     TRACE("(%p)->(%p)\n", This, pInfo);
534
535     if(This->hostui) {
536         hres = IDocHostUIHandler_GetHostInfo(This->hostui, pInfo);
537         if(SUCCEEDED(hres))
538             return hres;
539     }
540
541     pInfo->dwFlags = DOCHOSTUIFLAG_DISABLE_HELP_MENU | DOCHOSTUIFLAG_OPENNEWWIN
542         | DOCHOSTUIFLAG_URL_ENCODING_ENABLE_UTF8 | DOCHOSTUIFLAG_ENABLE_INPLACE_NAVIGATION
543         | DOCHOSTUIFLAG_IME_ENABLE_RECONVERSION;
544     return S_OK;
545 }
546
547 static HRESULT WINAPI DocHostUIHandler_ShowUI(IDocHostUIHandler2 *iface, DWORD dwID,
548         IOleInPlaceActiveObject *pActiveObject, IOleCommandTarget *pCommandTarget,
549         IOleInPlaceFrame *pFrame, IOleInPlaceUIWindow *pDoc)
550 {
551     DocHost *This = DOCHOSTUI_THIS(iface);
552     FIXME("(%p)->(%d %p %p %p %p)\n", This, dwID, pActiveObject, pCommandTarget,
553           pFrame, pDoc);
554     return E_NOTIMPL;
555 }
556
557 static HRESULT WINAPI DocHostUIHandler_HideUI(IDocHostUIHandler2 *iface)
558 {
559     DocHost *This = DOCHOSTUI_THIS(iface);
560     FIXME("(%p)\n", This);
561     return E_NOTIMPL;
562 }
563
564 static HRESULT WINAPI DocHostUIHandler_UpdateUI(IDocHostUIHandler2 *iface)
565 {
566     DocHost *This = DOCHOSTUI_THIS(iface);
567
568     TRACE("(%p)\n", This);
569
570     if(!This->hostui)
571         return S_FALSE;
572
573     return IDocHostUIHandler_UpdateUI(This->hostui);
574 }
575
576 static HRESULT WINAPI DocHostUIHandler_EnableModeless(IDocHostUIHandler2 *iface,
577                                                       BOOL fEnable)
578 {
579     DocHost *This = DOCHOSTUI_THIS(iface);
580     FIXME("(%p)->(%x)\n", This, fEnable);
581     return E_NOTIMPL;
582 }
583
584 static HRESULT WINAPI DocHostUIHandler_OnDocWindowActivate(IDocHostUIHandler2 *iface,
585                                                            BOOL fActivate)
586 {
587     DocHost *This = DOCHOSTUI_THIS(iface);
588     FIXME("(%p)->(%x)\n", This, fActivate);
589     return E_NOTIMPL;
590 }
591
592 static HRESULT WINAPI DocHostUIHandler_OnFrameWindowActivate(IDocHostUIHandler2 *iface,
593                                                              BOOL fActivate)
594 {
595     DocHost *This = DOCHOSTUI_THIS(iface);
596     FIXME("(%p)->(%x)\n", This, fActivate);
597     return E_NOTIMPL;
598 }
599
600 static HRESULT WINAPI DocHostUIHandler_ResizeBorder(IDocHostUIHandler2 *iface,
601         LPCRECT prcBorder, IOleInPlaceUIWindow *pUIWindow, BOOL fRameWindow)
602 {
603     DocHost *This = DOCHOSTUI_THIS(iface);
604     FIXME("(%p)->(%p %p %X)\n", This, prcBorder, pUIWindow, fRameWindow);
605     return E_NOTIMPL;
606 }
607
608 static HRESULT WINAPI DocHostUIHandler_TranslateAccelerator(IDocHostUIHandler2 *iface,
609         LPMSG lpMsg, const GUID *pguidCmdGroup, DWORD nCmdID)
610 {
611     DocHost *This = DOCHOSTUI_THIS(iface);
612     FIXME("(%p)->(%p %p %d)\n", This, lpMsg, pguidCmdGroup, nCmdID);
613     return E_NOTIMPL;
614 }
615
616 static HRESULT WINAPI DocHostUIHandler_GetOptionKeyPath(IDocHostUIHandler2 *iface,
617         LPOLESTR *pchKey, DWORD dw)
618 {
619     DocHost *This = DOCHOSTUI_THIS(iface);
620
621     TRACE("(%p)->(%p %d)\n", This, pchKey, dw);
622
623     if(This->hostui)
624         return IDocHostUIHandler_GetOptionKeyPath(This->hostui, pchKey, dw);
625
626     return S_OK;
627 }
628
629 static HRESULT WINAPI DocHostUIHandler_GetDropTarget(IDocHostUIHandler2 *iface,
630         IDropTarget *pDropTarget, IDropTarget **ppDropTarget)
631 {
632     DocHost *This = DOCHOSTUI_THIS(iface);
633     FIXME("(%p)\n", This);
634     return E_NOTIMPL;
635 }
636
637 static HRESULT WINAPI DocHostUIHandler_GetExternal(IDocHostUIHandler2 *iface,
638         IDispatch **ppDispatch)
639 {
640     DocHost *This = DOCHOSTUI_THIS(iface);
641
642     TRACE("(%p)->(%p)\n", This, ppDispatch);
643
644     if(This->hostui)
645         return IDocHostUIHandler_GetExternal(This->hostui, ppDispatch);
646
647     FIXME("default action not implemented\n");
648     return E_NOTIMPL;
649 }
650
651 static HRESULT WINAPI DocHostUIHandler_TranslateUrl(IDocHostUIHandler2 *iface,
652         DWORD dwTranslate, OLECHAR *pchURLIn, OLECHAR **ppchURLOut)
653 {
654     DocHost *This = DOCHOSTUI_THIS(iface);
655
656     TRACE("(%p)->(%d %s %p)\n", This, dwTranslate, debugstr_w(pchURLIn), ppchURLOut);
657
658     if(This->hostui)
659         return IDocHostUIHandler_TranslateUrl(This->hostui, dwTranslate,
660                                               pchURLIn, ppchURLOut);
661
662     return S_FALSE;
663 }
664
665 static HRESULT WINAPI DocHostUIHandler_FilterDataObject(IDocHostUIHandler2 *iface,
666         IDataObject *pDO, IDataObject **ppDORet)
667 {
668     DocHost *This = DOCHOSTUI_THIS(iface);
669     FIXME("(%p)->(%p %p)\n", This, pDO, ppDORet);
670     return E_NOTIMPL;
671 }
672
673 static HRESULT WINAPI DocHostUIHandler_GetOverrideKeyPath(IDocHostUIHandler2 *iface,
674         LPOLESTR *pchKey, DWORD dw)
675 {
676     DocHost *This = DOCHOSTUI_THIS(iface);
677     IDocHostUIHandler2 *handler;
678     HRESULT hres;
679
680     TRACE("(%p)->(%p %d)\n", This, pchKey, dw);
681
682     if(!This->hostui)
683         return S_OK;
684
685     hres = IDocHostUIHandler_QueryInterface(This->hostui, &IID_IDocHostUIHandler2,
686                                             (void**)&handler);
687     if(SUCCEEDED(hres)) {
688         hres = IDocHostUIHandler2_GetOverrideKeyPath(handler, pchKey, dw);
689         IDocHostUIHandler2_Release(handler);
690         return hres;
691     }
692
693     return S_OK;
694 }
695
696 #undef DOCHOSTUI_THIS
697
698 static const IDocHostUIHandler2Vtbl DocHostUIHandler2Vtbl = {
699     DocHostUIHandler_QueryInterface,
700     DocHostUIHandler_AddRef,
701     DocHostUIHandler_Release,
702     DocHostUIHandler_ShowContextMenu,
703     DocHostUIHandler_GetHostInfo,
704     DocHostUIHandler_ShowUI,
705     DocHostUIHandler_HideUI,
706     DocHostUIHandler_UpdateUI,
707     DocHostUIHandler_EnableModeless,
708     DocHostUIHandler_OnDocWindowActivate,
709     DocHostUIHandler_OnFrameWindowActivate,
710     DocHostUIHandler_ResizeBorder,
711     DocHostUIHandler_TranslateAccelerator,
712     DocHostUIHandler_GetOptionKeyPath,
713     DocHostUIHandler_GetDropTarget,
714     DocHostUIHandler_GetExternal,
715     DocHostUIHandler_TranslateUrl,
716     DocHostUIHandler_FilterDataObject,
717     DocHostUIHandler_GetOverrideKeyPath
718 };
719
720 #define PROPNOTIF_THIS(iface) DEFINE_THIS(DocHost, IPropertyNotifySink, iface)
721
722 static HRESULT WINAPI PropertyNotifySink_QueryInterface(IPropertyNotifySink *iface,
723         REFIID riid, void **ppv)
724 {
725     DocHost *This = PROPNOTIF_THIS(iface);
726     return IOleClientSite_QueryInterface(CLIENTSITE(This), riid, ppv);
727 }
728
729 static ULONG WINAPI PropertyNotifySink_AddRef(IPropertyNotifySink *iface)
730 {
731     DocHost *This = PROPNOTIF_THIS(iface);
732     return IOleClientSite_AddRef(CLIENTSITE(This));
733 }
734
735 static ULONG WINAPI PropertyNotifySink_Release(IPropertyNotifySink *iface)
736 {
737     DocHost *This = PROPNOTIF_THIS(iface);
738     return IOleClientSite_Release(CLIENTSITE(This));
739 }
740
741 static HRESULT WINAPI PropertyNotifySink_OnChanged(IPropertyNotifySink *iface, DISPID dispID)
742 {
743     DocHost *This = PROPNOTIF_THIS(iface);
744
745     TRACE("(%p)->(%d)\n", This, dispID);
746
747     switch(dispID) {
748     case DISPID_READYSTATE: {
749         READYSTATE ready_state;
750         HRESULT hres;
751
752         hres = get_doc_ready_state(This, &ready_state);
753         if(FAILED(hres))
754             return hres;
755
756         if(ready_state == READYSTATE_COMPLETE && !This->doc_navigate)
757             advise_prop_notif(This, FALSE);
758
759         push_ready_state_task(This, ready_state);
760         break;
761     }
762     default:
763         FIXME("unimplemented dispid %d\n", dispID);
764         return E_NOTIMPL;
765     }
766
767     return S_OK;
768 }
769
770 static HRESULT WINAPI PropertyNotifySink_OnRequestEdit(IPropertyNotifySink *iface, DISPID dispID)
771 {
772     DocHost *This = PROPNOTIF_THIS(iface);
773     FIXME("(%p)->(%d)\n", This, dispID);
774     return E_NOTIMPL;
775 }
776
777 #undef PROPNOTIF_THIS
778
779 static const IPropertyNotifySinkVtbl PropertyNotifySinkVtbl = {
780     PropertyNotifySink_QueryInterface,
781     PropertyNotifySink_AddRef,
782     PropertyNotifySink_Release,
783     PropertyNotifySink_OnChanged,
784     PropertyNotifySink_OnRequestEdit
785 };
786
787 void DocHost_Init(DocHost *This, IDispatch *disp)
788 {
789     This->lpDocHostUIHandlerVtbl = &DocHostUIHandler2Vtbl;
790     This->lpOleCommandTargetVtbl = &OleCommandTargetVtbl;
791     This->lpIPropertyNotifySinkVtbl = &PropertyNotifySinkVtbl;
792
793     This->disp = disp;
794
795     This->client_disp = NULL;
796
797     This->document = NULL;
798     This->hostui = NULL;
799     This->frame = NULL;
800
801     This->hwnd = NULL;
802     This->frame_hwnd = NULL;
803     This->url = NULL;
804
805     This->silent = VARIANT_FALSE;
806     This->offline = VARIANT_FALSE;
807
808     This->ready_state = READYSTATE_UNINITIALIZED;
809     This->is_prop_notif = FALSE;
810
811     DocHost_ClientSite_Init(This);
812     DocHost_Frame_Init(This);
813
814     ConnectionPointContainer_Init(&This->cps, (IUnknown*)disp);
815 }
816
817 void DocHost_Release(DocHost *This)
818 {
819     release_dochost_client(This);
820     DocHost_ClientSite_Release(This);
821
822     ConnectionPointContainer_Destroy(&This->cps);
823
824     heap_free(This->url);
825 }