d3d9/tests: Move a Present out of the loops in fog_with_shader_test().
[wine/multimedia.git] / dlls / mshtml / olecmd.c
blobf4bf002cef91b201c7b89ca730c46951152127d1
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 <stdarg.h>
21 #define COBJMACROS
23 #include "windef.h"
24 #include "winbase.h"
25 #include "winuser.h"
26 #include "ole2.h"
27 #include "shlguid.h"
28 #include "mshtmdid.h"
29 #include "idispids.h"
30 #include "mshtmcid.h"
32 #include "wine/debug.h"
34 #include "mshtml_private.h"
35 #include "binding.h"
36 #include "resource.h"
38 WINE_DEFAULT_DEBUG_CHANNEL(mshtml);
40 #define NSCMD_COPY "cmd_copy"
42 void do_ns_command(HTMLDocument *This, const char *cmd, nsICommandParams *nsparam)
44 nsICommandManager *cmdmgr;
45 nsresult nsres;
47 TRACE("(%p)\n", This);
49 if(!This->doc_obj || !This->doc_obj->nscontainer)
50 return;
52 nsres = get_nsinterface((nsISupports*)This->doc_obj->nscontainer->webbrowser, &IID_nsICommandManager, (void**)&cmdmgr);
53 if(NS_FAILED(nsres)) {
54 ERR("Could not get nsICommandManager: %08x\n", nsres);
55 return;
58 nsres = nsICommandManager_DoCommand(cmdmgr, cmd, nsparam, This->window->nswindow);
59 if(NS_FAILED(nsres))
60 ERR("DoCommand(%s) failed: %08x\n", debugstr_a(cmd), nsres);
62 nsICommandManager_Release(cmdmgr);
65 static nsIClipboardCommands *get_clipboard_commands(HTMLDocument *doc)
67 nsIClipboardCommands *clipboard_commands;
68 nsIDocShell *doc_shell;
69 nsresult nsres;
71 nsres = get_nsinterface((nsISupports*)doc->window->nswindow, &IID_nsIDocShell, (void**)&doc_shell);
72 if(NS_FAILED(nsres)) {
73 ERR("Could not get nsIDocShell interface\n");
74 return NULL;
77 nsres = nsIDocShell_QueryInterface(doc_shell, &IID_nsIClipboardCommands, (void**)&clipboard_commands);
78 nsIDocShell_Release(doc_shell);
79 if(NS_FAILED(nsres)) {
80 ERR("Could not get nsIClipboardCommands interface\n");
81 return NULL;
84 return clipboard_commands;
87 /**********************************************************
88 * IOleCommandTarget implementation
91 static inline HTMLDocument *impl_from_IOleCommandTarget(IOleCommandTarget *iface)
93 return CONTAINING_RECORD(iface, HTMLDocument, IOleCommandTarget_iface);
96 static HRESULT exec_open(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 HRESULT exec_new(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
104 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
105 return E_NOTIMPL;
108 static HRESULT exec_save(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
110 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
111 return E_NOTIMPL;
114 static HRESULT exec_save_as(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
116 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
117 return E_NOTIMPL;
120 static HRESULT exec_save_copy_as(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
122 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
123 return E_NOTIMPL;
126 static nsresult set_head_text(nsIPrintSettings *settings, LPCWSTR template, BOOL head, int pos)
128 if(head) {
129 switch(pos) {
130 case 0:
131 return nsIPrintSettings_SetHeaderStrLeft(settings, template);
132 case 1:
133 return nsIPrintSettings_SetHeaderStrRight(settings, template);
134 case 2:
135 return nsIPrintSettings_SetHeaderStrCenter(settings, template);
137 }else {
138 switch(pos) {
139 case 0:
140 return nsIPrintSettings_SetFooterStrLeft(settings, template);
141 case 1:
142 return nsIPrintSettings_SetFooterStrRight(settings, template);
143 case 2:
144 return nsIPrintSettings_SetFooterStrCenter(settings, template);
148 return NS_OK;
151 static void set_print_template(nsIPrintSettings *settings, LPCWSTR template, BOOL head)
153 PRUnichar nstemplate[200]; /* FIXME: Use dynamic allocation */
154 PRUnichar *p = nstemplate;
155 LPCWSTR ptr=template;
156 int pos=0;
158 while(*ptr) {
159 if(*ptr != '&') {
160 *p++ = *ptr++;
161 continue;
164 switch(*++ptr) {
165 case '&':
166 *p++ = '&';
167 *p++ = '&';
168 ptr++;
169 break;
170 case 'b': /* change align */
171 ptr++;
172 *p = 0;
173 set_head_text(settings, nstemplate, head, pos);
174 p = nstemplate;
175 pos++;
176 break;
177 case 'd': { /* short date */
178 SYSTEMTIME systime;
179 GetLocalTime(&systime);
180 GetDateFormatW(LOCALE_SYSTEM_DEFAULT, 0, &systime, NULL, p,
181 sizeof(nstemplate)-(p-nstemplate)*sizeof(WCHAR));
182 p += strlenW(p);
183 ptr++;
184 break;
186 case 'p': /* page number */
187 *p++ = '&';
188 *p++ = 'P';
189 ptr++;
190 break;
191 case 'P': /* page count */
192 *p++ = '?'; /* FIXME */
193 ptr++;
194 break;
195 case 'u':
196 *p++ = '&';
197 *p++ = 'U';
198 ptr++;
199 break;
200 case 'w':
201 /* FIXME: set window title */
202 ptr++;
203 break;
204 default:
205 *p++ = '&';
206 *p++ = *ptr++;
210 *p = 0;
211 set_head_text(settings, nstemplate, head, pos);
213 while(++pos < 3)
214 set_head_text(settings, p, head, pos);
217 static void set_default_templates(nsIPrintSettings *settings)
219 WCHAR buf[64];
221 static const PRUnichar empty[] = {0};
223 nsIPrintSettings_SetHeaderStrLeft(settings, empty);
224 nsIPrintSettings_SetHeaderStrRight(settings, empty);
225 nsIPrintSettings_SetHeaderStrCenter(settings, empty);
226 nsIPrintSettings_SetFooterStrLeft(settings, empty);
227 nsIPrintSettings_SetFooterStrRight(settings, empty);
228 nsIPrintSettings_SetFooterStrCenter(settings, empty);
230 if(LoadStringW(get_shdoclc(), IDS_PRINT_HEADER_TEMPLATE, buf,
231 sizeof(buf)/sizeof(WCHAR)))
232 set_print_template(settings, buf, TRUE);
235 if(LoadStringW(get_shdoclc(), IDS_PRINT_FOOTER_TEMPLATE, buf,
236 sizeof(buf)/sizeof(WCHAR)))
237 set_print_template(settings, buf, FALSE);
241 static HRESULT exec_print(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
243 nsIWebBrowserPrint *nsprint;
244 nsIPrintSettings *settings;
245 nsresult nsres;
247 TRACE("(%p)->(%d %s %p)\n", This, nCmdexecopt, debugstr_variant(pvaIn), pvaOut);
249 if(pvaOut)
250 FIXME("unsupported pvaOut\n");
252 if(!This->doc_obj->nscontainer)
253 return S_OK;
255 nsres = get_nsinterface((nsISupports*)This->doc_obj->nscontainer->webbrowser, &IID_nsIWebBrowserPrint,
256 (void**)&nsprint);
257 if(NS_FAILED(nsres)) {
258 ERR("Could not get nsIWebBrowserPrint: %08x\n", nsres);
259 return S_OK;
262 nsres = nsIWebBrowserPrint_GetGlobalPrintSettings(nsprint, &settings);
263 if(NS_FAILED(nsres))
264 ERR("GetCurrentPrintSettings failed: %08x\n", nsres);
266 set_default_templates(settings);
268 if(pvaIn) {
269 switch(V_VT(pvaIn)) {
270 case VT_BYREF|VT_ARRAY: {
271 VARIANT *opts;
272 DWORD opts_cnt;
274 if(V_ARRAY(pvaIn)->cDims != 1)
275 WARN("cDims = %d\n", V_ARRAY(pvaIn)->cDims);
277 SafeArrayAccessData(V_ARRAY(pvaIn), (void**)&opts);
278 opts_cnt = V_ARRAY(pvaIn)->rgsabound[0].cElements;
280 if(opts_cnt >= 1) {
281 switch(V_VT(opts)) {
282 case VT_BSTR:
283 TRACE("setting footer %s\n", debugstr_w(V_BSTR(opts)));
284 set_print_template(settings, V_BSTR(opts), TRUE);
285 break;
286 case VT_NULL:
287 break;
288 default:
289 WARN("opts = %s\n", debugstr_variant(opts));
293 if(opts_cnt >= 2) {
294 switch(V_VT(opts+1)) {
295 case VT_BSTR:
296 TRACE("setting footer %s\n", debugstr_w(V_BSTR(opts+1)));
297 set_print_template(settings, V_BSTR(opts+1), FALSE);
298 break;
299 case VT_NULL:
300 break;
301 default:
302 WARN("opts[1] = %s\n", debugstr_variant(opts+1));
306 if(opts_cnt >= 3)
307 FIXME("Unsupported opts_cnt %d\n", opts_cnt);
309 SafeArrayUnaccessData(V_ARRAY(pvaIn));
310 break;
312 default:
313 FIXME("unsupported arg %s\n", debugstr_variant(pvaIn));
317 nsres = nsIWebBrowserPrint_Print(nsprint, settings, NULL);
318 if(NS_FAILED(nsres))
319 ERR("Print failed: %08x\n", nsres);
321 nsIWebBrowserPrint_Release(nsprint);
323 return S_OK;
326 static HRESULT exec_print_preview(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_page_setup(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_spell(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_properties(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_cut(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_copy(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_paste(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_paste_special(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_undo(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_rendo(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_select_all(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_clear_selection(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_zoom(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_get_zoom_range(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
406 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
407 return E_NOTIMPL;
410 typedef struct {
411 task_t header;
412 HTMLOuterWindow *window;
413 }refresh_task_t;
415 static void refresh_proc(task_t *_task)
417 refresh_task_t *task = (refresh_task_t*)_task;
418 HTMLOuterWindow *window = task->window;
420 TRACE("%p\n", window);
422 window->readystate = READYSTATE_UNINITIALIZED;
424 if(window->doc_obj && window->doc_obj->client_cmdtrg) {
425 VARIANT var;
427 V_VT(&var) = VT_I4;
428 V_I4(&var) = 0;
429 IOleCommandTarget_Exec(window->doc_obj->client_cmdtrg, &CGID_ShellDocView, 37, 0, &var, NULL);
432 load_uri(task->window, task->window->uri, BINDING_REFRESH|BINDING_NOFRAG);
435 static void refresh_destr(task_t *_task)
437 refresh_task_t *task = (refresh_task_t*)_task;
439 IHTMLWindow2_Release(&task->window->base.IHTMLWindow2_iface);
440 heap_free(task);
443 static HRESULT exec_refresh(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
445 refresh_task_t *task;
446 HRESULT hres;
448 TRACE("(%p)->(%d %s %p)\n", This, nCmdexecopt, debugstr_variant(pvaIn), pvaOut);
450 if(This->doc_obj->client) {
451 IOleCommandTarget *olecmd;
453 hres = IOleClientSite_QueryInterface(This->doc_obj->client, &IID_IOleCommandTarget, (void**)&olecmd);
454 if(SUCCEEDED(hres)) {
455 hres = IOleCommandTarget_Exec(olecmd, &CGID_DocHostCommandHandler, 2300, nCmdexecopt, pvaIn, pvaOut);
456 IOleCommandTarget_Release(olecmd);
457 if(SUCCEEDED(hres))
458 return S_OK;
462 if(!This->window)
463 return E_UNEXPECTED;
465 task = heap_alloc(sizeof(*task));
466 if(!task)
467 return E_OUTOFMEMORY;
469 IHTMLWindow2_AddRef(&This->window->base.IHTMLWindow2_iface);
470 task->window = This->window;
472 return push_task(&task->header, refresh_proc, refresh_destr, This->window->task_magic);
475 static HRESULT exec_stop(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
477 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
478 return E_NOTIMPL;
481 static HRESULT exec_stop_download(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
483 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
484 return E_NOTIMPL;
487 static HRESULT exec_find(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
489 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
490 return E_NOTIMPL;
493 static HRESULT exec_delete(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
495 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
496 return E_NOTIMPL;
499 static HRESULT exec_enable_interaction(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
501 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
502 return E_NOTIMPL;
505 static HRESULT exec_on_unload(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
507 TRACE("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
509 /* Tests show that we have nothing more to do here */
511 if(pvaOut) {
512 V_VT(pvaOut) = VT_BOOL;
513 V_BOOL(pvaOut) = VARIANT_TRUE;
516 return S_OK;
519 static HRESULT exec_show_page_setup(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
521 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
522 return E_NOTIMPL;
525 static HRESULT exec_show_print(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
527 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
528 return E_NOTIMPL;
531 static HRESULT exec_close(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
533 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
534 return E_NOTIMPL;
537 static HRESULT exec_set_print_template(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
539 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
540 return E_NOTIMPL;
543 static HRESULT exec_get_print_template(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
545 FIXME("(%p)->(%d %p %p)\n", This, nCmdexecopt, pvaIn, pvaOut);
546 return E_NOTIMPL;
549 static HRESULT exec_optical_zoom(HTMLDocument *This, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
551 TRACE("(%p)->(%d %s %p)\n", This, nCmdexecopt, debugstr_variant(pvaIn), pvaOut);
553 if(!pvaIn || V_VT(pvaIn) != VT_I4) {
554 FIXME("Unsupported argument %s\n", debugstr_variant(pvaIn));
555 return E_NOTIMPL;
558 set_viewer_zoom(This->doc_obj->nscontainer, (float)V_I4(pvaIn)/100);
559 return S_OK;
562 static HRESULT query_mshtml_copy(HTMLDocument *This, OLECMD *cmd)
564 FIXME("(%p)\n", This);
565 cmd->cmdf = OLECMDF_SUPPORTED|OLECMDF_ENABLED;
566 return S_OK;
569 static HRESULT exec_mshtml_copy(HTMLDocument *This, DWORD cmdexecopt, VARIANT *in, VARIANT *out)
571 TRACE("(%p)->(%08x %p %p)\n", This, cmdexecopt, in, out);
573 if(This->doc_obj->usermode == EDITMODE)
574 return editor_exec_copy(This, cmdexecopt, in, out);
576 do_ns_command(This, NSCMD_COPY, NULL);
577 return S_OK;
580 static HRESULT query_mshtml_cut(HTMLDocument *This, OLECMD *cmd)
582 FIXME("(%p)\n", This);
583 cmd->cmdf = OLECMDF_SUPPORTED|OLECMDF_ENABLED;
584 return S_OK;
587 static HRESULT exec_mshtml_cut(HTMLDocument *This, DWORD cmdexecopt, VARIANT *in, VARIANT *out)
589 nsIClipboardCommands *clipboard_commands;
590 nsresult nsres;
592 TRACE("(%p)->(%08x %p %p)\n", This, cmdexecopt, in, out);
594 if(This->doc_obj->usermode == EDITMODE)
595 return editor_exec_cut(This, cmdexecopt, in, out);
597 clipboard_commands = get_clipboard_commands(This);
598 if(!clipboard_commands)
599 return E_UNEXPECTED;
601 nsres = nsIClipboardCommands_CutSelection(clipboard_commands);
602 nsIClipboardCommands_Release(clipboard_commands);
603 if(NS_FAILED(nsres)) {
604 ERR("Paste failed: %08x\n", nsres);
605 return E_FAIL;
608 return S_OK;
611 static HRESULT query_mshtml_paste(HTMLDocument *This, OLECMD *cmd)
613 FIXME("(%p)\n", This);
614 cmd->cmdf = OLECMDF_SUPPORTED|OLECMDF_ENABLED;
615 return S_OK;
618 static HRESULT exec_mshtml_paste(HTMLDocument *This, DWORD cmdexecopt, VARIANT *in, VARIANT *out)
620 nsIClipboardCommands *clipboard_commands;
621 nsresult nsres;
623 TRACE("(%p)->(%08x %p %p)\n", This, cmdexecopt, in, out);
625 if(This->doc_obj->usermode == EDITMODE)
626 return editor_exec_paste(This, cmdexecopt, in, out);
628 clipboard_commands = get_clipboard_commands(This);
629 if(!clipboard_commands)
630 return E_UNEXPECTED;
632 nsres = nsIClipboardCommands_Paste(clipboard_commands);
633 nsIClipboardCommands_Release(clipboard_commands);
634 if(NS_FAILED(nsres)) {
635 ERR("Paste failed: %08x\n", nsres);
636 return E_FAIL;
639 return S_OK;
642 static HRESULT exec_browsemode(HTMLDocument *This, DWORD cmdexecopt, VARIANT *in, VARIANT *out)
644 WARN("(%p)->(%08x %p %p)\n", This, cmdexecopt, in, out);
646 if(in || out)
647 FIXME("unsupported args\n");
649 This->doc_obj->usermode = BROWSEMODE;
651 return S_OK;
654 static HRESULT exec_editmode(HTMLDocument *This, DWORD cmdexecopt, VARIANT *in, VARIANT *out)
656 TRACE("(%p)->(%08x %p %p)\n", This, cmdexecopt, in, out);
658 if(in || out)
659 FIXME("unsupported args\n");
661 return setup_edit_mode(This->doc_obj);
664 static HRESULT exec_htmleditmode(HTMLDocument *This, DWORD cmdexecopt, VARIANT *in, VARIANT *out)
666 FIXME("(%p)->(%08x %p %p)\n", This, cmdexecopt, in, out);
667 return S_OK;
670 static HRESULT exec_baselinefont3(HTMLDocument *This, DWORD cmdexecopt, VARIANT *in, VARIANT *out)
672 FIXME("(%p)->(%08x %p %p)\n", This, cmdexecopt, in, out);
673 return S_OK;
676 static HRESULT exec_respectvisibility_indesign(HTMLDocument *This, DWORD cmdexecopt,
677 VARIANT *in, VARIANT *out)
679 TRACE("(%p)->(%x %s %p)\n", This, cmdexecopt, debugstr_variant(in), out);
681 /* This is turned on by default in Gecko. */
682 if(!in || V_VT(in) != VT_BOOL || !V_BOOL(in))
683 FIXME("Unsupported argument %s\n", debugstr_variant(in));
685 return S_OK;
688 static HRESULT query_enabled_stub(HTMLDocument *This, OLECMD *cmd)
690 switch(cmd->cmdID) {
691 case IDM_PRINT:
692 FIXME("CGID_MSHTML: IDM_PRINT\n");
693 cmd->cmdf = OLECMDF_SUPPORTED|OLECMDF_ENABLED;
694 break;
695 case IDM_BLOCKDIRLTR:
696 FIXME("CGID_MSHTML: IDM_BLOCKDIRLTR\n");
697 cmd->cmdf = OLECMDF_SUPPORTED|OLECMDF_ENABLED;
698 break;
699 case IDM_BLOCKDIRRTL:
700 FIXME("CGID_MSHTML: IDM_BLOCKDIRRTL\n");
701 cmd->cmdf = OLECMDF_SUPPORTED|OLECMDF_ENABLED;
702 break;
705 return S_OK;
708 static const struct {
709 OLECMDF cmdf;
710 HRESULT (*func)(HTMLDocument*,DWORD,VARIANT*,VARIANT*);
711 } exec_table[] = {
712 {0},
713 { OLECMDF_SUPPORTED, exec_open }, /* OLECMDID_OPEN */
714 { OLECMDF_SUPPORTED, exec_new }, /* OLECMDID_NEW */
715 { OLECMDF_SUPPORTED, exec_save }, /* OLECMDID_SAVE */
716 { OLECMDF_SUPPORTED|OLECMDF_ENABLED, exec_save_as }, /* OLECMDID_SAVEAS */
717 { OLECMDF_SUPPORTED, exec_save_copy_as }, /* OLECMDID_SAVECOPYAS */
718 { OLECMDF_SUPPORTED|OLECMDF_ENABLED, exec_print }, /* OLECMDID_PRINT */
719 { OLECMDF_SUPPORTED|OLECMDF_ENABLED, exec_print_preview }, /* OLECMDID_PRINTPREVIEW */
720 { OLECMDF_SUPPORTED|OLECMDF_ENABLED, exec_page_setup }, /* OLECMDID_PAGESETUP */
721 { OLECMDF_SUPPORTED, exec_spell }, /* OLECMDID_SPELL */
722 { OLECMDF_SUPPORTED|OLECMDF_ENABLED, exec_properties }, /* OLECMDID_PROPERTIES */
723 { OLECMDF_SUPPORTED, exec_cut }, /* OLECMDID_CUT */
724 { OLECMDF_SUPPORTED, exec_copy }, /* OLECMDID_COPY */
725 { OLECMDF_SUPPORTED, exec_paste }, /* OLECMDID_PASTE */
726 { OLECMDF_SUPPORTED, exec_paste_special }, /* OLECMDID_PASTESPECIAL */
727 { OLECMDF_SUPPORTED, exec_undo }, /* OLECMDID_UNDO */
728 { OLECMDF_SUPPORTED, exec_rendo }, /* OLECMDID_REDO */
729 { OLECMDF_SUPPORTED|OLECMDF_ENABLED, exec_select_all }, /* OLECMDID_SELECTALL */
730 { OLECMDF_SUPPORTED, exec_clear_selection }, /* OLECMDID_CLEARSELECTION */
731 { OLECMDF_SUPPORTED, exec_zoom }, /* OLECMDID_ZOOM */
732 { OLECMDF_SUPPORTED, exec_get_zoom_range }, /* OLECMDID_GETZOOMRANGE */
733 {0},
734 { OLECMDF_SUPPORTED|OLECMDF_ENABLED, exec_refresh }, /* OLECMDID_REFRESH */
735 { OLECMDF_SUPPORTED|OLECMDF_ENABLED, exec_stop }, /* OLECMDID_STOP */
736 {0},{0},{0},{0},{0},{0},
737 { OLECMDF_SUPPORTED, exec_stop_download }, /* OLECMDID_STOPDOWNLOAD */
738 {0},
739 { OLECMDF_SUPPORTED|OLECMDF_ENABLED, exec_find }, /* OLECMDID_FIND */
740 { OLECMDF_SUPPORTED, exec_delete }, /* OLECMDID_DELETE */
741 {0},{0},
742 { OLECMDF_SUPPORTED, exec_enable_interaction }, /* OLECMDID_ENABLE_INTERACTION */
743 { OLECMDF_SUPPORTED, exec_on_unload }, /* OLECMDID_ONUNLOAD */
744 {0},{0},{0},{0},{0},
745 { OLECMDF_SUPPORTED, exec_show_page_setup }, /* OLECMDID_SHOWPAGESETUP */
746 { OLECMDF_SUPPORTED, exec_show_print }, /* OLECMDID_SHOWPRINT */
747 {0},{0},
748 { OLECMDF_SUPPORTED, exec_close }, /* OLECMDID_CLOSE */
749 {0},{0},{0},
750 { OLECMDF_SUPPORTED, exec_set_print_template }, /* OLECMDID_SETPRINTTEMPLATE */
751 { OLECMDF_SUPPORTED, exec_get_print_template }, /* OLECMDID_GETPRINTTEMPLATE */
752 {0},{0},{0},{0},{0},{0},{0},{0},{0},{0},
753 { 0, /* not reported as supported */ exec_optical_zoom } /* OLECMDID_OPTICAL_ZOOM */
756 static const cmdtable_t base_cmds[] = {
757 {IDM_COPY, query_mshtml_copy, exec_mshtml_copy},
758 {IDM_PASTE, query_mshtml_paste, exec_mshtml_paste},
759 {IDM_CUT, query_mshtml_cut, exec_mshtml_cut},
760 {IDM_BROWSEMODE, NULL, exec_browsemode},
761 {IDM_EDITMODE, NULL, exec_editmode},
762 {IDM_PRINT, query_enabled_stub, exec_print},
763 {IDM_HTMLEDITMODE, NULL, exec_htmleditmode},
764 {IDM_BASELINEFONT3, NULL, exec_baselinefont3},
765 {IDM_BLOCKDIRLTR, query_enabled_stub, NULL},
766 {IDM_BLOCKDIRRTL, query_enabled_stub, NULL},
767 {IDM_RESPECTVISIBILITY_INDESIGN, NULL, exec_respectvisibility_indesign},
768 {0,NULL,NULL}
771 static HRESULT WINAPI OleCommandTarget_QueryInterface(IOleCommandTarget *iface, REFIID riid, void **ppv)
773 HTMLDocument *This = impl_from_IOleCommandTarget(iface);
774 return htmldoc_query_interface(This, riid, ppv);
777 static ULONG WINAPI OleCommandTarget_AddRef(IOleCommandTarget *iface)
779 HTMLDocument *This = impl_from_IOleCommandTarget(iface);
780 return htmldoc_addref(This);
783 static ULONG WINAPI OleCommandTarget_Release(IOleCommandTarget *iface)
785 HTMLDocument *This = impl_from_IOleCommandTarget(iface);
786 return htmldoc_release(This);
789 static HRESULT query_from_table(HTMLDocument *This, const cmdtable_t *cmdtable, OLECMD *cmd)
791 const cmdtable_t *iter = cmdtable;
793 cmd->cmdf = 0;
795 while(iter->id && iter->id != cmd->cmdID)
796 iter++;
798 if(!iter->id || !iter->query)
799 return OLECMDERR_E_NOTSUPPORTED;
801 return iter->query(This, cmd);
804 static HRESULT WINAPI OleCommandTarget_QueryStatus(IOleCommandTarget *iface, const GUID *pguidCmdGroup,
805 ULONG cCmds, OLECMD prgCmds[], OLECMDTEXT *pCmdText)
807 HTMLDocument *This = impl_from_IOleCommandTarget(iface);
808 HRESULT hres;
810 TRACE("(%p)->(%s %d %p %p)\n", This, debugstr_guid(pguidCmdGroup), cCmds, prgCmds, pCmdText);
812 if(pCmdText)
813 FIXME("Unsupported pCmdText\n");
814 if(!cCmds)
815 return S_OK;
817 if(!pguidCmdGroup) {
818 ULONG i;
820 for(i=0; i<cCmds; i++) {
821 if(prgCmds[i].cmdID < OLECMDID_OPEN || prgCmds[i].cmdID >= sizeof(exec_table)/sizeof(*exec_table)) {
822 WARN("Unsupported cmdID = %d\n", prgCmds[i].cmdID);
823 prgCmds[i].cmdf = 0;
824 }else {
825 if(prgCmds[i].cmdID == OLECMDID_OPEN || prgCmds[i].cmdID == OLECMDID_NEW) {
826 IOleCommandTarget *cmdtrg = NULL;
827 OLECMD olecmd;
829 prgCmds[i].cmdf = OLECMDF_SUPPORTED;
830 if(This->doc_obj->client) {
831 hres = IOleClientSite_QueryInterface(This->doc_obj->client, &IID_IOleCommandTarget,
832 (void**)&cmdtrg);
833 if(SUCCEEDED(hres)) {
834 olecmd.cmdID = prgCmds[i].cmdID;
835 olecmd.cmdf = 0;
837 hres = IOleCommandTarget_QueryStatus(cmdtrg, NULL, 1, &olecmd, NULL);
838 if(SUCCEEDED(hres) && olecmd.cmdf)
839 prgCmds[i].cmdf = olecmd.cmdf;
841 }else {
842 ERR("This->client == NULL, native would crash\n");
844 }else {
845 prgCmds[i].cmdf = exec_table[prgCmds[i].cmdID].cmdf;
846 TRACE("cmdID = %d returning %x\n", prgCmds[i].cmdID, prgCmds[i].cmdf);
851 return (prgCmds[cCmds-1].cmdf & OLECMDF_SUPPORTED) ? S_OK : OLECMDERR_E_NOTSUPPORTED;
854 if(IsEqualGUID(&CGID_MSHTML, pguidCmdGroup)) {
855 ULONG i;
857 for(i=0; i<cCmds; i++) {
858 hres = query_from_table(This, base_cmds, prgCmds+i);
859 if(hres == OLECMDERR_E_NOTSUPPORTED)
860 hres = query_from_table(This, editmode_cmds, prgCmds+i);
861 if(hres == OLECMDERR_E_NOTSUPPORTED)
862 FIXME("CGID_MSHTML: unsupported cmdID %d\n", prgCmds[i].cmdID);
865 return (prgCmds[cCmds-1].cmdf & OLECMDF_SUPPORTED) ? S_OK : OLECMDERR_E_NOTSUPPORTED;
868 FIXME("Unsupported pguidCmdGroup %s\n", debugstr_guid(pguidCmdGroup));
869 return OLECMDERR_E_UNKNOWNGROUP;
872 static HRESULT exec_from_table(HTMLDocument *This, const cmdtable_t *cmdtable, DWORD cmdid,
873 DWORD cmdexecopt, VARIANT *in, VARIANT *out)
875 const cmdtable_t *iter = cmdtable;
877 while(iter->id && iter->id != cmdid)
878 iter++;
880 if(!iter->id || !iter->exec)
881 return OLECMDERR_E_NOTSUPPORTED;
883 return iter->exec(This, cmdexecopt, in, out);
886 static HRESULT WINAPI OleCommandTarget_Exec(IOleCommandTarget *iface, const GUID *pguidCmdGroup,
887 DWORD nCmdID, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
889 HTMLDocument *This = impl_from_IOleCommandTarget(iface);
891 if(!pguidCmdGroup) {
892 if(nCmdID < OLECMDID_OPEN || nCmdID >= sizeof(exec_table)/sizeof(*exec_table) || !exec_table[nCmdID].func) {
893 WARN("Unsupported cmdID = %d\n", nCmdID);
894 return OLECMDERR_E_NOTSUPPORTED;
897 return exec_table[nCmdID].func(This, nCmdexecopt, pvaIn, pvaOut);
898 }else if(IsEqualGUID(&CGID_Explorer, pguidCmdGroup)) {
899 FIXME("unsupported nCmdID %d of CGID_Explorer group\n", nCmdID);
900 TRACE("%p %p\n", pvaIn, pvaOut);
901 return OLECMDERR_E_NOTSUPPORTED;
902 }else if(IsEqualGUID(&CGID_ShellDocView, pguidCmdGroup)) {
903 FIXME("unsupported nCmdID %d of CGID_ShellDocView group\n", nCmdID);
904 return OLECMDERR_E_NOTSUPPORTED;
905 }else if(IsEqualGUID(&CGID_MSHTML, pguidCmdGroup)) {
906 HRESULT hres = exec_from_table(This, base_cmds, nCmdID, nCmdexecopt, pvaIn, pvaOut);
907 if(hres == OLECMDERR_E_NOTSUPPORTED)
908 hres = exec_from_table(This, editmode_cmds, nCmdID,
909 nCmdexecopt, pvaIn, pvaOut);
910 if(hres == OLECMDERR_E_NOTSUPPORTED)
911 FIXME("unsupported nCmdID %d of CGID_MSHTML group\n", nCmdID);
913 return hres;
916 FIXME("Unsupported pguidCmdGroup %s\n", debugstr_guid(pguidCmdGroup));
917 return OLECMDERR_E_UNKNOWNGROUP;
920 static const IOleCommandTargetVtbl OleCommandTargetVtbl = {
921 OleCommandTarget_QueryInterface,
922 OleCommandTarget_AddRef,
923 OleCommandTarget_Release,
924 OleCommandTarget_QueryStatus,
925 OleCommandTarget_Exec
928 void show_context_menu(HTMLDocumentObj *This, DWORD dwID, POINT *ppt, IDispatch *elem)
930 HMENU menu_res, menu;
931 DWORD cmdid;
933 if(This->hostui && S_OK == IDocHostUIHandler_ShowContextMenu(This->hostui,
934 dwID, ppt, (IUnknown*)&This->basedoc.IOleCommandTarget_iface, elem))
935 return;
937 menu_res = LoadMenuW(get_shdoclc(), MAKEINTRESOURCEW(IDR_BROWSE_CONTEXT_MENU));
938 menu = GetSubMenu(menu_res, dwID);
940 cmdid = TrackPopupMenu(menu, TPM_LEFTALIGN | TPM_RIGHTBUTTON | TPM_RETURNCMD,
941 ppt->x, ppt->y, 0, This->hwnd, NULL);
942 DestroyMenu(menu_res);
944 if(cmdid)
945 IOleCommandTarget_Exec(&This->basedoc.IOleCommandTarget_iface, &CGID_MSHTML, cmdid, 0,
946 NULL, NULL);
949 void HTMLDocument_OleCmd_Init(HTMLDocument *This)
951 This->IOleCommandTarget_iface.lpVtbl = &OleCommandTargetVtbl;