push(c) 7dcddf6204ed5518706a76fe66fd836d81e70dd4
[wine/hacks.git] / dlls / mshtml / olecmd.c
bloba137057bf6ded000a04602a70fefdf0257242aa8
1 /*
2 * Copyright 2005-2007 Jacek Caban for CodeWeavers
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.
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.
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
19 #include "config.h"
21 #include <stdarg.h>
22 #include <stdio.h>
24 #define COBJMACROS
26 #include "windef.h"
27 #include "winbase.h"
28 #include "winuser.h"
29 #include "winnls.h"
30 #include "ole2.h"
31 #include "shlguid.h"
32 #include "mshtmdid.h"
33 #include "idispids.h"
34 #include "mshtmcid.h"
36 #include "wine/debug.h"
37 #include "wine/unicode.h"
39 #include "mshtml_private.h"
40 #include "resource.h"
42 WINE_DEFAULT_DEBUG_CHANNEL(mshtml);
44 #define NSCMD_COPY "cmd_copy"
46 void do_ns_command(NSContainer *This, const char *cmd, nsICommandParams *nsparam)
48 nsICommandManager *cmdmgr;
49 nsresult nsres;
51 TRACE("(%p)\n", This);
53 nsres = get_nsinterface((nsISupports*)This->webbrowser, &IID_nsICommandManager, (void**)&cmdmgr);
54 if(NS_FAILED(nsres)) {
55 ERR("Could not get nsICommandManager: %08x\n", nsres);
56 return;
59 nsres = nsICommandManager_DoCommand(cmdmgr, cmd, nsparam, NULL);
60 if(NS_FAILED(nsres))
61 ERR("DoCommand(%s) failed: %08x\n", debugstr_a(cmd), nsres);
63 nsICommandManager_Release(cmdmgr);
66 /**********************************************************
67 * IOleCommandTarget implementation
70 #define CMDTARGET_THIS(iface) DEFINE_THIS(HTMLDocument, OleCommandTarget, iface)
72 static HRESULT exec_open(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
74 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
75 return E_NOTIMPL;
78 static HRESULT exec_new(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
80 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
81 return E_NOTIMPL;
84 static HRESULT exec_save(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
86 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
87 return E_NOTIMPL;
90 static HRESULT exec_save_as(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
92 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
93 return E_NOTIMPL;
96 static HRESULT exec_save_copy_as(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
98 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
99 return E_NOTIMPL;
102 static nsresult set_head_text(nsIPrintSettings *settings, LPCWSTR template, BOOL head, int pos)
104 if(head) {
105 switch(pos) {
106 case 0:
107 return nsIPrintSettings_SetHeaderStrLeft(settings, template);
108 case 1:
109 return nsIPrintSettings_SetHeaderStrRight(settings, template);
110 case 2:
111 return nsIPrintSettings_SetHeaderStrCenter(settings, template);
113 }else {
114 switch(pos) {
115 case 0:
116 return nsIPrintSettings_SetFooterStrLeft(settings, template);
117 case 1:
118 return nsIPrintSettings_SetFooterStrRight(settings, template);
119 case 2:
120 return nsIPrintSettings_SetFooterStrCenter(settings, template);
124 return NS_OK;
127 static void set_print_template(nsIPrintSettings *settings, LPCWSTR template, BOOL head)
129 PRUnichar nstemplate[200]; /* FIXME: Use dynamic allocation */
130 PRUnichar *p = nstemplate;
131 LPCWSTR ptr=template;
132 int pos=0;
134 while(*ptr) {
135 if(*ptr != '&') {
136 *p++ = *ptr++;
137 continue;
140 switch(*++ptr) {
141 case '&':
142 *p++ = '&';
143 *p++ = '&';
144 ptr++;
145 break;
146 case 'b': /* change align */
147 ptr++;
148 *p = 0;
149 set_head_text(settings, nstemplate, head, pos);
150 p = nstemplate;
151 pos++;
152 break;
153 case 'd': { /* short date */
154 SYSTEMTIME systime;
155 GetLocalTime(&systime);
156 GetDateFormatW(LOCALE_SYSTEM_DEFAULT, 0, &systime, NULL, p,
157 sizeof(nstemplate)-(p-nstemplate)*sizeof(WCHAR));
158 p += strlenW(p);
159 ptr++;
160 break;
162 case 'p': /* page number */
163 *p++ = '&';
164 *p++ = 'P';
165 ptr++;
166 break;
167 case 'P': /* page count */
168 *p++ = '?'; /* FIXME */
169 ptr++;
170 break;
171 case 'u':
172 *p++ = '&';
173 *p++ = 'U';
174 ptr++;
175 break;
176 case 'w':
177 /* FIXME: set window title */
178 ptr++;
179 break;
180 default:
181 *p++ = '&';
182 *p++ = *ptr++;
186 *p = 0;
187 set_head_text(settings, nstemplate, head, pos);
189 while(++pos < 3)
190 set_head_text(settings, p, head, pos);
193 static void set_default_templates(nsIPrintSettings *settings)
195 WCHAR buf[64];
197 static const PRUnichar empty[] = {0};
199 nsIPrintSettings_SetHeaderStrLeft(settings, empty);
200 nsIPrintSettings_SetHeaderStrRight(settings, empty);
201 nsIPrintSettings_SetHeaderStrCenter(settings, empty);
202 nsIPrintSettings_SetFooterStrLeft(settings, empty);
203 nsIPrintSettings_SetFooterStrRight(settings, empty);
204 nsIPrintSettings_SetFooterStrCenter(settings, empty);
206 if(LoadStringW(get_shdoclc(), IDS_PRINT_HEADER_TEMPLATE, buf,
207 sizeof(buf)/sizeof(WCHAR)))
208 set_print_template(settings, buf, TRUE);
211 if(LoadStringW(get_shdoclc(), IDS_PRINT_FOOTER_TEMPLATE, buf,
212 sizeof(buf)/sizeof(WCHAR)))
213 set_print_template(settings, buf, FALSE);
217 static HRESULT exec_print(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
219 nsIWebBrowserPrint *nsprint;
220 nsIPrintSettings *settings;
221 nsresult nsres;
223 TRACE("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
225 if(pvaOut)
226 FIXME("unsupported pvaOut\n");
228 if(!This->nscontainer)
229 return S_OK;
231 nsres = get_nsinterface((nsISupports*)This->nscontainer->webbrowser, &IID_nsIWebBrowserPrint,
232 (void**)&nsprint);
233 if(NS_FAILED(nsres)) {
234 ERR("Could not get nsIWebBrowserPrint: %08x\n", nsres);
235 return S_OK;
238 nsres = nsIWebBrowserPrint_GetGlobalPrintSettings(nsprint, &settings);
239 if(NS_FAILED(nsres))
240 ERR("GetCurrentPrintSettings failed: %08x\n", nsres);
242 set_default_templates(settings);
244 if(pvaIn) {
245 switch(V_VT(pvaIn)) {
246 case VT_BYREF|VT_ARRAY: {
247 VARIANT *opts;
248 DWORD opts_cnt;
250 if(V_ARRAY(pvaIn)->cDims != 1)
251 WARN("cDims = %d\n", V_ARRAY(pvaIn)->cDims);
253 SafeArrayAccessData(V_ARRAY(pvaIn), (void**)&opts);
254 opts_cnt = V_ARRAY(pvaIn)->rgsabound[0].cElements;
256 if(opts_cnt >= 1) {
257 switch(V_VT(opts)) {
258 case VT_BSTR:
259 TRACE("setting footer %s\n", debugstr_w(V_BSTR(opts)));
260 set_print_template(settings, V_BSTR(opts), TRUE);
261 break;
262 case VT_NULL:
263 break;
264 default:
265 WARN("V_VT(opts) = %d\n", V_VT(opts));
269 if(opts_cnt >= 2) {
270 switch(V_VT(opts+1)) {
271 case VT_BSTR:
272 TRACE("setting footer %s\n", debugstr_w(V_BSTR(opts+1)));
273 set_print_template(settings, V_BSTR(opts+1), FALSE);
274 break;
275 case VT_NULL:
276 break;
277 default:
278 WARN("V_VT(opts) = %d\n", V_VT(opts+1));
282 if(opts_cnt >= 3)
283 FIXME("Unsupported opts_cnt %d\n", opts_cnt);
285 SafeArrayUnaccessData(V_ARRAY(pvaIn));
286 break;
288 default:
289 FIXME("unsupported vt %x\n", V_VT(pvaIn));
293 nsres = nsIWebBrowserPrint_Print(nsprint, settings, NULL);
294 if(NS_FAILED(nsres))
295 ERR("Print failed: %08x\n", nsres);
297 nsIWebBrowserPrint_Release(nsprint);
299 return S_OK;
302 static HRESULT exec_print_preview(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
304 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
305 return E_NOTIMPL;
308 static HRESULT exec_page_setup(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
310 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
311 return E_NOTIMPL;
314 static HRESULT exec_spell(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
316 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
317 return E_NOTIMPL;
320 static HRESULT exec_properties(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
322 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
323 return E_NOTIMPL;
326 static HRESULT exec_cut(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
328 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
329 return E_NOTIMPL;
332 static HRESULT exec_copy(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
334 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
335 return E_NOTIMPL;
338 static HRESULT exec_paste(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
340 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
341 return E_NOTIMPL;
344 static HRESULT exec_paste_special(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
346 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
347 return E_NOTIMPL;
350 static HRESULT exec_undo(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
352 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
353 return E_NOTIMPL;
356 static HRESULT exec_rendo(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
358 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
359 return E_NOTIMPL;
362 static HRESULT exec_select_all(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
364 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
365 return E_NOTIMPL;
368 static HRESULT exec_clear_selection(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
370 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
371 return E_NOTIMPL;
374 static HRESULT exec_zoom(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
376 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
377 return E_NOTIMPL;
380 static HRESULT exec_get_zoom_range(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
382 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
383 return E_NOTIMPL;
386 static HRESULT exec_refresh(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
388 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
389 return E_NOTIMPL;
392 static HRESULT exec_stop(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
394 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
395 return E_NOTIMPL;
398 static HRESULT exec_stop_download(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
400 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
401 return E_NOTIMPL;
404 static HRESULT exec_delete(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
406 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
407 return E_NOTIMPL;
410 static HRESULT exec_enable_interaction(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
412 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
413 return E_NOTIMPL;
416 static HRESULT exec_on_unload(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
418 TRACE("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
420 /* Tests show that we have nothing more to do here */
422 if(pvaOut) {
423 V_VT(pvaOut) = VT_BOOL;
424 V_BOOL(pvaOut) = VARIANT_TRUE;
427 return S_OK;
430 static HRESULT exec_show_page_setup(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
432 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
433 return E_NOTIMPL;
436 static HRESULT exec_show_print(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
438 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
439 return E_NOTIMPL;
442 static HRESULT exec_close(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
444 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
445 return E_NOTIMPL;
448 static HRESULT exec_set_print_template(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
450 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
451 return E_NOTIMPL;
454 static HRESULT exec_get_print_template(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
456 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
457 return E_NOTIMPL;
460 static HRESULT query_mshtml_copy(HTMLDocument *This, OLECMD *cmd)
462 FIXME("(%p)\n", This);
463 cmd->cmdf = OLECMDF_SUPPORTED|OLECMDF_ENABLED;
464 return S_OK;
467 static HRESULT exec_mshtml_copy(HTMLDocument *This, DWORD cmdexecopt, VARIANT *in, VARIANT *out)
469 TRACE("(%p)->(%08x %p %p)\n", This, cmdexecopt, in, out);
471 if(This->usermode == EDITMODE)
472 return editor_exec_copy(This, cmdexecopt, in, out);
474 do_ns_command(This->nscontainer, NSCMD_COPY, NULL);
475 return S_OK;
478 static HRESULT query_mshtml_cut(HTMLDocument *This, OLECMD *cmd)
480 FIXME("(%p)\n", This);
481 cmd->cmdf = OLECMDF_SUPPORTED|OLECMDF_ENABLED;
482 return S_OK;
485 static HRESULT exec_mshtml_cut(HTMLDocument *This, DWORD cmdexecopt, VARIANT *in, VARIANT *out)
487 TRACE("(%p)->(%08x %p %p)\n", This, cmdexecopt, in, out);
489 if(This->usermode == EDITMODE)
490 return editor_exec_cut(This, cmdexecopt, in, out);
492 FIXME("Unimplemented in browse mode\n");
493 return E_NOTIMPL;
496 static HRESULT query_mshtml_paste(HTMLDocument *This, OLECMD *cmd)
498 FIXME("(%p)\n", This);
499 cmd->cmdf = OLECMDF_SUPPORTED|OLECMDF_ENABLED;
500 return S_OK;
503 static HRESULT exec_mshtml_paste(HTMLDocument *This, DWORD cmdexecopt, VARIANT *in, VARIANT *out)
505 TRACE("(%p)->(%08x %p %p)\n", This, cmdexecopt, in, out);
507 if(This->usermode == EDITMODE)
508 return editor_exec_paste(This, cmdexecopt, in, out);
510 FIXME("Unimplemented in browse mode\n");
511 return E_NOTIMPL;
514 static HRESULT exec_browsemode(HTMLDocument *This, DWORD cmdexecopt, VARIANT *in, VARIANT *out)
516 WARN("(%p)->(%08x %p %p)\n", This, cmdexecopt, in, out);
518 if(in || out)
519 FIXME("unsupported args\n");
521 This->usermode = BROWSEMODE;
523 return S_OK;
526 static HRESULT exec_editmode(HTMLDocument *This, DWORD cmdexecopt, VARIANT *in, VARIANT *out)
528 IMoniker *mon;
529 HRESULT hres;
531 static const WCHAR wszAboutBlank[] = {'a','b','o','u','t',':','b','l','a','n','k',0};
533 TRACE("(%p)->(%08x %p %p)\n", This, cmdexecopt, in, out);
535 if(in || out)
536 FIXME("unsupported args\n");
538 if(This->usermode == EDITMODE)
539 return S_OK;
541 This->usermode = EDITMODE;
543 if(This->frame)
544 IOleInPlaceFrame_SetStatusText(This->frame, NULL);
546 if(This->client) {
547 IOleCommandTarget *cmdtrg;
549 hres = IOleClientSite_QueryInterface(This->client, &IID_IOleCommandTarget,
550 (void**)&cmdtrg);
551 if(SUCCEEDED(hres)) {
552 VARIANT var;
554 V_VT(&var) = VT_I4;
555 V_I4(&var) = 0;
556 IOleCommandTarget_Exec(cmdtrg, &CGID_ShellDocView, 37, 0, &var, NULL);
558 IOleCommandTarget_Release(cmdtrg);
562 if(This->hostui) {
563 DOCHOSTUIINFO hostinfo;
565 memset(&hostinfo, 0, sizeof(DOCHOSTUIINFO));
566 hostinfo.cbSize = sizeof(DOCHOSTUIINFO);
567 hres = IDocHostUIHandler_GetHostInfo(This->hostui, &hostinfo);
568 if(SUCCEEDED(hres))
569 /* FIXME: use hostinfo */
570 TRACE("hostinfo = {%u %08x %08x %s %s}\n",
571 hostinfo.cbSize, hostinfo.dwFlags, hostinfo.dwDoubleClick,
572 debugstr_w(hostinfo.pchHostCss), debugstr_w(hostinfo.pchHostNS));
575 if(This->nscontainer)
576 set_ns_editmode(This->nscontainer);
578 hres = CreateURLMoniker(NULL, wszAboutBlank, &mon);
579 if(FAILED(hres)) {
580 FIXME("CreateURLMoniker failed: %08x\n", hres);
581 return hres;
584 update_doc(This, UPDATE_UI);
586 return IPersistMoniker_Load(PERSISTMON(This), TRUE, mon, NULL, 0);
589 static HRESULT exec_htmleditmode(HTMLDocument *This, DWORD cmdexecopt, VARIANT *in, VARIANT *out)
591 FIXME("(%p)->(%08x %p %p)\n", This, cmdexecopt, in, out);
592 return S_OK;
595 static HRESULT exec_setdirty(HTMLDocument *This, DWORD cmdexecopt, VARIANT *in, VARIANT *out)
597 FIXME("(%p)->(%08x %p %p)\n", This, cmdexecopt, in, out);
598 return E_NOTIMPL;
601 static HRESULT exec_baselinefont3(HTMLDocument *This, DWORD cmdexecopt, VARIANT *in, VARIANT *out)
603 FIXME("(%p)->(%08x %p %p)\n", This, cmdexecopt, in, out);
604 return S_OK;
607 static HRESULT exec_respectvisibility_indesign(HTMLDocument *This, DWORD cmdexecopt,
608 VARIANT *in, VARIANT *out)
610 FIXME("(%p)->(%08x %p %p)\n", This, cmdexecopt, in, out);
611 return E_NOTIMPL;
614 static HRESULT query_enabled_stub(HTMLDocument *This, OLECMD *cmd)
616 switch(cmd->cmdID) {
617 case IDM_PRINT:
618 FIXME("CGID_MSHTML: IDM_PRINT\n");
619 cmd->cmdf = OLECMDF_SUPPORTED|OLECMDF_ENABLED;
620 break;
621 case IDM_BLOCKDIRLTR:
622 FIXME("CGID_MSHTML: IDM_BLOCKDIRLTR\n");
623 cmd->cmdf = OLECMDF_SUPPORTED|OLECMDF_ENABLED;
624 break;
625 case IDM_BLOCKDIRRTL:
626 FIXME("CGID_MSHTML: IDM_BLOCKDIRRTL\n");
627 cmd->cmdf = OLECMDF_SUPPORTED|OLECMDF_ENABLED;
628 break;
631 return S_OK;
634 static const struct {
635 OLECMDF cmdf;
636 HRESULT (*func)(HTMLDocument*,DWORD,VARIANT*,VARIANT*);
637 } exec_table[OLECMDID_GETPRINTTEMPLATE+1] = {
638 {0},
639 { OLECMDF_SUPPORTED, exec_open }, /* OLECMDID_OPEN */
640 { OLECMDF_SUPPORTED, exec_new }, /* OLECMDID_NEW */
641 { OLECMDF_SUPPORTED, exec_save }, /* OLECMDID_SAVE */
642 { OLECMDF_SUPPORTED|OLECMDF_ENABLED, exec_save_as }, /* OLECMDID_SAVEAS */
643 { OLECMDF_SUPPORTED, exec_save_copy_as }, /* OLECMDID_SAVECOPYAS */
644 { OLECMDF_SUPPORTED|OLECMDF_ENABLED, exec_print }, /* OLECMDID_PRINT */
645 { OLECMDF_SUPPORTED|OLECMDF_ENABLED, exec_print_preview }, /* OLECMDID_PRINTPREVIEW */
646 { OLECMDF_SUPPORTED|OLECMDF_ENABLED, exec_page_setup }, /* OLECMDID_PAGESETUP */
647 { OLECMDF_SUPPORTED, exec_spell }, /* OLECMDID_SPELL */
648 { OLECMDF_SUPPORTED|OLECMDF_ENABLED, exec_properties }, /* OLECMDID_PROPERTIES */
649 { OLECMDF_SUPPORTED, exec_cut }, /* OLECMDID_CUT */
650 { OLECMDF_SUPPORTED, exec_copy }, /* OLECMDID_COPY */
651 { OLECMDF_SUPPORTED, exec_paste }, /* OLECMDID_PASTE */
652 { OLECMDF_SUPPORTED, exec_paste_special }, /* OLECMDID_PASTESPECIAL */
653 { OLECMDF_SUPPORTED, exec_undo }, /* OLECMDID_UNDO */
654 { OLECMDF_SUPPORTED, exec_rendo }, /* OLECMDID_REDO */
655 { OLECMDF_SUPPORTED|OLECMDF_ENABLED, exec_select_all }, /* OLECMDID_SELECTALL */
656 { OLECMDF_SUPPORTED, exec_clear_selection }, /* OLECMDID_CLEARSELECTION */
657 { OLECMDF_SUPPORTED, exec_zoom }, /* OLECMDID_ZOOM */
658 { OLECMDF_SUPPORTED, exec_get_zoom_range }, /* OLECMDID_GETZOOMRANGE */
659 {0},
660 { OLECMDF_SUPPORTED|OLECMDF_ENABLED, exec_refresh }, /* OLECMDID_REFRESH */
661 { OLECMDF_SUPPORTED|OLECMDF_ENABLED, exec_stop }, /* OLECMDID_STOP */
662 {0},{0},{0},{0},{0},{0},
663 { OLECMDF_SUPPORTED, exec_stop_download }, /* OLECMDID_STOPDOWNLOAD */
664 {0},{0},
665 { OLECMDF_SUPPORTED, exec_delete }, /* OLECMDID_DELETE */
666 {0},{0},
667 { OLECMDF_SUPPORTED, exec_enable_interaction }, /* OLECMDID_ENABLE_INTERACTION */
668 { OLECMDF_SUPPORTED, exec_on_unload }, /* OLECMDID_ONUNLOAD */
669 {0},{0},{0},{0},{0},
670 { OLECMDF_SUPPORTED, exec_show_page_setup }, /* OLECMDID_SHOWPAGESETUP */
671 { OLECMDF_SUPPORTED, exec_show_print }, /* OLECMDID_SHOWPRINT */
672 {0},{0},
673 { OLECMDF_SUPPORTED, exec_close }, /* OLECMDID_CLOSE */
674 {0},{0},{0},
675 { OLECMDF_SUPPORTED, exec_set_print_template }, /* OLECMDID_SETPRINTTEMPLATE */
676 { OLECMDF_SUPPORTED, exec_get_print_template } /* OLECMDID_GETPRINTTEMPLATE */
679 static const cmdtable_t base_cmds[] = {
680 {IDM_COPY, query_mshtml_copy, exec_mshtml_copy},
681 {IDM_PASTE, query_mshtml_paste, exec_mshtml_paste},
682 {IDM_CUT, query_mshtml_cut, exec_mshtml_cut},
683 {IDM_BROWSEMODE, NULL, exec_browsemode},
684 {IDM_EDITMODE, NULL, exec_editmode},
685 {IDM_PRINT, query_enabled_stub, exec_print},
686 {IDM_SETDIRTY, NULL, exec_setdirty},
687 {IDM_HTMLEDITMODE, NULL, exec_htmleditmode},
688 {IDM_BASELINEFONT3, NULL, exec_baselinefont3},
689 {IDM_BLOCKDIRLTR, query_enabled_stub, NULL},
690 {IDM_BLOCKDIRRTL, query_enabled_stub, NULL},
691 {IDM_RESPECTVISIBILITY_INDESIGN, NULL, exec_respectvisibility_indesign},
692 {0,NULL,NULL}
695 static HRESULT WINAPI OleCommandTarget_QueryInterface(IOleCommandTarget *iface, REFIID riid, void **ppv)
697 HTMLDocument *This = CMDTARGET_THIS(iface);
698 return IHTMLDocument2_QueryInterface(HTMLDOC(This), riid, ppv);
701 static ULONG WINAPI OleCommandTarget_AddRef(IOleCommandTarget *iface)
703 HTMLDocument *This = CMDTARGET_THIS(iface);
704 return IHTMLDocument2_AddRef(HTMLDOC(This));
707 static ULONG WINAPI OleCommandTarget_Release(IOleCommandTarget *iface)
709 HTMLDocument *This = CMDTARGET_THIS(iface);
710 return IHTMLDocument_Release(HTMLDOC(This));
713 static HRESULT query_from_table(HTMLDocument *This, const cmdtable_t *cmdtable, OLECMD *cmd)
715 const cmdtable_t *iter = cmdtable;
717 cmd->cmdf = 0;
719 while(iter->id && iter->id != cmd->cmdID)
720 iter++;
722 if(!iter->id || !iter->query)
723 return OLECMDERR_E_NOTSUPPORTED;
725 return iter->query(This, cmd);
728 static HRESULT WINAPI OleCommandTarget_QueryStatus(IOleCommandTarget *iface, const GUID *pguidCmdGroup,
729 ULONG cCmds, OLECMD prgCmds[], OLECMDTEXT *pCmdText)
731 HTMLDocument *This = CMDTARGET_THIS(iface);
732 HRESULT hres = S_OK, hr;
734 TRACE("(%p)->(%s %d %p %p)\n", This, debugstr_guid(pguidCmdGroup), cCmds, prgCmds, pCmdText);
736 if(!pguidCmdGroup) {
737 ULONG i;
739 for(i=0; i<cCmds; i++) {
740 if(prgCmds[i].cmdID<OLECMDID_OPEN || prgCmds[i].cmdID>OLECMDID_GETPRINTTEMPLATE) {
741 WARN("Unsupported cmdID = %d\n", prgCmds[i].cmdID);
742 prgCmds[i].cmdf = 0;
743 hres = OLECMDERR_E_NOTSUPPORTED;
744 }else {
745 if(prgCmds[i].cmdID == OLECMDID_OPEN || prgCmds[i].cmdID == OLECMDID_NEW) {
746 IOleCommandTarget *cmdtrg = NULL;
747 OLECMD olecmd;
749 prgCmds[i].cmdf = OLECMDF_SUPPORTED;
750 if(This->client) {
751 hr = IOleClientSite_QueryInterface(This->client, &IID_IOleCommandTarget,
752 (void**)&cmdtrg);
753 if(SUCCEEDED(hr)) {
754 olecmd.cmdID = prgCmds[i].cmdID;
755 olecmd.cmdf = 0;
757 hr = IOleCommandTarget_QueryStatus(cmdtrg, NULL, 1, &olecmd, NULL);
758 if(SUCCEEDED(hr) && olecmd.cmdf)
759 prgCmds[i].cmdf = olecmd.cmdf;
761 }else {
762 ERR("This->client == NULL, native would crash\n");
764 }else {
765 prgCmds[i].cmdf = exec_table[prgCmds[i].cmdID].cmdf;
766 TRACE("cmdID = %d returning %x\n", prgCmds[i].cmdID, prgCmds[i].cmdf);
768 hres = S_OK;
772 if(pCmdText)
773 FIXME("Set pCmdText\n");
774 }else if(IsEqualGUID(&CGID_MSHTML, pguidCmdGroup)) {
775 ULONG i;
777 for(i=0; i<cCmds; i++) {
778 HRESULT hres = query_from_table(This, base_cmds, prgCmds+i);
779 if(hres == OLECMDERR_E_NOTSUPPORTED)
780 hres = query_from_table(This, editmode_cmds, prgCmds+i);
781 if(hres == OLECMDERR_E_NOTSUPPORTED)
782 FIXME("CGID_MSHTML: unsupported cmdID %d\n", prgCmds[i].cmdID);
785 hres = prgCmds[i-1].cmdf ? S_OK : OLECMDERR_E_NOTSUPPORTED;
787 if(pCmdText)
788 FIXME("Set pCmdText\n");
789 }else {
790 FIXME("Unsupported pguidCmdGroup %s\n", debugstr_guid(pguidCmdGroup));
791 hres = OLECMDERR_E_UNKNOWNGROUP;
794 return hres;
797 static HRESULT exec_from_table(HTMLDocument *This, const cmdtable_t *cmdtable, DWORD cmdid,
798 DWORD cmdexecopt, VARIANT *in, VARIANT *out)
800 const cmdtable_t *iter = cmdtable;
802 while(iter->id && iter->id != cmdid)
803 iter++;
805 if(!iter->id || !iter->exec)
806 return OLECMDERR_E_NOTSUPPORTED;
808 return iter->exec(This, cmdexecopt, in, out);
811 static HRESULT WINAPI OleCommandTarget_Exec(IOleCommandTarget *iface, const GUID *pguidCmdGroup,
812 DWORD nCmdID, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
814 HTMLDocument *This = CMDTARGET_THIS(iface);
816 if(!pguidCmdGroup) {
817 if(nCmdID<OLECMDID_OPEN || nCmdID>OLECMDID_GETPRINTTEMPLATE || !exec_table[nCmdID].func) {
818 WARN("Unsupported cmdID = %d\n", nCmdID);
819 return OLECMDERR_E_NOTSUPPORTED;
822 return exec_table[nCmdID].func(This, nCmdexecopt, pvaIn, pvaOut);
823 }else if(IsEqualGUID(&CGID_Explorer, pguidCmdGroup)) {
824 FIXME("unsupported nCmdID %d of CGID_Explorer group\n", nCmdID);
825 TRACE("%p %p\n", pvaIn, pvaOut);
826 return OLECMDERR_E_NOTSUPPORTED;
827 }else if(IsEqualGUID(&CGID_ShellDocView, pguidCmdGroup)) {
828 FIXME("unsupported nCmdID %d of CGID_ShellDocView group\n", nCmdID);
829 return OLECMDERR_E_NOTSUPPORTED;
830 }else if(IsEqualGUID(&CGID_MSHTML, pguidCmdGroup)) {
831 HRESULT hres = exec_from_table(This, base_cmds, nCmdID, nCmdexecopt, pvaIn, pvaOut);
832 if(hres == OLECMDERR_E_NOTSUPPORTED)
833 hres = exec_from_table(This, editmode_cmds, nCmdID,
834 nCmdexecopt, pvaIn, pvaOut);
835 if(hres == OLECMDERR_E_NOTSUPPORTED)
836 FIXME("unsupported nCmdID %d of CGID_MSHTML group\n", nCmdID);
838 return hres;
841 FIXME("Unsupported pguidCmdGroup %s\n", debugstr_guid(pguidCmdGroup));
842 return OLECMDERR_E_UNKNOWNGROUP;
845 #undef CMDTARGET_THIS
847 static const IOleCommandTargetVtbl OleCommandTargetVtbl = {
848 OleCommandTarget_QueryInterface,
849 OleCommandTarget_AddRef,
850 OleCommandTarget_Release,
851 OleCommandTarget_QueryStatus,
852 OleCommandTarget_Exec
855 void show_context_menu(HTMLDocument *This, DWORD dwID, POINT *ppt, IDispatch *elem)
857 HMENU menu_res, menu;
858 DWORD cmdid;
859 HRESULT hres;
861 hres = IDocHostUIHandler_ShowContextMenu(This->hostui, dwID, ppt,
862 (IUnknown*)CMDTARGET(This), elem);
863 if(hres == S_OK)
864 return;
866 menu_res = LoadMenuW(get_shdoclc(), MAKEINTRESOURCEW(IDR_BROWSE_CONTEXT_MENU));
867 menu = GetSubMenu(menu_res, dwID);
869 cmdid = TrackPopupMenu(menu, TPM_LEFTALIGN | TPM_RIGHTBUTTON | TPM_RETURNCMD,
870 ppt->x, ppt->y, 0, This->hwnd, NULL);
871 DestroyMenu(menu_res);
873 if(cmdid)
874 IOleCommandTarget_Exec(CMDTARGET(This), &CGID_MSHTML, cmdid, 0, NULL, NULL);
877 void HTMLDocument_OleCmd_Init(HTMLDocument *This)
879 This->lpOleCommandTargetVtbl = &OleCommandTargetVtbl;