Fix some -Wstrict-prototype warnings () -> (void).
[wine] / dlls / mshtml / olecmd.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 "config.h"
20
21 #include <stdarg.h>
22 #include <stdio.h>
23
24 #define COBJMACROS
25
26 #include "windef.h"
27 #include "winbase.h"
28 #include "winuser.h"
29 #include "ole2.h"
30 #include "shlguid.h"
31 #include "mshtmdid.h"
32 #include "idispids.h"
33 #include "mshtmcid.h"
34
35 #include "wine/debug.h"
36
37 #include "mshtml_private.h"
38
39 WINE_DEFAULT_DEBUG_CHANNEL(mshtml);
40
41 /**********************************************************
42  * IOleCommandTarget implementation
43  */
44
45 #define CMDTARGET_THIS(iface) DEFINE_THIS(HTMLDocument, OleCommandTarget, iface)
46
47 static HRESULT exec_open(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
48 {
49     FIXME("(%p)->(%ld %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
50     return E_NOTIMPL;
51 }
52
53 static HRESULT exec_new(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
54 {
55     FIXME("(%p)->(%ld %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
56     return E_NOTIMPL;
57 }
58
59 static HRESULT exec_save(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
60 {
61     FIXME("(%p)->(%ld %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
62     return E_NOTIMPL;
63 }
64
65 static HRESULT exec_save_as(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
66 {
67     FIXME("(%p)->(%ld %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
68     return E_NOTIMPL;
69 }
70
71 static HRESULT exec_save_copy_as(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
72 {
73     FIXME("(%p)->(%ld %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
74     return E_NOTIMPL;
75 }
76
77 static HRESULT exec_print(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
78 {
79     FIXME("(%p)->(%ld %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
80     return E_NOTIMPL;
81 }
82
83 static HRESULT exec_print_preview(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
84 {
85     FIXME("(%p)->(%ld %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
86     return E_NOTIMPL;
87 }
88
89 static HRESULT exec_page_setup(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
90 {
91     FIXME("(%p)->(%ld %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
92     return E_NOTIMPL;
93 }
94
95 static HRESULT exec_spell(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
96 {
97     FIXME("(%p)->(%ld %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
98     return E_NOTIMPL;
99 }
100
101 static HRESULT exec_properties(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
102 {
103     FIXME("(%p)->(%ld %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
104     return E_NOTIMPL;
105 }
106
107 static HRESULT exec_cut(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
108 {
109     FIXME("(%p)->(%ld %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
110     return E_NOTIMPL;
111 }
112
113 static HRESULT exec_copy(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
114 {
115     FIXME("(%p)->(%ld %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
116     return E_NOTIMPL;
117 }
118
119 static HRESULT exec_paste(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
120 {
121     FIXME("(%p)->(%ld %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
122     return E_NOTIMPL;
123 }
124
125 static HRESULT exec_paste_special(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
126 {
127     FIXME("(%p)->(%ld %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
128     return E_NOTIMPL;
129 }
130
131 static HRESULT exec_undo(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
132 {
133     FIXME("(%p)->(%ld %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
134     return E_NOTIMPL;
135 }
136
137 static HRESULT exec_rendo(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
138 {
139     FIXME("(%p)->(%ld %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
140     return E_NOTIMPL;
141 }
142
143 static HRESULT exec_select_all(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
144 {
145     FIXME("(%p)->(%ld %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
146     return E_NOTIMPL;
147 }
148
149 static HRESULT exec_clear_selection(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
150 {
151     FIXME("(%p)->(%ld %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
152     return E_NOTIMPL;
153 }
154
155 static HRESULT exec_zoom(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
156 {
157     FIXME("(%p)->(%ld %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
158     return E_NOTIMPL;
159 }
160
161 static HRESULT exec_get_zoom_range(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
162 {
163     FIXME("(%p)->(%ld %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
164     return E_NOTIMPL;
165 }
166
167 static HRESULT exec_refresh(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
168 {
169     FIXME("(%p)->(%ld %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
170     return E_NOTIMPL;
171 }
172
173 static HRESULT exec_stop(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
174 {
175     FIXME("(%p)->(%ld %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
176     return E_NOTIMPL;
177 }
178
179 static HRESULT exec_stop_download(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
180 {
181     FIXME("(%p)->(%ld %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
182     return E_NOTIMPL;
183 }
184
185 static HRESULT exec_delete(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
186 {
187     FIXME("(%p)->(%ld %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
188     return E_NOTIMPL;
189 }
190
191 static HRESULT exec_enable_interaction(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
192 {
193     FIXME("(%p)->(%ld %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
194     return E_NOTIMPL;
195 }
196
197 static HRESULT exec_on_unload(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
198 {
199     TRACE("(%p)->(%ld %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
200
201     /* Tests show that we have nothing more to do here */
202
203     if(pvaOut) {
204         V_VT(pvaOut) = VT_BOOL;
205         V_BOOL(pvaOut) = VARIANT_TRUE;
206     }
207
208     return S_OK;
209 }
210
211 static HRESULT exec_show_page_setup(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
212 {
213     FIXME("(%p)->(%ld %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
214     return E_NOTIMPL;
215 }
216
217 static HRESULT exec_show_print(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
218 {
219     FIXME("(%p)->(%ld %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
220     return E_NOTIMPL;
221 }
222
223 static HRESULT exec_close(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
224 {
225     FIXME("(%p)->(%ld %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
226     return E_NOTIMPL;
227 }
228
229 static HRESULT exec_set_print_template(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
230 {
231     FIXME("(%p)->(%ld %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
232     return E_NOTIMPL;
233 }
234
235 static HRESULT exec_get_print_template(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
236 {
237     FIXME("(%p)->(%ld %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
238     return E_NOTIMPL;
239 }
240
241 static HRESULT exec_browsemode(HTMLDocument *This)
242 {
243     WARN("(%p)\n", This);
244
245     This->usermode = BROWSEMODE;
246
247     return S_OK;
248 }
249
250 static void setup_ns_editing(NSContainer *This)
251 {
252     nsIInterfaceRequestor *iface_req;
253     nsIEditingSession *editing_session = NULL;
254     nsIURIContentListener *listener = NULL;
255     nsIDOMWindow *dom_window = NULL;
256     nsresult nsres;
257
258     nsres = nsIWebBrowser_QueryInterface(This->webbrowser,
259             &IID_nsIInterfaceRequestor, (void**)&iface_req);
260     if(NS_FAILED(nsres)) {
261         ERR("Could not get nsIInterfaceRequestor: %08lx\n", nsres);
262         return;
263     }
264
265     nsres = nsIInterfaceRequestor_GetInterface(iface_req, &IID_nsIEditingSession,
266                                                (void**)&editing_session);
267     nsIInterfaceRequestor_Release(iface_req);
268     if(NS_FAILED(nsres)) {
269         ERR("Could not get nsIEditingSession: %08lx\n", nsres);
270         return;
271     }
272
273     nsres = nsIWebBrowser_GetContentDOMWindow(This->webbrowser, &dom_window);
274     nsIDOMWindow_Release(dom_window);
275     if(NS_FAILED(nsres)) {
276         ERR("Could not get content DOM window: %08lx\n", nsres);
277         nsIEditingSession_Release(editing_session);
278         return;
279     }
280
281     nsres = nsIEditingSession_MakeWindowEditable(editing_session, dom_window, NULL, FALSE);
282     nsIEditingSession_Release(editing_session);
283     if(NS_FAILED(nsres)) {
284         ERR("MakeWindowEditable failed: %08lx\n", nsres);
285         return;
286     }
287
288     /* MakeWindowEditable changes WebBrowser's parent URI content listener.
289      * It seams to be a bug in Gecko. To workaround it we set our content
290      * listener again and Gecko's one as its parent.
291      */
292     nsIWebBrowser_GetParentURIContentListener(This->webbrowser, &listener);
293     nsIURIContentListener_SetParentContentListener(NSURICL(This), listener);
294     nsIURIContentListener_Release(listener);
295     nsIWebBrowser_SetParentURIContentListener(This->webbrowser, NSURICL(This));
296 }
297
298 static HRESULT exec_editmode(HTMLDocument *This)
299 {
300     HRESULT hres;
301
302     TRACE("(%p)\n", This);
303
304     This->usermode = EDITMODE;
305
306     if(This->frame)
307         IOleInPlaceFrame_SetStatusText(This->frame, NULL);
308
309     if(This->client) {
310         IOleCommandTarget *cmdtrg = NULL;
311
312         hres = IOleClientSite_QueryInterface(This->client, &IID_IOleCommandTarget,
313                 (void**)&cmdtrg);
314         if(SUCCEEDED(hres)) {
315             VARIANT var;
316
317             V_VT(&var) = VT_I4;
318             V_I4(&var) = 0;
319             IOleCommandTarget_Exec(cmdtrg, &CGID_ShellDocView, 37, 0, &var, NULL);
320         }
321     }
322
323     if(This->hostui) {
324         DOCHOSTUIINFO hostinfo;
325
326         memset(&hostinfo, 0, sizeof(DOCHOSTUIINFO));
327         hostinfo.cbSize = sizeof(DOCHOSTUIINFO);
328         hres = IDocHostUIHandler_GetHostInfo(This->hostui, &hostinfo);
329         if(SUCCEEDED(hres))
330             /* FIXME: use hostinfo */
331             TRACE("hostinfo = {%lu %08lx %08lx %s %s}\n",
332                     hostinfo.cbSize, hostinfo.dwFlags, hostinfo.dwDoubleClick,
333                     debugstr_w(hostinfo.pchHostCss), debugstr_w(hostinfo.pchHostNS));
334     }
335
336     if(This->client) {
337         VARIANT silent, offline;
338
339         hres = get_client_disp_property(This->client, DISPID_AMBIENT_SILENT, &silent);
340         if(SUCCEEDED(hres)) {
341             if(V_VT(&silent) != VT_BOOL)
342                 WARN("V_VT(silent) = %d\n", V_VT(&silent));
343             else if(V_BOOL(&silent))
344                 FIXME("silent == true\n");
345         }
346
347         hres = get_client_disp_property(This->client,
348                 DISPID_AMBIENT_OFFLINEIFNOTCONNECTED, &offline); 
349         if(SUCCEEDED(hres)) {
350             if(V_VT(&silent) != VT_BOOL)
351                 WARN("V_VT(offline) = %d\n", V_VT(&silent));
352             else if(V_BOOL(&silent))
353                 FIXME("offline == true\n");
354         }
355     }
356
357     if(This->nscontainer)
358         setup_ns_editing(This->nscontainer);
359
360     return S_OK;
361 }
362
363 static HRESULT exec_baselinefont3(HTMLDocument *This)
364 {
365     FIXME("(%p)\n", This);
366     return S_OK;
367 }
368
369 static const struct {
370     OLECMDF cmdf;
371     HRESULT (*func)(HTMLDocument*,DWORD,VARIANT*,VARIANT*);
372 } exec_table[OLECMDID_GETPRINTTEMPLATE+1] = {
373     {0},
374     { OLECMDF_SUPPORTED,                  exec_open                 }, /* OLECMDID_OPEN */
375     { OLECMDF_SUPPORTED,                  exec_new                  }, /* OLECMDID_NEW */
376     { OLECMDF_SUPPORTED,                  exec_save                 }, /* OLECMDID_SAVE */
377     { OLECMDF_SUPPORTED|OLECMDF_ENABLED,  exec_save_as              }, /* OLECMDID_SAVEAS */
378     { OLECMDF_SUPPORTED,                  exec_save_copy_as         }, /* OLECMDID_SAVECOPYAS */
379     { OLECMDF_SUPPORTED|OLECMDF_ENABLED,  exec_print                }, /* OLECMDID_PRINT */
380     { OLECMDF_SUPPORTED|OLECMDF_ENABLED,  exec_print_preview        }, /* OLECMDID_PRINTPREVIEW */
381     { OLECMDF_SUPPORTED|OLECMDF_ENABLED,  exec_page_setup           }, /* OLECMDID_PAGESETUP */
382     { OLECMDF_SUPPORTED,                  exec_spell                }, /* OLECMDID_SPELL */
383     { OLECMDF_SUPPORTED|OLECMDF_ENABLED,  exec_properties           }, /* OLECMDID_PROPERTIES */
384     { OLECMDF_SUPPORTED,                  exec_cut                  }, /* OLECMDID_CUT */
385     { OLECMDF_SUPPORTED,                  exec_copy                 }, /* OLECMDID_COPY */
386     { OLECMDF_SUPPORTED,                  exec_paste                }, /* OLECMDID_PASTE */
387     { OLECMDF_SUPPORTED,                  exec_paste_special        }, /* OLECMDID_PASTESPECIAL */
388     { OLECMDF_SUPPORTED,                  exec_undo                 }, /* OLECMDID_UNDO */
389     { OLECMDF_SUPPORTED,                  exec_rendo                }, /* OLECMDID_REDO */
390     { OLECMDF_SUPPORTED|OLECMDF_ENABLED,  exec_select_all           }, /* OLECMDID_SELECTALL */
391     { OLECMDF_SUPPORTED,                  exec_clear_selection      }, /* OLECMDID_CLEARSELECTION */
392     { OLECMDF_SUPPORTED,                  exec_zoom                 }, /* OLECMDID_ZOOM */
393     { OLECMDF_SUPPORTED,                  exec_get_zoom_range       }, /* OLECMDID_GETZOOMRANGE */
394     {0},
395     { OLECMDF_SUPPORTED|OLECMDF_ENABLED,  exec_refresh              }, /* OLECMDID_REFRESH */
396     { OLECMDF_SUPPORTED|OLECMDF_ENABLED,  exec_stop                 }, /* OLECMDID_STOP */
397     {0},{0},{0},{0},{0},{0},
398     { OLECMDF_SUPPORTED,                  exec_stop_download        }, /* OLECMDID_STOPDOWNLOAD */
399     {0},{0},
400     { OLECMDF_SUPPORTED,                  exec_delete               }, /* OLECMDID_DELETE */
401     {0},{0},
402     { OLECMDF_SUPPORTED,                  exec_enable_interaction   }, /* OLECMDID_ENABLE_INTERACTION */
403     { OLECMDF_SUPPORTED,                  exec_on_unload            }, /* OLECMDID_ONUNLOAD */
404     {0},{0},{0},{0},{0},
405     { OLECMDF_SUPPORTED,                  exec_show_page_setup      }, /* OLECMDID_SHOWPAGESETUP */
406     { OLECMDF_SUPPORTED,                  exec_show_print           }, /* OLECMDID_SHOWPRINT */
407     {0},{0},
408     { OLECMDF_SUPPORTED,                  exec_close                }, /* OLECMDID_CLOSE */
409     {0},{0},{0},
410     { OLECMDF_SUPPORTED,                  exec_set_print_template   }, /* OLECMDID_SETPRINTTEMPLATE */
411     { OLECMDF_SUPPORTED,                  exec_get_print_template   }  /* OLECMDID_GETPRINTTEMPLATE */
412 };
413
414 static HRESULT WINAPI OleCommandTarget_QueryInterface(IOleCommandTarget *iface, REFIID riid, void **ppv)
415 {
416     HTMLDocument *This = CMDTARGET_THIS(iface);
417     return IHTMLDocument2_QueryInterface(HTMLDOC(This), riid, ppv);
418 }
419
420 static ULONG WINAPI OleCommandTarget_AddRef(IOleCommandTarget *iface)
421 {
422     HTMLDocument *This = CMDTARGET_THIS(iface);
423     return IHTMLDocument2_AddRef(HTMLDOC(This));
424 }
425
426 static ULONG WINAPI OleCommandTarget_Release(IOleCommandTarget *iface)
427 {
428     HTMLDocument *This = CMDTARGET_THIS(iface);
429     return IHTMLDocument_Release(HTMLDOC(This));
430 }
431
432 static HRESULT WINAPI OleCommandTarget_QueryStatus(IOleCommandTarget *iface, const GUID *pguidCmdGroup,
433         ULONG cCmds, OLECMD prgCmds[], OLECMDTEXT *pCmdText)
434 {
435     HTMLDocument *This = CMDTARGET_THIS(iface);
436     HRESULT hres = S_OK, hr;
437
438     TRACE("(%p)->(%s %ld %p %p)\n", This, debugstr_guid(pguidCmdGroup), cCmds, prgCmds, pCmdText);
439
440     if(!pguidCmdGroup) {
441         ULONG i;
442
443         for(i=0; i<cCmds; i++) {
444             if(prgCmds[i].cmdID<OLECMDID_OPEN || prgCmds[i].cmdID>OLECMDID_GETPRINTTEMPLATE) {
445                 WARN("Unsupported cmdID = %ld\n", prgCmds[i].cmdID);
446                 prgCmds[i].cmdf = 0;
447                 hres = OLECMDERR_E_NOTSUPPORTED;
448             }else {
449                 if(prgCmds[i].cmdID == OLECMDID_OPEN || prgCmds[i].cmdID == OLECMDID_NEW) {
450                     IOleCommandTarget *cmdtrg = NULL;
451                     OLECMD olecmd;
452
453                     prgCmds[i].cmdf = OLECMDF_SUPPORTED;
454                     if(This->client) {
455                         hr = IOleClientSite_QueryInterface(This->client, &IID_IOleCommandTarget,
456                                 (void**)&cmdtrg);
457                         if(SUCCEEDED(hr)) {
458                             olecmd.cmdID = prgCmds[i].cmdID;
459                             olecmd.cmdf = 0;
460
461                             hr = IOleCommandTarget_QueryStatus(cmdtrg, NULL, 1, &olecmd, NULL);
462                             if(SUCCEEDED(hr) && olecmd.cmdf)
463                                 prgCmds[i].cmdf = olecmd.cmdf;
464                         }
465                     }else {
466                         ERR("This->client == NULL, native would crash\n");
467                     }
468                 }else {
469                     prgCmds[i].cmdf = exec_table[prgCmds[i].cmdID].cmdf;
470                     TRACE("cmdID = %ld  returning %lx\n", prgCmds[i].cmdID, prgCmds[i].cmdf);
471                 }
472                 hres = S_OK;
473             }
474         }
475
476         if(pCmdText)
477             FIXME("Set pCmdText\n");
478     }else {
479         FIXME("Unsupported pguidCmdGroup %s\n", debugstr_guid(pguidCmdGroup));
480         hres = OLECMDERR_E_UNKNOWNGROUP;
481     }
482
483     return hres;
484 }
485
486 static HRESULT WINAPI OleCommandTarget_Exec(IOleCommandTarget *iface, const GUID *pguidCmdGroup,
487         DWORD nCmdID, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
488 {
489     HTMLDocument *This = CMDTARGET_THIS(iface);
490
491     if(!pguidCmdGroup) {
492         if(nCmdID<OLECMDID_OPEN || nCmdID>OLECMDID_GETPRINTTEMPLATE || !exec_table[nCmdID].func) {
493             WARN("Unsupported cmdID = %ld\n", nCmdID);
494             return OLECMDERR_E_NOTSUPPORTED;
495         }
496
497         return exec_table[nCmdID].func(This, nCmdexecopt, pvaIn, pvaOut);
498     }else if(IsEqualGUID(&CGID_Explorer, pguidCmdGroup)) {
499         FIXME("unsupported nCmdID %ld of CGID_Explorer group\n", nCmdID);
500         TRACE("%p %p\n", pvaIn, pvaOut);
501         return OLECMDERR_E_NOTSUPPORTED;
502     }else if(IsEqualGUID(&CGID_ShellDocView, pguidCmdGroup)) {
503         FIXME("unsupported nCmdID %ld of CGID_ShellDocView group\n", nCmdID);
504         return OLECMDERR_E_NOTSUPPORTED;
505     }else if(IsEqualGUID(&CGID_MSHTML, pguidCmdGroup)) {
506         switch(nCmdID) {
507         case IDM_BROWSEMODE:
508             return exec_browsemode(This);
509         case IDM_EDITMODE:
510             if(pvaIn || pvaOut)
511                 FIXME("unsupported arguments\n");
512             return exec_editmode(This);
513         case IDM_BASELINEFONT3:
514             return exec_baselinefont3(This);
515         default:
516             FIXME("unsupported nCmdID %ld of CGID_MSHTML group\n", nCmdID);
517             return OLECMDERR_E_NOTSUPPORTED;
518         }
519     }
520
521     FIXME("Unsupported pguidCmdGroup %s\n", debugstr_guid(pguidCmdGroup));
522     return OLECMDERR_E_UNKNOWNGROUP;
523 }
524
525 #undef CMDTARGET_THIS
526
527 static const IOleCommandTargetVtbl OleCommandTargetVtbl = {
528     OleCommandTarget_QueryInterface,
529     OleCommandTarget_AddRef,
530     OleCommandTarget_Release,
531     OleCommandTarget_QueryStatus,
532     OleCommandTarget_Exec
533 };
534
535 void HTMLDocument_OleCmd_Init(HTMLDocument *This)
536 {
537     This->lpOleCommandTargetVtbl = &OleCommandTargetVtbl;
538 }