shdocvw: Added DocHost::Exec(OLECMDID_UPDATECOMMANDS) implementation.
[wine] / dlls / shdocvw / webbrowser.c
1 /*
2  * Implementation of IWebBrowser interface for WebBrowser control
3  *
4  * Copyright 2001 John R. Sheets (for CodeWeavers)
5  * Copyright 2005 Jacek Caban
6  *
7  * This library is free software; you can redistribute it and/or
8  * modify it under the terms of the GNU Lesser General Public
9  * License as published by the Free Software Foundation; either
10  * version 2.1 of the License, or (at your option) any later version.
11  *
12  * This library is distributed in the hope that it will be useful,
13  * but WITHOUT ANY WARRANTY; without even the implied warranty of
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
15  * Lesser General Public License for more details.
16  *
17  * You should have received a copy of the GNU Lesser General Public
18  * License along with this library; if not, write to the Free Software
19  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
20  */
21
22 #include "wine/debug.h"
23 #include "shdocvw.h"
24 #include "exdispid.h"
25 #include "mshtml.h"
26
27 WINE_DEFAULT_DEBUG_CHANNEL(shdocvw);
28
29 /**********************************************************************
30  * Implement the IWebBrowser interface
31  */
32
33 #define WEBBROWSER_THIS(iface) DEFINE_THIS(WebBrowser, WebBrowser2, iface)
34
35 static HRESULT WINAPI WebBrowser_QueryInterface(IWebBrowser2 *iface, REFIID riid, LPVOID *ppv)
36 {
37     WebBrowser *This = WEBBROWSER_THIS(iface);
38
39     if (ppv == NULL)
40         return E_POINTER;
41     *ppv = NULL;
42
43     if(IsEqualGUID(&IID_IUnknown, riid)) {
44         TRACE("(%p)->(IID_IUnknown %p)\n", This, ppv);
45         *ppv = WEBBROWSER(This);
46     }else if(IsEqualGUID(&IID_IDispatch, riid)) {
47         TRACE("(%p)->(IID_IDispatch %p)\n", This, ppv);
48         *ppv = WEBBROWSER(This);
49     }else if(IsEqualGUID(&IID_IWebBrowser, riid)) {
50         TRACE("(%p)->(IID_IWebBrowser %p)\n", This, ppv);
51         *ppv = WEBBROWSER(This);
52     }else if(IsEqualGUID(&IID_IWebBrowserApp, riid)) {
53         TRACE("(%p)->(IID_IWebBrowserApp %p)\n", This, ppv);
54         *ppv = WEBBROWSER(This);
55     }else if(IsEqualGUID(&IID_IWebBrowser2, riid)) {
56         TRACE("(%p)->(IID_IWebBrowser2 %p)\n", This, ppv);
57         *ppv = WEBBROWSER(This);
58     }else if(IsEqualGUID(&IID_IOleObject, riid)) {
59         TRACE("(%p)->(IID_IOleObject %p)\n", This, ppv);
60         *ppv = OLEOBJ(This);
61     }else if(IsEqualGUID(&IID_IOleWindow, riid)) {
62         TRACE("(%p)->(IID_IOleWindow %p)\n", This, ppv);
63         *ppv = INPLACEOBJ(This);
64     }else if(IsEqualGUID (&IID_IOleInPlaceObject, riid)) {
65         TRACE("(%p)->(IID_IOleInPlaceObject %p)\n", This, ppv);
66         *ppv = INPLACEOBJ(This);
67     }else if(IsEqualGUID(&IID_IOleControl, riid)) {
68         TRACE("(%p)->(IID_IOleControl %p)\n", This, ppv);
69         *ppv = CONTROL(This);
70     }else if(IsEqualGUID(&IID_IPersist, riid)) {
71         TRACE("(%p)->(IID_IPersist %p)\n", This, ppv);
72         *ppv = PERSTORAGE(This);
73     }else if(IsEqualGUID(&IID_IPersistStorage, riid)) {
74         TRACE("(%p)->(IID_IPersistStorage %p)\n", This, ppv);
75         *ppv = PERSTORAGE(This);
76     }else if(IsEqualGUID(&IID_IPersistMemory, riid)) {
77         TRACE("(%p)->(IID_IPersistStorage %p)\n", This, ppv);
78         *ppv = PERMEMORY(This);
79     }else if(IsEqualGUID (&IID_IPersistStreamInit, riid)) {
80         TRACE("(%p)->(IID_IPersistStreamInit %p)\n", This, ppv);
81         *ppv = PERSTRINIT(This);
82     }else if(IsEqualGUID(&IID_IProvideClassInfo, riid)) {
83         TRACE("(%p)->(IID_IProvideClassInfo %p)\n", This, ppv);
84         *ppv = CLASSINFO(This);
85     }else if(IsEqualGUID(&IID_IProvideClassInfo2, riid)) {
86         TRACE("(%p)->(IID_IProvideClassInfo2 %p)\n", This, ppv);
87         *ppv = CLASSINFO(This);
88     }else if(IsEqualGUID(&IID_IConnectionPointContainer, riid)) {
89         TRACE("(%p)->(IID_IConnectionPointContainer %p)\n", This, ppv);
90         *ppv = CONPTCONT(&This->doc_host.cps);
91     }else if(IsEqualGUID(&IID_IViewObject, riid)) {
92         TRACE("(%p)->(IID_IViewObject %p)\n", This, ppv);
93         *ppv = VIEWOBJ(This);
94     }else if(IsEqualGUID(&IID_IViewObject2, riid)) {
95         TRACE("(%p)->(IID_IViewObject2 %p)\n", This, ppv);
96         *ppv = VIEWOBJ2(This);
97     }else if(IsEqualGUID(&IID_IOleInPlaceActiveObject, riid)) {
98         TRACE("(%p)->(IID_IOleInPlaceActiveObject %p)\n", This, ppv);
99         *ppv = ACTIVEOBJ(This);
100     }else if(IsEqualGUID(&IID_IOleCommandTarget, riid)) {
101         TRACE("(%p)->(IID_IOleCommandTarget %p)\n", This, ppv);
102         *ppv = OLECMD(This);
103     }else if(IsEqualGUID(&IID_IServiceProvider, riid)) {
104         *ppv = SERVPROV(This);
105         TRACE("(%p)->(IID_IServiceProvider %p)\n", This, ppv);
106     }else if(IsEqualGUID(&IID_IDataObject, riid)) {
107         *ppv = DATAOBJECT(This);
108         TRACE("(%p)->(IID_IDataObject %p)\n", This, ppv);
109     }else if(IsEqualGUID(&IID_IQuickActivate, riid)) {
110         TRACE("(%p)->(IID_IQuickActivate %p) returning NULL\n", This, ppv);
111         return E_NOINTERFACE;
112     }else if(IsEqualGUID(&IID_IRunnableObject, riid)) {
113         TRACE("(%p)->(IID_IRunnableObject %p) returning NULL\n", This, ppv);
114         return E_NOINTERFACE;
115     }else if(IsEqualGUID(&IID_IPerPropertyBrowsing, riid)) {
116         TRACE("(%p)->(IID_IPerPropertyBrowsing %p) returning NULL\n", This, ppv);
117         return E_NOINTERFACE;
118     }else if(IsEqualGUID(&IID_IOleCache, riid)) {
119         TRACE("(%p)->(IID_IOleCache %p) returning NULL\n", This, ppv);
120         return E_NOINTERFACE;
121     }else if(IsEqualGUID(&IID_IOleInPlaceSite, riid)) {
122         TRACE("(%p)->(IID_IOleInPlaceSite %p) returning NULL\n", This, ppv);
123         return E_NOINTERFACE;
124     }else if(IsEqualGUID(&IID_IObjectWithSite, riid)) {
125         TRACE("(%p)->(IID_IObjectWithSite %p) returning NULL\n", This, ppv);
126         return E_NOINTERFACE;
127     }else if(IsEqualGUID(&IID_IViewObjectEx, riid)) {
128         TRACE("(%p)->(IID_IViewObjectEx %p) returning NULL\n", This, ppv);
129         return E_NOINTERFACE;
130     }else if(IsEqualGUID(&IID_IOleLink, riid)) {
131         TRACE("(%p)->(IID_IOleLink %p) returning NULL\n", This, ppv);
132         return E_NOINTERFACE;
133     }else if(IsEqualGUID(&IID_IMarshal, riid)) {
134         TRACE("(%p)->(IID_IMarshal %p) returning NULL\n", This, ppv);
135         return E_NOINTERFACE;
136     }else if(IsEqualGUID(&IID_IStdMarshalInfo, riid)) {
137         TRACE("(%p)->(IID_IStdMarshalInfo %p) returning NULL\n", This, ppv);
138         return E_NOINTERFACE;
139     }else if(HlinkFrame_QI(&This->hlink_frame, riid, ppv)) {
140         return S_OK;
141     }
142
143     if(*ppv) {
144         IUnknown_AddRef((IUnknown*)*ppv);
145         return S_OK;
146     }
147
148     FIXME("(%p)->(%s %p) interface not supported\n", This, debugstr_guid(riid), ppv);
149     return E_NOINTERFACE;
150 }
151
152 static ULONG WINAPI WebBrowser_AddRef(IWebBrowser2 *iface)
153 {
154     WebBrowser *This = WEBBROWSER_THIS(iface);
155     LONG ref = InterlockedIncrement(&This->ref);
156     TRACE("(%p) ref=%d\n", This, ref);
157     return ref;
158 }
159
160 static ULONG WINAPI WebBrowser_Release(IWebBrowser2 *iface)
161 {
162     WebBrowser *This = WEBBROWSER_THIS(iface);
163     LONG ref = InterlockedDecrement(&This->ref);
164
165     TRACE("(%p) ref=%d\n", This, ref);
166
167     if(!ref) {
168         if(This->doc_host.document)
169             IUnknown_Release(This->doc_host.document);
170
171         DocHost_Release(&This->doc_host);
172
173         WebBrowser_OleObject_Destroy(This);
174
175         heap_free(This);
176         SHDOCVW_UnlockModule();
177     }
178
179     return ref;
180 }
181
182 /* IDispatch methods */
183 static HRESULT WINAPI WebBrowser_GetTypeInfoCount(IWebBrowser2 *iface, UINT *pctinfo)
184 {
185     WebBrowser *This = WEBBROWSER_THIS(iface);
186
187     TRACE("(%p)->(%p)\n", This, pctinfo);
188
189     *pctinfo = 1;
190     return S_OK;
191 }
192
193 static HRESULT WINAPI WebBrowser_GetTypeInfo(IWebBrowser2 *iface, UINT iTInfo, LCID lcid,
194                                      LPTYPEINFO *ppTInfo)
195 {
196     WebBrowser *This = WEBBROWSER_THIS(iface);
197     ITypeInfo *typeinfo;
198     HRESULT hres;
199
200     TRACE("(%p)->(%d %d %p)\n", This, iTInfo, lcid, ppTInfo);
201
202     hres = get_typeinfo(&typeinfo);
203     if(FAILED(hres))
204         return hres;
205
206     ITypeInfo_AddRef(typeinfo);
207     *ppTInfo = typeinfo;
208     return S_OK;
209 }
210
211 static HRESULT WINAPI WebBrowser_GetIDsOfNames(IWebBrowser2 *iface, REFIID riid,
212                                        LPOLESTR *rgszNames, UINT cNames,
213                                        LCID lcid, DISPID *rgDispId)
214 {
215     WebBrowser *This = WEBBROWSER_THIS(iface);
216     ITypeInfo *typeinfo;
217     HRESULT hres;
218
219     TRACE("(%p)->(%s %p %d %d %p)\n", This, debugstr_guid(riid), rgszNames, cNames,
220           lcid, rgDispId);
221
222     hres = get_typeinfo(&typeinfo);
223     if(FAILED(hres))
224         return hres;
225
226     return ITypeInfo_GetIDsOfNames(typeinfo, rgszNames, cNames, rgDispId);
227 }
228
229 static HRESULT WINAPI WebBrowser_Invoke(IWebBrowser2 *iface, DISPID dispIdMember,
230                                 REFIID riid, LCID lcid, WORD wFlags,
231                                 DISPPARAMS *pDispParams, VARIANT *pVarResult,
232                                 EXCEPINFO *pExepInfo, UINT *puArgErr)
233 {
234     WebBrowser *This = WEBBROWSER_THIS(iface);
235     ITypeInfo *typeinfo;
236     HRESULT hres;
237
238     TRACE("(%p)->(%d %s %d %08x %p %p %p %p)\n", This, dispIdMember, debugstr_guid(riid),
239             lcid, wFlags, pDispParams, pVarResult, pExepInfo, puArgErr);
240
241     hres = get_typeinfo(&typeinfo);
242     if(FAILED(hres))
243         return hres;
244
245     return ITypeInfo_Invoke(typeinfo, WEBBROWSER2(This), dispIdMember, wFlags, pDispParams,
246                             pVarResult, pExepInfo, puArgErr);
247 }
248
249 /* IWebBrowser methods */
250 static HRESULT WINAPI WebBrowser_GoBack(IWebBrowser2 *iface)
251 {
252     WebBrowser *This = WEBBROWSER_THIS(iface);
253     FIXME("(%p)\n", This);
254     return E_NOTIMPL;
255 }
256
257 static HRESULT WINAPI WebBrowser_GoForward(IWebBrowser2 *iface)
258 {
259     WebBrowser *This = WEBBROWSER_THIS(iface);
260     FIXME("(%p)\n", This);
261     return E_NOTIMPL;
262 }
263
264 static HRESULT WINAPI WebBrowser_GoHome(IWebBrowser2 *iface)
265 {
266     WebBrowser *This = WEBBROWSER_THIS(iface);
267     TRACE("(%p)\n", This);
268     return go_home(&This->doc_host);
269 }
270
271 static HRESULT WINAPI WebBrowser_GoSearch(IWebBrowser2 *iface)
272 {
273     WebBrowser *This = WEBBROWSER_THIS(iface);
274     FIXME("(%p)\n", This);
275     return E_NOTIMPL;
276 }
277
278 static HRESULT WINAPI WebBrowser_Navigate(IWebBrowser2 *iface, BSTR szUrl,
279                                   VARIANT *Flags, VARIANT *TargetFrameName,
280                                   VARIANT *PostData, VARIANT *Headers)
281 {
282     WebBrowser *This = WEBBROWSER_THIS(iface);
283
284     TRACE("(%p)->(%s %s %s %s %s)\n", This, debugstr_w(szUrl), debugstr_variant(Flags),
285           debugstr_variant(TargetFrameName), debugstr_variant(PostData),
286           debugstr_variant(Headers));
287
288     return navigate_url(&This->doc_host, szUrl, Flags, TargetFrameName, PostData, Headers);
289 }
290
291 static HRESULT WINAPI WebBrowser_Refresh(IWebBrowser2 *iface)
292 {
293     WebBrowser *This = WEBBROWSER_THIS(iface);
294     FIXME("(%p)\n", This);
295     return E_NOTIMPL;
296 }
297
298 static HRESULT WINAPI WebBrowser_Refresh2(IWebBrowser2 *iface, VARIANT *Level)
299 {
300     WebBrowser *This = WEBBROWSER_THIS(iface);
301     FIXME("(%p)->(%s)\n", This, debugstr_variant(Level));
302     return E_NOTIMPL;
303 }
304
305 static HRESULT WINAPI WebBrowser_Stop(IWebBrowser2 *iface)
306 {
307     WebBrowser *This = WEBBROWSER_THIS(iface);
308     FIXME("(%p)\n", This);
309     return S_OK;
310 }
311
312 static HRESULT WINAPI WebBrowser_get_Application(IWebBrowser2 *iface, IDispatch **ppDisp)
313 {
314     WebBrowser *This = WEBBROWSER_THIS(iface);
315
316     TRACE("(%p)->(%p)\n", This, ppDisp);
317
318     if(!ppDisp)
319         return E_POINTER;
320
321     *ppDisp = (IDispatch*)WEBBROWSER2(This);
322     IDispatch_AddRef(*ppDisp);
323     return S_OK;
324 }
325
326 static HRESULT WINAPI WebBrowser_get_Parent(IWebBrowser2 *iface, IDispatch **ppDisp)
327 {
328     WebBrowser *This = WEBBROWSER_THIS(iface);
329     FIXME("(%p)->(%p)\n", This, ppDisp);
330     return E_NOTIMPL;
331 }
332
333 static HRESULT WINAPI WebBrowser_get_Container(IWebBrowser2 *iface, IDispatch **ppDisp)
334 {
335     WebBrowser *This = WEBBROWSER_THIS(iface);
336     FIXME("(%p)->(%p)\n", This, ppDisp);
337     return E_NOTIMPL;
338 }
339
340 static HRESULT WINAPI WebBrowser_get_Document(IWebBrowser2 *iface, IDispatch **ppDisp)
341 {
342     WebBrowser *This = WEBBROWSER_THIS(iface);
343     IDispatch *disp = NULL;
344
345     TRACE("(%p)->(%p)\n", This, ppDisp);
346
347     if(This->doc_host.document) {
348         HRESULT hres;
349
350         hres = IUnknown_QueryInterface(This->doc_host.document, &IID_IDispatch, (void**)&disp);
351         if(SUCCEEDED(hres)) {
352             IDispatch *html_doc;
353
354             /* Some broken apps cast returned IDispatch to IHTMLDocument2
355              * without QueryInterface call */
356             hres = IDispatch_QueryInterface(disp, &IID_IHTMLDocument2, (void**)&html_doc);
357             if(SUCCEEDED(hres)) {
358                 IDispatch_Release(disp);
359                 disp = html_doc;
360             }
361         }
362     }
363
364     *ppDisp = disp;
365     return S_OK;
366 }
367
368 static HRESULT WINAPI WebBrowser_get_TopLevelContainer(IWebBrowser2 *iface, VARIANT_BOOL *pBool)
369 {
370     WebBrowser *This = WEBBROWSER_THIS(iface);
371     FIXME("(%p)->(%p)\n", This, pBool);
372     return E_NOTIMPL;
373 }
374
375 static HRESULT WINAPI WebBrowser_get_Type(IWebBrowser2 *iface, BSTR *Type)
376 {
377     WebBrowser *This = WEBBROWSER_THIS(iface);
378     FIXME("(%p)->(%p)\n", This, Type);
379     return E_NOTIMPL;
380 }
381
382 static HRESULT WINAPI WebBrowser_get_Left(IWebBrowser2 *iface, LONG *pl)
383 {
384     WebBrowser *This = WEBBROWSER_THIS(iface);
385
386     TRACE("(%p)->(%p)\n", This, pl);
387
388     *pl = This->pos_rect.left;
389     return S_OK;
390 }
391
392 static HRESULT WINAPI WebBrowser_put_Left(IWebBrowser2 *iface, LONG Left)
393 {
394     WebBrowser *This = WEBBROWSER_THIS(iface);
395     RECT rect;
396
397     TRACE("(%p)->(%d)\n", This, Left);
398
399     if(!This->inplace)
400         return E_UNEXPECTED;
401
402     rect = This->pos_rect;
403     rect.left = Left;
404
405     /* We don't really change the window position here.
406      * We just notify the embedder that he should do so. */
407     return IOleInPlaceSite_OnPosRectChange(This->inplace, &rect);
408 }
409
410 static HRESULT WINAPI WebBrowser_get_Top(IWebBrowser2 *iface, LONG *pl)
411 {
412     WebBrowser *This = WEBBROWSER_THIS(iface);
413
414     TRACE("(%p)->(%p)\n", This, pl);
415
416     *pl = This->pos_rect.top;
417     return S_OK;
418 }
419
420 static HRESULT WINAPI WebBrowser_put_Top(IWebBrowser2 *iface, LONG Top)
421 {
422     WebBrowser *This = WEBBROWSER_THIS(iface);
423     RECT rect;
424
425     TRACE("(%p)->(%d)\n", This, Top);
426
427     if(!This->inplace)
428         return E_UNEXPECTED;
429
430     rect = This->pos_rect;
431     rect.top = Top;
432
433     /* We don't really change the window position here.
434      * We just notify the embedder that he should do so. */
435     return IOleInPlaceSite_OnPosRectChange(This->inplace, &rect);
436 }
437
438 static HRESULT WINAPI WebBrowser_get_Width(IWebBrowser2 *iface, LONG *pl)
439 {
440     WebBrowser *This = WEBBROWSER_THIS(iface);
441
442     TRACE("(%p)->(%p)\n", This, pl);
443
444     *pl = This->pos_rect.right - This->pos_rect.left;
445     return S_OK;
446 }
447
448 static HRESULT WINAPI WebBrowser_put_Width(IWebBrowser2 *iface, LONG Width)
449 {
450     WebBrowser *This = WEBBROWSER_THIS(iface);
451     RECT rect;
452
453     TRACE("(%p)->(%d)\n", This, Width);
454
455     if(!This->inplace)
456         return E_UNEXPECTED;
457
458     rect = This->pos_rect;
459     rect.right = rect.left+Width;
460
461     /* We don't really change the window size here.
462      * We just notify the embedder that he should do so. */
463    return IOleInPlaceSite_OnPosRectChange(This->inplace, &rect);
464 }
465
466 static HRESULT WINAPI WebBrowser_get_Height(IWebBrowser2 *iface, LONG *pl)
467 {
468     WebBrowser *This = WEBBROWSER_THIS(iface);
469
470     TRACE("(%p)->(%p)\n", This, pl);
471
472     *pl = This->pos_rect.bottom - This->pos_rect.top;
473     return S_OK;
474 }
475
476 static HRESULT WINAPI WebBrowser_put_Height(IWebBrowser2 *iface, LONG Height)
477 {
478     WebBrowser *This = WEBBROWSER_THIS(iface);
479     RECT rect;
480
481     TRACE("(%p)->(%d)\n", This, Height);
482
483     if(!This->inplace)
484         return E_UNEXPECTED;
485
486     rect = This->pos_rect;
487     rect.bottom = rect.top+Height;
488
489     /* We don't really change the window size here.
490      * We just notify the embedder that he should do so. */
491     return IOleInPlaceSite_OnPosRectChange(This->inplace, &rect);
492 }
493
494 static HRESULT WINAPI WebBrowser_get_LocationName(IWebBrowser2 *iface, BSTR *LocationName)
495 {
496     WebBrowser *This = WEBBROWSER_THIS(iface);
497     FIXME("(%p)->(%p)\n", This, LocationName);
498     return E_NOTIMPL;
499 }
500
501 static HRESULT WINAPI WebBrowser_get_LocationURL(IWebBrowser2 *iface, BSTR *LocationURL)
502 {
503     WebBrowser *This = WEBBROWSER_THIS(iface);
504
505     FIXME("(%p)->(%p)\n", This, LocationURL);
506
507     if(!This->doc_host.url) {
508         static const WCHAR null_char = 0;
509         *LocationURL = SysAllocString(&null_char);
510         return S_FALSE;
511     }
512
513     *LocationURL = SysAllocString(This->doc_host.url);
514     return S_OK;
515 }
516
517 static HRESULT WINAPI WebBrowser_get_Busy(IWebBrowser2 *iface, VARIANT_BOOL *pBool)
518 {
519     WebBrowser *This = WEBBROWSER_THIS(iface);
520
521     TRACE("(%p)->(%p)\n", This, pBool);
522
523     *pBool = This->doc_host.busy;
524     return S_OK;
525 }
526
527 static HRESULT WINAPI WebBrowser_Quit(IWebBrowser2 *iface)
528 {
529     WebBrowser *This = WEBBROWSER_THIS(iface);
530
531     TRACE("(%p)\n", This);
532
533     /* It's a InternetExplorer specific method, we have nothing to do here. */
534     return E_FAIL;
535 }
536
537 static HRESULT WINAPI WebBrowser_ClientToWindow(IWebBrowser2 *iface, int *pcx, int *pcy)
538 {
539     WebBrowser *This = WEBBROWSER_THIS(iface);
540     FIXME("(%p)->(%p %p)\n", This, pcx, pcy);
541     return E_NOTIMPL;
542 }
543
544 static HRESULT WINAPI WebBrowser_PutProperty(IWebBrowser2 *iface, BSTR szProperty, VARIANT vtValue)
545 {
546     WebBrowser *This = WEBBROWSER_THIS(iface);
547     FIXME("(%p)->(%s %s)\n", This, debugstr_w(szProperty), debugstr_variant(&vtValue));
548     return E_NOTIMPL;
549 }
550
551 static HRESULT WINAPI WebBrowser_GetProperty(IWebBrowser2 *iface, BSTR szProperty, VARIANT *pvtValue)
552 {
553     WebBrowser *This = WEBBROWSER_THIS(iface);
554     FIXME("(%p)->(%s %s)\n", This, debugstr_w(szProperty), debugstr_variant(pvtValue));
555     return E_NOTIMPL;
556 }
557
558 static HRESULT WINAPI WebBrowser_get_Name(IWebBrowser2 *iface, BSTR *Name)
559 {
560     static const WCHAR sName[] = {'M','i','c','r','o','s','o','f','t',' ','W','e','b',
561                                   ' ','B','r','o','w','s','e','r',' ','C','o','n','t','r','o','l',0};
562     WebBrowser *This = WEBBROWSER_THIS(iface);
563
564     TRACE("(%p)->(%p)\n", This, Name);
565
566     *Name = SysAllocString(sName);
567
568     return S_OK;
569 }
570
571 static HRESULT WINAPI WebBrowser_get_HWND(IWebBrowser2 *iface, LONG *pHWND)
572 {
573     WebBrowser *This = WEBBROWSER_THIS(iface);
574
575     TRACE("(%p)->(%p)\n", This, pHWND);
576
577     /* WebBrowser control never has a frame window (in opposition to InternetExplorer) */
578     *pHWND = 0;
579     return E_FAIL;
580 }
581
582 static HRESULT WINAPI WebBrowser_get_FullName(IWebBrowser2 *iface, BSTR *FullName)
583 {
584     WebBrowser *This = WEBBROWSER_THIS(iface);
585     FIXME("(%p)->(%p)\n", This, FullName);
586     return E_NOTIMPL;
587 }
588
589 static HRESULT WINAPI WebBrowser_get_Path(IWebBrowser2 *iface, BSTR *Path)
590 {
591     WebBrowser *This = WEBBROWSER_THIS(iface);
592     FIXME("(%p)->(%p)\n", This, Path);
593     return E_NOTIMPL;
594 }
595
596 static HRESULT WINAPI WebBrowser_get_Visible(IWebBrowser2 *iface, VARIANT_BOOL *pBool)
597 {
598     WebBrowser *This = WEBBROWSER_THIS(iface);
599
600     TRACE("(%p)->(%p)\n", This, pBool);
601
602     *pBool = This->visible;
603     return S_OK;
604 }
605
606 static HRESULT WINAPI WebBrowser_put_Visible(IWebBrowser2 *iface, VARIANT_BOOL Value)
607 {
608     WebBrowser *This = WEBBROWSER_THIS(iface);
609     VARIANTARG arg;
610     DISPPARAMS dispparams = {&arg, NULL, 1, 0};
611
612     TRACE("(%p)->(%x)\n", This, Value);
613
614     This->visible = Value;
615
616     V_VT(&arg) = VT_BOOL;
617     V_BOOL(&arg) = Value;
618     call_sink(This->doc_host.cps.wbe2, DISPID_ONVISIBLE, &dispparams);
619
620     return S_OK;
621 }
622
623 static HRESULT WINAPI WebBrowser_get_StatusBar(IWebBrowser2 *iface, VARIANT_BOOL *pBool)
624 {
625     WebBrowser *This = WEBBROWSER_THIS(iface);
626
627     TRACE("(%p)->(%p)\n", This, pBool);
628
629     *pBool = This->status_bar;
630     return S_OK;
631 }
632
633 static HRESULT WINAPI WebBrowser_put_StatusBar(IWebBrowser2 *iface, VARIANT_BOOL Value)
634 {
635     WebBrowser *This = WEBBROWSER_THIS(iface);
636     VARIANTARG arg;
637     DISPPARAMS dispparams = {&arg, NULL, 1, 0};
638
639     TRACE("(%p)->(%x)\n", This, Value);
640
641     This->status_bar = Value ? VARIANT_TRUE : VARIANT_FALSE;
642
643     /* In opposition to InternetExplorer, all we should do here is
644      * inform the embedder about the status bar change. */
645
646     V_VT(&arg) = VT_BOOL;
647     V_BOOL(&arg) = Value;
648     call_sink(This->doc_host.cps.wbe2, DISPID_ONSTATUSBAR, &dispparams);
649
650     return S_OK;
651 }
652
653 static HRESULT WINAPI WebBrowser_get_StatusText(IWebBrowser2 *iface, BSTR *StatusText)
654 {
655     WebBrowser *This = WEBBROWSER_THIS(iface);
656     FIXME("(%p)->(%p)\n", This, StatusText);
657     return E_NOTIMPL;
658 }
659
660 static HRESULT WINAPI WebBrowser_put_StatusText(IWebBrowser2 *iface, BSTR StatusText)
661 {
662     WebBrowser *This = WEBBROWSER_THIS(iface);
663     FIXME("(%p)->(%s)\n", This, debugstr_w(StatusText));
664     return E_NOTIMPL;
665 }
666
667 static HRESULT WINAPI WebBrowser_get_ToolBar(IWebBrowser2 *iface, int *Value)
668 {
669     WebBrowser *This = WEBBROWSER_THIS(iface);
670
671     TRACE("(%p)->(%p)\n", This, Value);
672
673     *Value = This->tool_bar;
674     return S_OK;
675 }
676
677 static HRESULT WINAPI WebBrowser_put_ToolBar(IWebBrowser2 *iface, int Value)
678 {
679     WebBrowser *This = WEBBROWSER_THIS(iface);
680     VARIANTARG arg;
681     DISPPARAMS dispparams = {&arg, NULL, 1, 0};
682
683     TRACE("(%p)->(%x)\n", This, Value);
684
685     This->tool_bar = Value ? VARIANT_TRUE : VARIANT_FALSE;
686
687     /* In opposition to InternetExplorer, all we should do here is
688      * inform the embedder about the tool bar change. */
689
690     V_VT(&arg) = VT_BOOL;
691     V_BOOL(&arg) = This->tool_bar;
692     call_sink(This->doc_host.cps.wbe2, DISPID_ONTOOLBAR, &dispparams);
693
694     return S_OK;
695 }
696
697 static HRESULT WINAPI WebBrowser_get_MenuBar(IWebBrowser2 *iface, VARIANT_BOOL *Value)
698 {
699     WebBrowser *This = WEBBROWSER_THIS(iface);
700
701     TRACE("(%p)->(%p)\n", This, Value);
702
703     *Value = This->menu_bar;
704     return S_OK;
705 }
706
707 static HRESULT WINAPI WebBrowser_put_MenuBar(IWebBrowser2 *iface, VARIANT_BOOL Value)
708 {
709     WebBrowser *This = WEBBROWSER_THIS(iface);
710     VARIANTARG arg;
711     DISPPARAMS dispparams = {&arg, NULL, 1, 0};
712
713     TRACE("(%p)->(%x)\n", This, Value);
714
715     This->menu_bar = Value ? VARIANT_TRUE : VARIANT_FALSE;
716
717     /* In opposition to InternetExplorer, all we should do here is
718      * inform the embedder about the menu bar change. */
719
720     V_VT(&arg) = VT_BOOL;
721     V_BOOL(&arg) = Value;
722     call_sink(This->doc_host.cps.wbe2, DISPID_ONMENUBAR, &dispparams);
723
724     return S_OK;
725 }
726
727 static HRESULT WINAPI WebBrowser_get_FullScreen(IWebBrowser2 *iface, VARIANT_BOOL *pbFullScreen)
728 {
729     WebBrowser *This = WEBBROWSER_THIS(iface);
730
731     TRACE("(%p)->(%p)\n", This, pbFullScreen);
732
733     *pbFullScreen = This->full_screen;
734     return S_OK;
735 }
736
737 static HRESULT WINAPI WebBrowser_put_FullScreen(IWebBrowser2 *iface, VARIANT_BOOL bFullScreen)
738 {
739     WebBrowser *This = WEBBROWSER_THIS(iface);
740     VARIANTARG arg;
741     DISPPARAMS dispparams = {&arg, NULL, 1, 0};
742
743     /* In opposition to InternetExplorer, all we should do here is
744      * inform the embedder about the fullscreen change. */
745
746     TRACE("(%p)->(%x)\n", This, bFullScreen);
747
748     This->full_screen = bFullScreen ? VARIANT_TRUE : VARIANT_FALSE;
749
750     V_VT(&arg) = VT_BOOL;
751     V_BOOL(&arg) = bFullScreen;
752     call_sink(This->doc_host.cps.wbe2, DISPID_ONFULLSCREEN, &dispparams);
753
754     return S_OK;
755 }
756
757 static HRESULT WINAPI WebBrowser_Navigate2(IWebBrowser2 *iface, VARIANT *URL, VARIANT *Flags,
758         VARIANT *TargetFrameName, VARIANT *PostData, VARIANT *Headers)
759 {
760     WebBrowser *This = WEBBROWSER_THIS(iface);
761     LPCWSTR url;
762
763     TRACE("(%p)->(%s %s %s %s %s)\n", This, debugstr_variant(URL), debugstr_variant(Flags),
764           debugstr_variant(TargetFrameName), debugstr_variant(PostData), debugstr_variant(Headers));
765
766     if(!This->client)
767         return E_FAIL;
768
769     if(!URL)
770         return S_OK;
771
772     switch (V_VT(URL))
773     {
774     case VT_BSTR:
775         url = V_BSTR(URL);
776         break;
777     case VT_BSTR|VT_BYREF:
778         url = *V_BSTRREF(URL);
779         break;
780     default:
781         FIXME("Unsupported V_VT(URL) %d\n", V_VT(URL));
782         return E_INVALIDARG;
783     }
784
785     return navigate_url(&This->doc_host, url, Flags, TargetFrameName, PostData, Headers);
786 }
787
788 static HRESULT WINAPI WebBrowser_QueryStatusWB(IWebBrowser2 *iface, OLECMDID cmdID, OLECMDF *pcmdf)
789 {
790     WebBrowser *This = WEBBROWSER_THIS(iface);
791     FIXME("(%p)->(%d %p)\n", This, cmdID, pcmdf);
792     return E_NOTIMPL;
793 }
794
795 static HRESULT WINAPI WebBrowser_ExecWB(IWebBrowser2 *iface, OLECMDID cmdID,
796         OLECMDEXECOPT cmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
797 {
798     WebBrowser *This = WEBBROWSER_THIS(iface);
799     FIXME("(%p)->(%d %d %s %p)\n", This, cmdID, cmdexecopt, debugstr_variant(pvaIn), pvaOut);
800     return E_NOTIMPL;
801 }
802
803 static HRESULT WINAPI WebBrowser_ShowBrowserBar(IWebBrowser2 *iface, VARIANT *pvaClsid,
804         VARIANT *pvarShow, VARIANT *pvarSize)
805 {
806     WebBrowser *This = WEBBROWSER_THIS(iface);
807     FIXME("(%p)->(%s %s %s)\n", This, debugstr_variant(pvaClsid), debugstr_variant(pvarShow),
808           debugstr_variant(pvarSize));
809     return E_NOTIMPL;
810 }
811
812 static HRESULT WINAPI WebBrowser_get_ReadyState(IWebBrowser2 *iface, READYSTATE *lpReadyState)
813 {
814     WebBrowser *This = WEBBROWSER_THIS(iface);
815
816     TRACE("(%p)->(%p)\n", This, lpReadyState);
817
818     *lpReadyState = This->doc_host.ready_state;
819     return S_OK;
820 }
821
822 static HRESULT WINAPI WebBrowser_get_Offline(IWebBrowser2 *iface, VARIANT_BOOL *pbOffline)
823 {
824     WebBrowser *This = WEBBROWSER_THIS(iface);
825
826     TRACE("(%p)->(%p)\n", This, pbOffline);
827
828     *pbOffline = This->doc_host.offline;
829     return S_OK;
830 }
831
832 static HRESULT WINAPI WebBrowser_put_Offline(IWebBrowser2 *iface, VARIANT_BOOL bOffline)
833 {
834     WebBrowser *This = WEBBROWSER_THIS(iface);
835
836     TRACE("(%p)->(%x)\n", This, bOffline);
837
838     This->doc_host.offline = bOffline ? VARIANT_TRUE : VARIANT_FALSE;
839     return S_OK;
840 }
841
842 static HRESULT WINAPI WebBrowser_get_Silent(IWebBrowser2 *iface, VARIANT_BOOL *pbSilent)
843 {
844     WebBrowser *This = WEBBROWSER_THIS(iface);
845
846     TRACE("(%p)->(%p)\n", This, pbSilent);
847
848     *pbSilent = This->doc_host.silent;
849     return S_OK;
850 }
851
852 static HRESULT WINAPI WebBrowser_put_Silent(IWebBrowser2 *iface, VARIANT_BOOL bSilent)
853 {
854     WebBrowser *This = WEBBROWSER_THIS(iface);
855
856     TRACE("(%p)->(%x)\n", This, bSilent);
857
858     This->doc_host.silent = bSilent ? VARIANT_TRUE : VARIANT_FALSE;
859     return S_OK;
860 }
861
862 static HRESULT WINAPI WebBrowser_get_RegisterAsBrowser(IWebBrowser2 *iface,
863         VARIANT_BOOL *pbRegister)
864 {
865     WebBrowser *This = WEBBROWSER_THIS(iface);
866
867     FIXME("(%p)->(%p)\n", This, pbRegister);
868
869     *pbRegister = This->register_browser;
870     return S_OK;
871 }
872
873 static HRESULT WINAPI WebBrowser_put_RegisterAsBrowser(IWebBrowser2 *iface,
874         VARIANT_BOOL bRegister)
875 {
876     WebBrowser *This = WEBBROWSER_THIS(iface);
877
878     FIXME("(%p)->(%x)\n", This, bRegister);
879
880     This->register_browser = bRegister ? VARIANT_TRUE : VARIANT_FALSE;
881     return S_OK;
882 }
883
884 static HRESULT WINAPI WebBrowser_get_RegisterAsDropTarget(IWebBrowser2 *iface,
885         VARIANT_BOOL *pbRegister)
886 {
887     WebBrowser *This = WEBBROWSER_THIS(iface);
888     FIXME("(%p)->(%p)\n", This, pbRegister);
889     *pbRegister=0;
890     return S_OK;
891 }
892
893 static HRESULT WINAPI WebBrowser_put_RegisterAsDropTarget(IWebBrowser2 *iface,
894         VARIANT_BOOL bRegister)
895 {
896     WebBrowser *This = WEBBROWSER_THIS(iface);
897     FIXME("(%p)->(%x)\n", This, bRegister);
898     return S_OK;
899 }
900
901 static HRESULT WINAPI WebBrowser_get_TheaterMode(IWebBrowser2 *iface, VARIANT_BOOL *pbRegister)
902 {
903     WebBrowser *This = WEBBROWSER_THIS(iface);
904
905     TRACE("(%p)->(%p)\n", This, pbRegister);
906
907     *pbRegister = This->theater_mode;
908     return S_OK;
909 }
910
911 static HRESULT WINAPI WebBrowser_put_TheaterMode(IWebBrowser2 *iface, VARIANT_BOOL bRegister)
912 {
913     WebBrowser *This = WEBBROWSER_THIS(iface);
914     VARIANTARG arg;
915     DISPPARAMS dispparams = {&arg, NULL, 1, 0};
916
917     TRACE("(%p)->(%x)\n", This, bRegister);
918
919     This->theater_mode = bRegister ? VARIANT_TRUE : VARIANT_FALSE;
920
921     /* In opposition to InternetExplorer, all we should do here is
922      * inform the embedder about the theater mode change. */
923
924     V_VT(&arg) = VT_BOOL;
925     V_BOOL(&arg) = bRegister;
926     call_sink(This->doc_host.cps.wbe2, DISPID_ONTHEATERMODE, &dispparams);
927
928     return S_OK;
929 }
930
931 static HRESULT WINAPI WebBrowser_get_AddressBar(IWebBrowser2 *iface, VARIANT_BOOL *Value)
932 {
933     WebBrowser *This = WEBBROWSER_THIS(iface);
934
935     TRACE("(%p)->(%p)\n", This, Value);
936
937     *Value = This->address_bar;
938     return S_OK;
939 }
940
941 static HRESULT WINAPI WebBrowser_put_AddressBar(IWebBrowser2 *iface, VARIANT_BOOL Value)
942 {
943     WebBrowser *This = WEBBROWSER_THIS(iface);
944     VARIANTARG arg;
945     DISPPARAMS dispparams = {&arg, NULL, 1, 0};
946
947     TRACE("(%p)->(%x)\n", This, Value);
948
949     This->address_bar = Value ? VARIANT_TRUE : VARIANT_FALSE;
950
951     /* In opposition to InternetExplorer, all we should do here is
952      * inform the embedder about the address bar change. */
953
954     V_VT(&arg) = VT_BOOL;
955     V_BOOL(&arg) = Value;
956     call_sink(This->doc_host.cps.wbe2, DISPID_ONADDRESSBAR, &dispparams);
957
958     return S_OK;
959 }
960
961 static HRESULT WINAPI WebBrowser_get_Resizable(IWebBrowser2 *iface, VARIANT_BOOL *Value)
962 {
963     WebBrowser *This = WEBBROWSER_THIS(iface);
964
965     TRACE("(%p)->(%p)\n", This, Value);
966
967     /* It's InternetExplorer object's method. We have nothing to do here. */
968     return E_NOTIMPL;
969 }
970
971 static HRESULT WINAPI WebBrowser_put_Resizable(IWebBrowser2 *iface, VARIANT_BOOL Value)
972 {
973     WebBrowser *This = WEBBROWSER_THIS(iface);
974     VARIANTARG arg;
975     DISPPARAMS dispparams = {&arg, NULL, 1, 0};
976
977     TRACE("(%p)->(%x)\n", This, Value);
978
979     /* In opposition to InternetExplorer, all we should do here is
980      * inform the embedder about the resizable change. */
981
982     V_VT(&arg) = VT_BOOL;
983     V_BOOL(&arg) = Value;
984     call_sink(This->doc_host.cps.wbe2, DISPID_WINDOWSETRESIZABLE, &dispparams);
985
986     return S_OK;
987 }
988
989 #undef WEBBROWSER_THIS
990
991 static const IWebBrowser2Vtbl WebBrowser2Vtbl =
992 {
993     WebBrowser_QueryInterface,
994     WebBrowser_AddRef,
995     WebBrowser_Release,
996     WebBrowser_GetTypeInfoCount,
997     WebBrowser_GetTypeInfo,
998     WebBrowser_GetIDsOfNames,
999     WebBrowser_Invoke,
1000     WebBrowser_GoBack,
1001     WebBrowser_GoForward,
1002     WebBrowser_GoHome,
1003     WebBrowser_GoSearch,
1004     WebBrowser_Navigate,
1005     WebBrowser_Refresh,
1006     WebBrowser_Refresh2,
1007     WebBrowser_Stop,
1008     WebBrowser_get_Application,
1009     WebBrowser_get_Parent,
1010     WebBrowser_get_Container,
1011     WebBrowser_get_Document,
1012     WebBrowser_get_TopLevelContainer,
1013     WebBrowser_get_Type,
1014     WebBrowser_get_Left,
1015     WebBrowser_put_Left,
1016     WebBrowser_get_Top,
1017     WebBrowser_put_Top,
1018     WebBrowser_get_Width,
1019     WebBrowser_put_Width,
1020     WebBrowser_get_Height,
1021     WebBrowser_put_Height,
1022     WebBrowser_get_LocationName,
1023     WebBrowser_get_LocationURL,
1024     WebBrowser_get_Busy,
1025     WebBrowser_Quit,
1026     WebBrowser_ClientToWindow,
1027     WebBrowser_PutProperty,
1028     WebBrowser_GetProperty,
1029     WebBrowser_get_Name,
1030     WebBrowser_get_HWND,
1031     WebBrowser_get_FullName,
1032     WebBrowser_get_Path,
1033     WebBrowser_get_Visible,
1034     WebBrowser_put_Visible,
1035     WebBrowser_get_StatusBar,
1036     WebBrowser_put_StatusBar,
1037     WebBrowser_get_StatusText,
1038     WebBrowser_put_StatusText,
1039     WebBrowser_get_ToolBar,
1040     WebBrowser_put_ToolBar,
1041     WebBrowser_get_MenuBar,
1042     WebBrowser_put_MenuBar,
1043     WebBrowser_get_FullScreen,
1044     WebBrowser_put_FullScreen,
1045     WebBrowser_Navigate2,
1046     WebBrowser_QueryStatusWB,
1047     WebBrowser_ExecWB,
1048     WebBrowser_ShowBrowserBar,
1049     WebBrowser_get_ReadyState,
1050     WebBrowser_get_Offline,
1051     WebBrowser_put_Offline,
1052     WebBrowser_get_Silent,
1053     WebBrowser_put_Silent,
1054     WebBrowser_get_RegisterAsBrowser,
1055     WebBrowser_put_RegisterAsBrowser,
1056     WebBrowser_get_RegisterAsDropTarget,
1057     WebBrowser_put_RegisterAsDropTarget,
1058     WebBrowser_get_TheaterMode,
1059     WebBrowser_put_TheaterMode,
1060     WebBrowser_get_AddressBar,
1061     WebBrowser_put_AddressBar,
1062     WebBrowser_get_Resizable,
1063     WebBrowser_put_Resizable
1064 };
1065
1066 #define SERVPROV_THIS(iface) DEFINE_THIS(WebBrowser, OleObject, iface)
1067 /*
1068  *  IServiceProvider interface.
1069  */
1070 static HRESULT WINAPI WebBrowser_IServiceProvider_QueryInterface(IServiceProvider *iface,
1071             REFIID riid, LPVOID *ppv)
1072 {
1073     WebBrowser *This = SERVPROV_THIS(iface);
1074
1075     if (ppv == NULL)
1076         return E_POINTER;
1077     *ppv = NULL;
1078
1079     if(IsEqualGUID(&IID_IUnknown, riid)) {
1080         *ppv = WEBBROWSER(This);
1081         TRACE("(%p)->(IID_IUnknown %p)\n", This, ppv);
1082     }else if(IsEqualGUID(&IID_IServiceProvider, riid)) {
1083         *ppv = WEBBROWSER(This);
1084         TRACE("(%p)->(IID_IServiceProvider %p)\n", This, ppv);
1085     }
1086
1087     if(*ppv) {
1088         IUnknown_AddRef((IUnknown*)*ppv);
1089         return S_OK;
1090     }
1091
1092     FIXME("(%p)->(%s %p) interface not supported\n", This, debugstr_guid(riid), ppv);
1093     return E_NOINTERFACE;
1094 }
1095
1096 static ULONG WINAPI WebBrowser_IServiceProvider_AddRef(IServiceProvider *iface)
1097 {
1098     WebBrowser *This = SERVPROV_THIS(iface);
1099     return IWebBrowser_AddRef(WEBBROWSER(This));
1100 }
1101
1102 static ULONG WINAPI WebBrowser_IServiceProvider_Release(IServiceProvider *iface)
1103 {
1104     WebBrowser *This = SERVPROV_THIS(iface);
1105     return IWebBrowser_Release(WEBBROWSER(This));
1106 }
1107
1108 static HRESULT STDMETHODCALLTYPE WebBrowser_IServiceProvider_QueryService(IServiceProvider *iface,
1109             REFGUID guidService, REFIID riid, void **ppv)
1110 {
1111     WebBrowser *This = SERVPROV_THIS(iface);
1112     static const IID IID_IBrowserService2 =
1113         {0x68BD21CC,0x438B,0x11d2,{0xA5,0x60,0x00,0xA0,0xC,0x2D,0xBF,0xE8}};
1114
1115     if(*ppv)
1116         ppv = NULL;
1117
1118     if(IsEqualGUID(&IID_IBrowserService2, riid)) {
1119         TRACE("(%p)->(IID_IBrowserService2 return E_FAIL)\n", This);
1120         return E_FAIL;
1121     }
1122
1123     FIXME("(%p)->(%s, %s %p)\n", This, debugstr_guid(guidService), debugstr_guid(riid), ppv);
1124
1125     return E_NOINTERFACE;
1126 }
1127
1128 #undef SERVPROV_THIS
1129
1130 static const IServiceProviderVtbl ServiceProviderVtbl =
1131 {
1132     WebBrowser_IServiceProvider_QueryInterface,
1133     WebBrowser_IServiceProvider_AddRef,
1134     WebBrowser_IServiceProvider_Release,
1135     WebBrowser_IServiceProvider_QueryService
1136 };
1137
1138 #define DOCHOST_THIS(iface) DEFINE_THIS2(WebBrowser,doc_host,iface)
1139
1140 static void WINAPI DocHostContainer_GetDocObjRect(DocHost* This, RECT* rc)
1141 {
1142     GetClientRect(This->frame_hwnd, rc);
1143 }
1144
1145 static HRESULT WINAPI DocHostContainer_SetStatusText(DocHost* This, LPCWSTR text)
1146 {
1147     return E_NOTIMPL;
1148 }
1149
1150 static void WINAPI DocHostContainer_SetURL(DocHost* This, LPCWSTR url)
1151 {
1152
1153 }
1154
1155 static HRESULT DocHostContainer_exec(DocHost *doc_host, const GUID *cmd_group, DWORD cmdid, DWORD execopt, VARIANT *in,
1156         VARIANT *out)
1157 {
1158     WebBrowser *This = DOCHOST_THIS(doc_host);
1159     IOleCommandTarget *cmdtrg = NULL;
1160     HRESULT hres;
1161
1162     if(This->client) {
1163         hres = IOleClientSite_QueryInterface(This->client, &IID_IOleCommandTarget, (void**)&cmdtrg);
1164         if(FAILED(hres))
1165             cmdtrg = NULL;
1166     }
1167
1168     if(!cmdtrg && This->container) {
1169         hres = IOleContainer_QueryInterface(This->container, &IID_IOleCommandTarget, (void**)&cmdtrg);
1170         if(FAILED(hres))
1171             cmdtrg = NULL;
1172     }
1173
1174     if(!cmdtrg)
1175         return S_OK;
1176
1177     hres = IOleCommandTarget_Exec(cmdtrg, cmd_group, cmdid, execopt, in, out);
1178     IOleCommandTarget_Release(cmdtrg);
1179     if(FAILED(hres))
1180         FIXME("Exec failed\n");
1181
1182     return hres;
1183 }
1184
1185 #undef DOCHOST_THIS
1186
1187 static const IDocHostContainerVtbl DocHostContainerVtbl = {
1188     DocHostContainer_GetDocObjRect,
1189     DocHostContainer_SetStatusText,
1190     DocHostContainer_SetURL,
1191     DocHostContainer_exec
1192 };
1193
1194 static HRESULT WebBrowser_Create(INT version, IUnknown *pOuter, REFIID riid, void **ppv)
1195 {
1196     WebBrowser *ret;
1197     HRESULT hres;
1198
1199     TRACE("(%p %s %p) version=%d\n", pOuter, debugstr_guid(riid), ppv, version);
1200
1201     ret = heap_alloc_zero(sizeof(WebBrowser));
1202
1203     ret->lpWebBrowser2Vtbl = &WebBrowser2Vtbl;
1204     ret->lpServiceProviderVtbl = &ServiceProviderVtbl;
1205     ret->ref = 1;
1206     ret->version = version;
1207
1208     DocHost_Init(&ret->doc_host, (IDispatch*)WEBBROWSER2(ret), &DocHostContainerVtbl);
1209
1210     ret->visible = VARIANT_TRUE;
1211     ret->menu_bar = VARIANT_TRUE;
1212     ret->address_bar = VARIANT_TRUE;
1213     ret->status_bar = VARIANT_TRUE;
1214     ret->tool_bar = VARIANT_TRUE;
1215
1216     WebBrowser_OleObject_Init(ret);
1217     WebBrowser_ViewObject_Init(ret);
1218     WebBrowser_DataObject_Init(ret);
1219     WebBrowser_Persist_Init(ret);
1220     WebBrowser_ClassInfo_Init(ret);
1221
1222     HlinkFrame_Init(&ret->hlink_frame, (IUnknown*)WEBBROWSER2(ret), &ret->doc_host);
1223
1224     SHDOCVW_LockModule();
1225
1226     hres = IWebBrowser_QueryInterface(WEBBROWSER(ret), riid, ppv);
1227
1228     IWebBrowser2_Release(WEBBROWSER(ret));
1229     return hres;
1230 }
1231
1232 HRESULT WebBrowserV1_Create(IUnknown *pOuter, REFIID riid, void **ppv)
1233 {
1234     return WebBrowser_Create(1, pOuter, riid, ppv);
1235 }
1236
1237 HRESULT WebBrowserV2_Create(IUnknown *pOuter, REFIID riid, void **ppv)
1238 {
1239     return WebBrowser_Create(2, pOuter, riid, ppv);
1240 }