2 * Copyright 2010 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
32 #include "mshtml_private.h"
33 #include "pluginhost.h"
35 #include "wine/debug.h"
37 WINE_DEFAULT_DEBUG_CHANNEL(mshtml);
39 const IID IID_HTMLPluginContainer =
40 {0xbd7a6050,0xb373,0x4f6f,{0xa4,0x93,0xdd,0x40,0xc5,0x23,0xa8,0x6a}};
42 static BOOL check_load_safety(PluginHost *host)
44 DWORD policy_size, policy;
45 struct CONFIRMSAFETY cs;
49 cs.clsid = host->clsid;
50 cs.pUnk = host->plugin_unk;
51 cs.dwFlags = CONFIRMSAFETYACTION_LOADOBJECT;
53 hres = IInternetHostSecurityManager_QueryCustomPolicy(&host->doc->IInternetHostSecurityManager_iface,
54 &GUID_CUSTOM_CONFIRMOBJECTSAFETY, &ppolicy, &policy_size, (BYTE*)&cs, sizeof(cs), 0);
58 policy = *(DWORD*)ppolicy;
59 CoTaskMemFree(ppolicy);
60 return policy == URLPOLICY_ALLOW;
63 static BOOL check_script_safety(PluginHost *host)
65 DISPPARAMS params = {NULL,NULL,0,0};
66 DWORD policy_size, policy;
67 struct CONFIRMSAFETY cs;
73 cs.clsid = host->clsid;
74 cs.pUnk = host->plugin_unk;
77 hres = IInternetHostSecurityManager_QueryCustomPolicy(&host->doc->IInternetHostSecurityManager_iface,
78 &GUID_CUSTOM_CONFIRMOBJECTSAFETY, &ppolicy, &policy_size, (BYTE*)&cs, sizeof(cs), 0);
82 policy = *(DWORD*)ppolicy;
83 CoTaskMemFree(ppolicy);
85 if(policy != URLPOLICY_ALLOW)
89 hres = IDispatch_Invoke(host->disp, DISPID_SECURITYCTX, &IID_NULL, 0, DISPATCH_PROPERTYGET, ¶ms, &v, NULL, &err);
91 FIXME("Handle security ctx %s\n", debugstr_variant(&v));
98 static void update_readystate(PluginHost *host)
100 DISPPARAMS params = {NULL,NULL,0,0};
107 hres = IUnknown_QueryInterface(host->plugin_unk, &IID_IDispatchEx, (void**)&dispex);
108 if(SUCCEEDED(hres)) {
109 FIXME("Use IDispatchEx\n");
110 IDispatchEx_Release(dispex);
113 hres = IUnknown_QueryInterface(host->plugin_unk, &IID_IDispatch, (void**)&disp);
117 hres = IDispatch_Invoke(disp, DISPID_READYSTATE, &IID_NULL, 0, DISPATCH_PROPERTYGET, ¶ms, &v, NULL, &err);
118 IDispatch_Release(disp);
119 if(SUCCEEDED(hres)) {
120 /* FIXME: make plugin readystate affect document readystate */
121 TRACE("readystate = %s\n", debugstr_variant(&v));
126 /* FIXME: We shouldn't need this function and we should embed plugin directly in the main document */
127 static void get_pos_rect(PluginHost *host, RECT *ret)
131 ret->bottom = host->rect.bottom - host->rect.top;
132 ret->right = host->rect.right - host->rect.left;
135 static void load_prop_bag(PluginHost *host, IPersistPropertyBag *persist_prop_bag)
137 IPropertyBag *prop_bag;
140 hres = create_param_prop_bag(host->element->element.nselem, &prop_bag);
144 if(prop_bag && !check_load_safety(host)) {
145 IPropertyBag_Release(prop_bag);
150 hres = IPersistPropertyBag_Load(persist_prop_bag, prop_bag, NULL);
151 IPropertyBag_Release(prop_bag);
153 WARN("Load failed: %08x\n", hres);
155 hres = IPersistPropertyBag_InitNew(persist_prop_bag);
157 WARN("InitNew failed: %08x\n", hres);
161 static void load_plugin(PluginHost *host)
163 IPersistPropertyBag2 *persist_prop_bag2;
164 IPersistPropertyBag *persist_prop_bag;
167 hres = IUnknown_QueryInterface(host->plugin_unk, &IID_IPersistPropertyBag2, (void**)&persist_prop_bag2);
168 if(SUCCEEDED(hres)) {
169 FIXME("Use IPersistPropertyBag2 iface\n");
170 IPersistPropertyBag2_Release(persist_prop_bag2);
174 hres = IUnknown_QueryInterface(host->plugin_unk, &IID_IPersistPropertyBag, (void**)&persist_prop_bag);
175 if(SUCCEEDED(hres)) {
176 load_prop_bag(host, persist_prop_bag);
177 IPersistPropertyBag_Release(persist_prop_bag);
181 FIXME("No IPersistPropertyBag iface\n");
184 static void activate_plugin(PluginHost *host)
186 IClientSecurity *client_security;
187 IQuickActivate *quick_activate;
188 IOleObject *ole_obj = NULL;
189 IOleCommandTarget *cmdtrg;
190 IViewObjectEx *view_obj;
196 if(!host->plugin_unk)
199 /* Note native calls QI on plugin for an undocumented IID and CLSID_HTMLDocument */
201 /* FIXME: call FreezeEvents(TRUE) */
203 hres = IUnknown_QueryInterface(host->plugin_unk, &IID_IClientSecurity, (void**)&client_security);
204 if(SUCCEEDED(hres)) {
205 FIXME("Handle IClientSecurity\n");
206 IClientSecurity_Release(client_security);
210 hres = IUnknown_QueryInterface(host->plugin_unk, &IID_IQuickActivate, (void**)&quick_activate);
211 if(SUCCEEDED(hres)) {
212 QACONTAINER container = {sizeof(container)};
213 QACONTROL control = {sizeof(control)};
215 TRACE("Using IQuickActivate\n");
217 container.pClientSite = &host->IOleClientSite_iface;
218 container.dwAmbientFlags = QACONTAINER_SUPPORTSMNEMONICS|QACONTAINER_MESSAGEREFLECT|QACONTAINER_USERMODE;
219 container.pAdviseSink = &host->IAdviseSinkEx_iface;
220 container.pPropertyNotifySink = &host->IPropertyNotifySink_iface;
222 hres = IQuickActivate_QuickActivate(quick_activate, &container, &control);
223 IQuickActivate_Release(quick_activate);
225 FIXME("QuickActivate failed: %08x\n", hres);
229 hres = IUnknown_QueryInterface(host->plugin_unk, &IID_IOleObject, (void**)&ole_obj);
230 if(SUCCEEDED(hres)) {
231 hres = IOleObject_GetMiscStatus(ole_obj, DVASPECT_CONTENT, &status);
232 TRACE("GetMiscStatus returned %08x %x\n", hres, status);
234 hres = IOleObject_SetClientSite(ole_obj, &host->IOleClientSite_iface);
235 IOleObject_Release(ole_obj);
237 FIXME("SetClientSite failed: %08x\n", hres);
241 TRACE("Plugin does not support IOleObject\n");
248 hres = IUnknown_QueryInterface(host->plugin_unk, &IID_IViewObjectEx, (void**)&view_obj);
249 if(SUCCEEDED(hres)) {
250 DWORD view_status = 0;
252 hres = IViewObjectEx_SetAdvise(view_obj, DVASPECT_CONTENT, 0, (IAdviseSink*)&host->IAdviseSinkEx_iface);
254 WARN("SetAdvise failed: %08x\n", hres);
256 hres = IViewObjectEx_GetViewStatus(view_obj, &view_status);
257 IViewObjectEx_Release(view_obj);
258 TRACE("GetViewStatus returned %08x %x\n", hres, view_status);
262 update_readystate(host);
264 /* NOTE: Native QIs for IActiveScript, an undocumented IID, IOleControl and IRunnableObject */
266 hres = IUnknown_QueryInterface(host->plugin_unk, &IID_IDispatchEx, (void**)&dispex);
267 if(SUCCEEDED(hres)) {
268 FIXME("Use IDispatchEx\n");
269 host->disp = (IDispatch*)dispex;
271 hres = IUnknown_QueryInterface(host->plugin_unk, &IID_IDispatch, (void**)&disp);
275 TRACE("no IDispatch iface\n");
278 hres = IUnknown_QueryInterface(host->plugin_unk, &IID_IOleCommandTarget, (void**)&cmdtrg);
279 if(SUCCEEDED(hres)) {
280 FIXME("Use IOleCommandTarget\n");
281 IOleCommandTarget_Release(cmdtrg);
284 hres = IUnknown_QueryInterface(host->plugin_unk, &IID_IOleObject, (void**)&ole_obj);
286 FIXME("Plugin does not support IOleObject\n");
291 get_pos_rect(host, &rect);
292 hres = IOleObject_DoVerb(ole_obj, OLEIVERB_INPLACEACTIVATE, NULL, &host->IOleClientSite_iface, 0, host->hwnd, &rect);
293 IOleObject_Release(ole_obj);
295 WARN("DoVerb failed: %08x\n", hres);
298 if(host->ip_object) {
301 hres = IOleInPlaceObject_GetWindow(host->ip_object, &hwnd);
303 TRACE("hwnd %p\n", hwnd);
307 void update_plugin_window(PluginHost *host, HWND hwnd, const RECT *rect)
309 BOOL rect_changed = FALSE;
311 if(!hwnd || (host->hwnd && host->hwnd != hwnd)) {
312 FIXME("unhandled hwnd\n");
316 TRACE("%p %s\n", hwnd, wine_dbgstr_rect(rect));
318 if(memcmp(rect, &host->rect, sizeof(RECT))) {
325 activate_plugin(host);
328 if(rect_changed && host->ip_object)
329 IOleInPlaceObject_SetObjectRects(host->ip_object, &host->rect, &host->rect);
332 static void notif_enabled(PluginHost *plugin_host)
334 DISPPARAMS args = {NULL, NULL, 0, 0};
340 hres = IUnknown_QueryInterface(plugin_host->plugin_unk, &IID_IDispatch, (void**)&disp);
342 FIXME("Could not get IDispatch iface: %08x\n", hres);
346 V_VT(&res) = VT_EMPTY;
347 hres = IDispatch_Invoke(disp, DISPID_ENABLED, &IID_NULL, 0/*FIXME*/, DISPATCH_PROPERTYGET, &args, &res, NULL, &err);
348 IDispatch_Release(disp);
349 if(SUCCEEDED(hres)) {
350 FIXME("Got enabled %s\n", debugstr_variant(&res));
355 void notif_container_change(HTMLPluginContainer *plugin_container, DISPID dispid)
357 IOleControl *ole_control;
360 if(!plugin_container->plugin_host || !plugin_container->plugin_host->plugin_unk)
363 notif_enabled(plugin_container->plugin_host);
365 hres = IUnknown_QueryInterface(plugin_container->plugin_host->plugin_unk, &IID_IOleControl, (void**)&ole_control);
366 if(SUCCEEDED(hres)) {
367 IOleControl_OnAmbientPropertyChange(ole_control, dispid);
368 IOleControl_Release(ole_control);
372 HRESULT get_plugin_disp(HTMLPluginContainer *plugin_container, IDispatch **ret)
376 host = plugin_container->plugin_host;
378 ERR("No plugin host\n");
387 if(!check_script_safety(host)) {
388 FIXME("Insecure object\n");
392 IDispatch_AddRef(host->disp);
397 HRESULT get_plugin_dispid(HTMLPluginContainer *plugin_container, WCHAR *name, DISPID *ret)
404 if(!plugin_container->plugin_host) {
405 WARN("no plugin host\n");
406 return DISP_E_UNKNOWNNAME;
409 disp = plugin_container->plugin_host->disp;
411 return DISP_E_UNKNOWNNAME;
413 hres = IDispatch_GetIDsOfNames(disp, &IID_NULL, &name, 1, 0, &id);
415 TRACE("no prop %s\n", debugstr_w(name));
416 return DISP_E_UNKNOWNNAME;
419 for(i=0; i < plugin_container->props_len; i++) {
420 if(id == plugin_container->props[i]) {
421 *ret = MSHTML_DISPID_CUSTOM_MIN+i;
426 if(!plugin_container->props) {
427 plugin_container->props = heap_alloc(8*sizeof(DISPID));
428 if(!plugin_container->props)
429 return E_OUTOFMEMORY;
430 plugin_container->props_size = 8;
431 }else if(plugin_container->props_len == plugin_container->props_size) {
434 new_props = heap_realloc(plugin_container->props, plugin_container->props_size*2*sizeof(DISPID));
436 return E_OUTOFMEMORY;
438 plugin_container->props = new_props;
439 plugin_container->props_size *= 2;
442 plugin_container->props[plugin_container->props_len] = id;
443 *ret = MSHTML_DISPID_CUSTOM_MIN+plugin_container->props_len;
444 plugin_container->props_len++;
448 HRESULT invoke_plugin_prop(HTMLPluginContainer *plugin_container, DISPID id, LCID lcid, WORD flags, DISPPARAMS *params,
449 VARIANT *res, EXCEPINFO *ei)
453 host = plugin_container->plugin_host;
454 if(!host || !host->disp) {
455 FIXME("Called with no disp\n");
459 if(!check_script_safety(host)) {
460 FIXME("Insecure object\n");
464 if(id < MSHTML_DISPID_CUSTOM_MIN || id > MSHTML_DISPID_CUSTOM_MIN + plugin_container->props_len) {
469 return IDispatch_Invoke(host->disp, plugin_container->props[id-MSHTML_DISPID_CUSTOM_MIN], &IID_NULL,
470 lcid, flags, params, res, ei, NULL);
473 static inline PluginHost *impl_from_IOleClientSite(IOleClientSite *iface)
475 return CONTAINING_RECORD(iface, PluginHost, IOleClientSite_iface);
478 static HRESULT WINAPI PHClientSite_QueryInterface(IOleClientSite *iface, REFIID riid, void **ppv)
480 PluginHost *This = impl_from_IOleClientSite(iface);
482 if(IsEqualGUID(&IID_IUnknown, riid)) {
483 TRACE("(%p)->(IID_IUnknown %p)\n", This, ppv);
484 *ppv = &This->IOleClientSite_iface;
485 }else if(IsEqualGUID(&IID_IOleClientSite, riid)) {
486 TRACE("(%p)->(IID_IOleClientSite %p)\n", This, ppv);
487 *ppv = &This->IOleClientSite_iface;
488 }else if(IsEqualGUID(&IID_IAdviseSink, riid)) {
489 TRACE("(%p)->(IID_IAdviseSink %p)\n", This, ppv);
490 *ppv = &This->IAdviseSinkEx_iface;
491 }else if(IsEqualGUID(&IID_IAdviseSinkEx, riid)) {
492 TRACE("(%p)->(IID_IAdviseSinkEx %p)\n", This, ppv);
493 *ppv = &This->IAdviseSinkEx_iface;
494 }else if(IsEqualGUID(&IID_IPropertyNotifySink, riid)) {
495 TRACE("(%p)->(IID_IPropertyNotifySink %p)\n", This, ppv);
496 *ppv = &This->IPropertyNotifySink_iface;
497 }else if(IsEqualGUID(&IID_IDispatch, riid)) {
498 TRACE("(%p)->(IID_IDispatch %p)\n", This, ppv);
499 *ppv = &This->IDispatch_iface;
500 }else if(IsEqualGUID(&IID_IOleWindow, riid)) {
501 TRACE("(%p)->(IID_IOleWindow %p)\n", This, ppv);
502 *ppv = &This->IOleInPlaceSiteEx_iface;
503 }else if(IsEqualGUID(&IID_IOleInPlaceSite, riid)) {
504 TRACE("(%p)->(IID_IOleInPlaceSite %p)\n", This, ppv);
505 *ppv = &This->IOleInPlaceSiteEx_iface;
506 }else if(IsEqualGUID(&IID_IOleInPlaceSiteEx, riid)) {
507 TRACE("(%p)->(IID_IOleInPlaceSiteEx %p)\n", This, ppv);
508 *ppv = &This->IOleInPlaceSiteEx_iface;
509 }else if(IsEqualGUID(&IID_IOleControlSite, riid)) {
510 TRACE("(%p)->(IID_IOleControlSite %p)\n", This, ppv);
511 *ppv = &This->IOleControlSite_iface;
512 }else if(IsEqualGUID(&IID_IBindHost, riid)) {
513 TRACE("(%p)->(IID_IBindHost %p)\n", This, ppv);
514 *ppv = &This->IBindHost_iface;
515 }else if(IsEqualGUID(&IID_IServiceProvider, riid)) {
516 TRACE("(%p)->(IID_IServiceProvider %p)\n", This, ppv);
517 *ppv = &This->IServiceProvider_iface;
519 WARN("Unsupported interface %s\n", debugstr_guid(riid));
521 return E_NOINTERFACE;
524 IUnknown_AddRef((IUnknown*)*ppv);
528 static ULONG WINAPI PHClientSite_AddRef(IOleClientSite *iface)
530 PluginHost *This = impl_from_IOleClientSite(iface);
531 LONG ref = InterlockedIncrement(&This->ref);
533 TRACE("(%p) ref=%d\n", This, ref);
538 static ULONG WINAPI PHClientSite_Release(IOleClientSite *iface)
540 PluginHost *This = impl_from_IOleClientSite(iface);
541 LONG ref = InterlockedDecrement(&This->ref);
543 TRACE("(%p) ref=%d\n", This, ref);
547 IDispatch_Release(This->disp);
549 IOleInPlaceObject_Release(This->ip_object);
550 list_remove(&This->entry);
552 This->element->plugin_host = NULL;
554 IUnknown_Release(This->plugin_unk);
561 static HRESULT WINAPI PHClientSite_SaveObject(IOleClientSite *iface)
563 PluginHost *This = impl_from_IOleClientSite(iface);
564 FIXME("(%p)\n", This);
568 static HRESULT WINAPI PHClientSite_GetMoniker(IOleClientSite *iface, DWORD dwAssign,
569 DWORD dwWhichMoniker, IMoniker **ppmk)
571 PluginHost *This = impl_from_IOleClientSite(iface);
573 TRACE("(%p)->(%d %d %p)\n", This, dwAssign, dwWhichMoniker, ppmk);
575 switch(dwWhichMoniker) {
576 case OLEWHICHMK_CONTAINER:
577 if(!This->doc || !This->doc->window || !This->doc->window->mon) {
578 FIXME("no moniker\n");
582 *ppmk = This->doc->window->mon;
583 IMoniker_AddRef(*ppmk);
586 FIXME("which %d\n", dwWhichMoniker);
593 static HRESULT WINAPI PHClientSite_GetContainer(IOleClientSite *iface, IOleContainer **ppContainer)
595 PluginHost *This = impl_from_IOleClientSite(iface);
597 TRACE("(%p)->(%p)\n", This, ppContainer);
600 ERR("Called on detached object\n");
604 *ppContainer = &This->doc->basedoc.IOleContainer_iface;
605 IOleContainer_AddRef(*ppContainer);
609 static HRESULT WINAPI PHClientSite_ShowObject(IOleClientSite *iface)
611 PluginHost *This = impl_from_IOleClientSite(iface);
613 TRACE("(%p)\n", This);
618 static HRESULT WINAPI PHClientSite_OnShowWindow(IOleClientSite *iface, BOOL fShow)
620 PluginHost *This = impl_from_IOleClientSite(iface);
621 FIXME("(%p)->(%x)\n", This, fShow);
625 static HRESULT WINAPI PHClientSite_RequestNewObjectLayout(IOleClientSite *iface)
627 PluginHost *This = impl_from_IOleClientSite(iface);
628 FIXME("(%p)\n", This);
632 static const IOleClientSiteVtbl OleClientSiteVtbl = {
633 PHClientSite_QueryInterface,
635 PHClientSite_Release,
636 PHClientSite_SaveObject,
637 PHClientSite_GetMoniker,
638 PHClientSite_GetContainer,
639 PHClientSite_ShowObject,
640 PHClientSite_OnShowWindow,
641 PHClientSite_RequestNewObjectLayout
644 static inline PluginHost *impl_from_IAdviseSinkEx(IAdviseSinkEx *iface)
646 return CONTAINING_RECORD(iface, PluginHost, IAdviseSinkEx_iface);
649 static HRESULT WINAPI PHAdviseSinkEx_QueryInterface(IAdviseSinkEx *iface, REFIID riid, void **ppv)
651 PluginHost *This = impl_from_IAdviseSinkEx(iface);
652 return IOleClientSite_QueryInterface(&This->IOleClientSite_iface, riid, ppv);
655 static ULONG WINAPI PHAdviseSinkEx_AddRef(IAdviseSinkEx *iface)
657 PluginHost *This = impl_from_IAdviseSinkEx(iface);
658 return IOleClientSite_AddRef(&This->IOleClientSite_iface);
661 static ULONG WINAPI PHAdviseSinkEx_Release(IAdviseSinkEx *iface)
663 PluginHost *This = impl_from_IAdviseSinkEx(iface);
664 return IOleClientSite_Release(&This->IOleClientSite_iface);
667 static void WINAPI PHAdviseSinkEx_OnDataChange(IAdviseSinkEx *iface, FORMATETC *pFormatetc, STGMEDIUM *pStgMedium)
669 PluginHost *This = impl_from_IAdviseSinkEx(iface);
670 FIXME("(%p)->(%p %p)\n", This, pFormatetc, pStgMedium);
673 static void WINAPI PHAdviseSinkEx_OnViewChange(IAdviseSinkEx *iface, DWORD dwAspect, LONG lindex)
675 PluginHost *This = impl_from_IAdviseSinkEx(iface);
676 FIXME("(%p)->(%d %d)\n", This, dwAspect, lindex);
679 static void WINAPI PHAdviseSinkEx_OnRename(IAdviseSinkEx *iface, IMoniker *pmk)
681 PluginHost *This = impl_from_IAdviseSinkEx(iface);
682 FIXME("(%p)->(%p)\n", This, pmk);
685 static void WINAPI PHAdviseSinkEx_OnSave(IAdviseSinkEx *iface)
687 PluginHost *This = impl_from_IAdviseSinkEx(iface);
688 FIXME("(%p)\n", This);
691 static void WINAPI PHAdviseSinkEx_OnClose(IAdviseSinkEx *iface)
693 PluginHost *This = impl_from_IAdviseSinkEx(iface);
694 FIXME("(%p)\n", This);
697 static void WINAPI PHAdviseSinkEx_OnViewStatusChange(IAdviseSinkEx *iface, DWORD dwViewStatus)
699 PluginHost *This = impl_from_IAdviseSinkEx(iface);
700 FIXME("(%p)->(%d)\n", This, dwViewStatus);
703 static const IAdviseSinkExVtbl AdviseSinkExVtbl = {
704 PHAdviseSinkEx_QueryInterface,
705 PHAdviseSinkEx_AddRef,
706 PHAdviseSinkEx_Release,
707 PHAdviseSinkEx_OnDataChange,
708 PHAdviseSinkEx_OnViewChange,
709 PHAdviseSinkEx_OnRename,
710 PHAdviseSinkEx_OnSave,
711 PHAdviseSinkEx_OnClose,
712 PHAdviseSinkEx_OnViewStatusChange
715 static inline PluginHost *impl_from_IPropertyNotifySink(IPropertyNotifySink *iface)
717 return CONTAINING_RECORD(iface, PluginHost, IPropertyNotifySink_iface);
720 static HRESULT WINAPI PHPropertyNotifySink_QueryInterface(IPropertyNotifySink *iface, REFIID riid, void **ppv)
722 PluginHost *This = impl_from_IPropertyNotifySink(iface);
723 return IOleClientSite_QueryInterface(&This->IOleClientSite_iface, riid, ppv);
726 static ULONG WINAPI PHPropertyNotifySink_AddRef(IPropertyNotifySink *iface)
728 PluginHost *This = impl_from_IPropertyNotifySink(iface);
729 return IOleClientSite_AddRef(&This->IOleClientSite_iface);
732 static ULONG WINAPI PHPropertyNotifySink_Release(IPropertyNotifySink *iface)
734 PluginHost *This = impl_from_IPropertyNotifySink(iface);
735 return IOleClientSite_Release(&This->IOleClientSite_iface);
738 static HRESULT WINAPI PHPropertyNotifySink_OnChanged(IPropertyNotifySink *iface, DISPID dispID)
740 PluginHost *This = impl_from_IPropertyNotifySink(iface);
742 TRACE("(%p)->(%d)\n", This, dispID);
745 case DISPID_READYSTATE:
746 update_readystate(This);
749 FIXME("Unimplemented dispID %d\n", dispID);
756 static HRESULT WINAPI PHPropertyNotifySink_OnRequestEdit(IPropertyNotifySink *iface, DISPID dispID)
758 PluginHost *This = impl_from_IPropertyNotifySink(iface);
759 FIXME("(%p)->(%d)\n", This, dispID);
763 static const IPropertyNotifySinkVtbl PropertyNotifySinkVtbl = {
764 PHPropertyNotifySink_QueryInterface,
765 PHPropertyNotifySink_AddRef,
766 PHPropertyNotifySink_Release,
767 PHPropertyNotifySink_OnChanged,
768 PHPropertyNotifySink_OnRequestEdit
771 static inline PluginHost *impl_from_IDispatch(IDispatch *iface)
773 return CONTAINING_RECORD(iface, PluginHost, IDispatch_iface);
776 static HRESULT WINAPI PHDispatch_QueryInterface(IDispatch *iface, REFIID riid, void **ppv)
778 PluginHost *This = impl_from_IDispatch(iface);
779 return IOleClientSite_QueryInterface(&This->IOleClientSite_iface, riid, ppv);
782 static ULONG WINAPI PHDispatch_AddRef(IDispatch *iface)
784 PluginHost *This = impl_from_IDispatch(iface);
785 return IOleClientSite_AddRef(&This->IOleClientSite_iface);
788 static ULONG WINAPI PHDispatch_Release(IDispatch *iface)
790 PluginHost *This = impl_from_IDispatch(iface);
791 return IOleClientSite_Release(&This->IOleClientSite_iface);
794 static HRESULT WINAPI PHDispatch_GetTypeInfoCount(IDispatch *iface, UINT *pctinfo)
796 PluginHost *This = impl_from_IDispatch(iface);
797 FIXME("(%p)->(%p)\n", This, pctinfo);
801 static HRESULT WINAPI PHDispatch_GetTypeInfo(IDispatch *iface, UINT iTInfo,
802 LCID lcid, ITypeInfo **ppTInfo)
804 PluginHost *This = impl_from_IDispatch(iface);
805 FIXME("(%p)->(%d %d %p)\n", This, iTInfo, lcid, ppTInfo);
809 static HRESULT WINAPI PHDispatch_GetIDsOfNames(IDispatch *iface, REFIID riid,
810 LPOLESTR *rgszNames, UINT cNames, LCID lcid, DISPID *rgDispId)
812 PluginHost *This = impl_from_IDispatch(iface);
813 FIXME("(%p)->(%s %p %d %d %p)\n", This, debugstr_guid(riid), rgszNames, cNames, lcid, rgDispId);
817 static HRESULT WINAPI PHDispatch_Invoke(IDispatch *iface, DISPID dispid, REFIID riid, LCID lcid,
818 WORD wFlags, DISPPARAMS *pDispParams, VARIANT *pVarResult, EXCEPINFO *pExcepInfo, UINT *puArgErr)
820 PluginHost *This = impl_from_IDispatch(iface);
821 FIXME("(%p)->(%d %x %p %p)\n", This, dispid, wFlags, pDispParams, pVarResult);
825 static const IDispatchVtbl DispatchVtbl = {
826 PHDispatch_QueryInterface,
829 PHDispatch_GetTypeInfoCount,
830 PHDispatch_GetTypeInfo,
831 PHDispatch_GetIDsOfNames,
835 static inline PluginHost *impl_from_IOleInPlaceSiteEx(IOleInPlaceSiteEx *iface)
837 return CONTAINING_RECORD(iface, PluginHost, IOleInPlaceSiteEx_iface);
840 static HRESULT WINAPI PHInPlaceSite_QueryInterface(IOleInPlaceSiteEx *iface, REFIID riid, void **ppv)
842 PluginHost *This = impl_from_IOleInPlaceSiteEx(iface);
843 return IOleClientSite_QueryInterface(&This->IOleClientSite_iface, riid, ppv);
846 static ULONG WINAPI PHInPlaceSite_AddRef(IOleInPlaceSiteEx *iface)
848 PluginHost *This = impl_from_IOleInPlaceSiteEx(iface);
849 return IOleClientSite_AddRef(&This->IOleClientSite_iface);
852 static ULONG WINAPI PHInPlaceSite_Release(IOleInPlaceSiteEx *iface)
854 PluginHost *This = impl_from_IOleInPlaceSiteEx(iface);
855 return IOleClientSite_Release(&This->IOleClientSite_iface);
858 static HRESULT WINAPI PHInPlaceSite_GetWindow(IOleInPlaceSiteEx *iface, HWND *phwnd)
860 PluginHost *This = impl_from_IOleInPlaceSiteEx(iface);
862 TRACE("(%p)->(%p)\n", This, phwnd);
868 static HRESULT WINAPI PHInPlaceSite_ContextSensitiveHelp(IOleInPlaceSiteEx *iface, BOOL fEnterMode)
870 PluginHost *This = impl_from_IOleInPlaceSiteEx(iface);
871 FIXME("(%p)->(%x)\n", This, fEnterMode);
875 static HRESULT WINAPI PHInPlaceSite_CanInPlaceActivate(IOleInPlaceSiteEx *iface)
877 PluginHost *This = impl_from_IOleInPlaceSiteEx(iface);
879 TRACE("(%p)\n", This);
884 static HRESULT WINAPI PHInPlaceSite_OnInPlaceActivate(IOleInPlaceSiteEx *iface)
886 PluginHost *This = impl_from_IOleInPlaceSiteEx(iface);
887 FIXME("(%p)\n", This);
891 static HRESULT WINAPI PHInPlaceSite_OnUIActivate(IOleInPlaceSiteEx *iface)
893 PluginHost *This = impl_from_IOleInPlaceSiteEx(iface);
895 TRACE("(%p)\n", This);
897 if(!This->plugin_unk) {
898 ERR("No plugin object\n");
902 This->ui_active = TRUE;
908 static HRESULT WINAPI PHInPlaceSite_GetWindowContext(IOleInPlaceSiteEx *iface,
909 IOleInPlaceFrame **ppFrame, IOleInPlaceUIWindow **ppDoc, RECT *lprcPosRect,
910 RECT *lprcClipRect, OLEINPLACEFRAMEINFO *frame_info)
912 PluginHost *This = impl_from_IOleInPlaceSiteEx(iface);
913 IOleInPlaceUIWindow *ip_window;
914 IOleInPlaceFrame *ip_frame;
918 TRACE("(%p)->(%p %p %p %p %p)\n", This, ppFrame, ppDoc, lprcPosRect, lprcClipRect, frame_info);
920 if(!This->doc || !This->doc->basedoc.doc_obj || !This->doc->basedoc.doc_obj->ipsite) {
921 FIXME("No ipsite\n");
925 hres = IOleInPlaceSite_GetWindowContext(This->doc->basedoc.doc_obj->ipsite, &ip_frame, &ip_window, &pr, &cr, frame_info);
927 WARN("GetWindowContext failed: %08x\n", hres);
932 IOleInPlaceUIWindow_Release(ip_window);
934 IOleInPlaceFrame_Release(ip_frame);
936 hres = create_ip_frame(&ip_frame);
940 hres = create_ip_window(ppDoc);
945 *lprcPosRect = This->rect;
946 *lprcClipRect = This->rect;
950 static HRESULT WINAPI PHInPlaceSite_Scroll(IOleInPlaceSiteEx *iface, SIZE scrollExtent)
952 PluginHost *This = impl_from_IOleInPlaceSiteEx(iface);
953 FIXME("(%p)->({%d %d})\n", This, scrollExtent.cx, scrollExtent.cy);
957 static HRESULT WINAPI PHInPlaceSite_OnUIDeactivate(IOleInPlaceSiteEx *iface, BOOL fUndoable)
959 PluginHost *This = impl_from_IOleInPlaceSiteEx(iface);
960 FIXME("(%p)->(%x)\n", This, fUndoable);
964 static HRESULT WINAPI PHInPlaceSite_OnInPlaceDeactivate(IOleInPlaceSiteEx *iface)
966 PluginHost *This = impl_from_IOleInPlaceSiteEx(iface);
968 TRACE("(%p)\n", This);
970 if(This->ip_object) {
971 IOleInPlaceObject_Release(This->ip_object);
972 This->ip_object = NULL;
978 static HRESULT WINAPI PHInPlaceSite_DiscardUndoState(IOleInPlaceSiteEx *iface)
980 PluginHost *This = impl_from_IOleInPlaceSiteEx(iface);
981 FIXME("(%p)\n", This);
985 static HRESULT WINAPI PHInPlaceSite_DeactivateAndUndo(IOleInPlaceSiteEx *iface)
987 PluginHost *This = impl_from_IOleInPlaceSiteEx(iface);
988 FIXME("(%p)\n", This);
992 static HRESULT WINAPI PHInPlaceSite_OnPosRectChange(IOleInPlaceSiteEx *iface, LPCRECT lprcPosRect)
994 PluginHost *This = impl_from_IOleInPlaceSiteEx(iface);
995 FIXME("(%p)->(%p)\n", This, lprcPosRect);
999 static HRESULT WINAPI PHInPlaceSiteEx_OnInPlaceActivateEx(IOleInPlaceSiteEx *iface, BOOL *pfNoRedraw, DWORD dwFlags)
1001 PluginHost *This = impl_from_IOleInPlaceSiteEx(iface);
1005 TRACE("(%p)->(%p %x)\n", This, pfNoRedraw, dwFlags);
1010 hres = IUnknown_QueryInterface(This->plugin_unk, &IID_IOleInPlaceObject, (void**)&This->ip_object);
1014 hres = IOleInPlaceObject_GetWindow(This->ip_object, &hwnd);
1016 FIXME("Use hwnd %p\n", hwnd);
1018 *pfNoRedraw = FALSE;
1022 static HRESULT WINAPI PHInPlaceSiteEx_OnInPlaceDeactivateEx(IOleInPlaceSiteEx *iface, BOOL fNoRedraw)
1024 PluginHost *This = impl_from_IOleInPlaceSiteEx(iface);
1025 FIXME("(%p)->(%x)\n", This, fNoRedraw);
1029 static HRESULT WINAPI PHInPlaceSiteEx_RequestUIActivate(IOleInPlaceSiteEx *iface)
1031 PluginHost *This = impl_from_IOleInPlaceSiteEx(iface);
1032 FIXME("(%p)\n", This);
1036 static const IOleInPlaceSiteExVtbl OleInPlaceSiteExVtbl = {
1037 PHInPlaceSite_QueryInterface,
1038 PHInPlaceSite_AddRef,
1039 PHInPlaceSite_Release,
1040 PHInPlaceSite_GetWindow,
1041 PHInPlaceSite_ContextSensitiveHelp,
1042 PHInPlaceSite_CanInPlaceActivate,
1043 PHInPlaceSite_OnInPlaceActivate,
1044 PHInPlaceSite_OnUIActivate,
1045 PHInPlaceSite_GetWindowContext,
1046 PHInPlaceSite_Scroll,
1047 PHInPlaceSite_OnUIDeactivate,
1048 PHInPlaceSite_OnInPlaceDeactivate,
1049 PHInPlaceSite_DiscardUndoState,
1050 PHInPlaceSite_DeactivateAndUndo,
1051 PHInPlaceSite_OnPosRectChange,
1052 PHInPlaceSiteEx_OnInPlaceActivateEx,
1053 PHInPlaceSiteEx_OnInPlaceDeactivateEx,
1054 PHInPlaceSiteEx_RequestUIActivate
1057 static inline PluginHost *impl_from_IOleControlSite(IOleControlSite *iface)
1059 return CONTAINING_RECORD(iface, PluginHost, IOleControlSite_iface);
1062 static HRESULT WINAPI PHControlSite_QueryInterface(IOleControlSite *iface, REFIID riid, void **ppv)
1064 PluginHost *This = impl_from_IOleControlSite(iface);
1065 return IOleClientSite_QueryInterface(&This->IOleClientSite_iface, riid, ppv);
1068 static ULONG WINAPI PHControlSite_AddRef(IOleControlSite *iface)
1070 PluginHost *This = impl_from_IOleControlSite(iface);
1071 return IOleClientSite_AddRef(&This->IOleClientSite_iface);
1074 static ULONG WINAPI PHControlSite_Release(IOleControlSite *iface)
1076 PluginHost *This = impl_from_IOleControlSite(iface);
1077 return IOleClientSite_Release(&This->IOleClientSite_iface);
1080 static HRESULT WINAPI PHControlSite_OnControlInfoChanged(IOleControlSite *iface)
1082 PluginHost *This = impl_from_IOleControlSite(iface);
1083 FIXME("(%p)\n", This);
1087 static HRESULT WINAPI PHControlSite_LockInPlaceActive(IOleControlSite *iface, BOOL fLock)
1089 PluginHost *This = impl_from_IOleControlSite(iface);
1090 FIXME("(%p)->(%x)\n", This, fLock);
1094 static HRESULT WINAPI PHControlSite_GetExtendedControl(IOleControlSite *iface, IDispatch **ppDisp)
1096 PluginHost *This = impl_from_IOleControlSite(iface);
1097 FIXME("(%p)->(%p)\n", This, ppDisp);
1101 static HRESULT WINAPI PHControlSite_TransformCoords(IOleControlSite *iface, POINTL *pPtlHimetric, POINTF *pPtfContainer, DWORD dwFlags)
1103 PluginHost *This = impl_from_IOleControlSite(iface);
1104 FIXME("(%p)->(%p %p %x)\n", This, pPtlHimetric, pPtfContainer, dwFlags);
1108 static HRESULT WINAPI PHControlSite_TranslateAccelerator(IOleControlSite *iface, MSG *pMsg, DWORD grfModifiers)
1110 PluginHost *This = impl_from_IOleControlSite(iface);
1111 FIXME("(%p)->(%x)\n", This, grfModifiers);
1115 static HRESULT WINAPI PHControlSite_OnFocus(IOleControlSite *iface, BOOL fGotFocus)
1117 PluginHost *This = impl_from_IOleControlSite(iface);
1118 FIXME("(%p)->(%x)\n", This, fGotFocus);
1122 static HRESULT WINAPI PHControlSite_ShowPropertyFrame(IOleControlSite *iface)
1124 PluginHost *This = impl_from_IOleControlSite(iface);
1125 FIXME("(%p)\n", This);
1129 static const IOleControlSiteVtbl OleControlSiteVtbl = {
1130 PHControlSite_QueryInterface,
1131 PHControlSite_AddRef,
1132 PHControlSite_Release,
1133 PHControlSite_OnControlInfoChanged,
1134 PHControlSite_LockInPlaceActive,
1135 PHControlSite_GetExtendedControl,
1136 PHControlSite_TransformCoords,
1137 PHControlSite_TranslateAccelerator,
1138 PHControlSite_OnFocus,
1139 PHControlSite_ShowPropertyFrame
1142 static inline PluginHost *impl_from_IBindHost(IBindHost *iface)
1144 return CONTAINING_RECORD(iface, PluginHost, IBindHost_iface);
1147 static HRESULT WINAPI PHBindHost_QueryInterface(IBindHost *iface, REFIID riid, void **ppv)
1149 PluginHost *This = impl_from_IBindHost(iface);
1150 return IOleClientSite_QueryInterface(&This->IOleClientSite_iface, riid, ppv);
1153 static ULONG WINAPI PHBindHost_AddRef(IBindHost *iface)
1155 PluginHost *This = impl_from_IBindHost(iface);
1156 return IOleClientSite_AddRef(&This->IOleClientSite_iface);
1159 static ULONG WINAPI PHBindHost_Release(IBindHost *iface)
1161 PluginHost *This = impl_from_IBindHost(iface);
1162 return IOleClientSite_Release(&This->IOleClientSite_iface);
1165 static HRESULT WINAPI PHBindHost_CreateMoniker(IBindHost *iface, LPOLESTR szName, IBindCtx *pBC, IMoniker **ppmk, DWORD dwReserved)
1167 PluginHost *This = impl_from_IBindHost(iface);
1169 TRACE("(%p)->(%s %p %p %x)\n", This, debugstr_w(szName), pBC, ppmk, dwReserved);
1171 if(!This->doc || !This->doc->window || !This->doc->window->mon) {
1172 FIXME("no moniker\n");
1173 return E_UNEXPECTED;
1176 return CreateURLMoniker(This->doc->window->mon, szName, ppmk);
1179 static HRESULT WINAPI PHBindHost_MonikerBindToStorage(IBindHost *iface, IMoniker *pMk, IBindCtx *pBC,
1180 IBindStatusCallback *pBSC, REFIID riid, void **ppvObj)
1182 PluginHost *This = impl_from_IBindHost(iface);
1183 FIXME("(%p)->(%p %p %p %s %p)\n", This, pMk, pBC, pBSC, debugstr_guid(riid), ppvObj);
1187 static HRESULT WINAPI PHBindHost_MonikerBindToObject(IBindHost *iface, IMoniker *pMk, IBindCtx *pBC,
1188 IBindStatusCallback *pBSC, REFIID riid, void **ppvObj)
1190 PluginHost *This = impl_from_IBindHost(iface);
1191 FIXME("(%p)->(%p %p %p %s %p)\n", This, pMk, pBC, pBSC, debugstr_guid(riid), ppvObj);
1195 static const IBindHostVtbl BindHostVtbl = {
1196 PHBindHost_QueryInterface,
1199 PHBindHost_CreateMoniker,
1200 PHBindHost_MonikerBindToStorage,
1201 PHBindHost_MonikerBindToObject
1204 static inline PluginHost *impl_from_IServiceProvider(IServiceProvider *iface)
1206 return CONTAINING_RECORD(iface, PluginHost, IServiceProvider_iface);
1209 static HRESULT WINAPI PHServiceProvider_QueryInterface(IServiceProvider *iface, REFIID riid, void **ppv)
1211 PluginHost *This = impl_from_IServiceProvider(iface);
1212 return IOleClientSite_QueryInterface(&This->IOleClientSite_iface, riid, ppv);
1215 static ULONG WINAPI PHServiceProvider_AddRef(IServiceProvider *iface)
1217 PluginHost *This = impl_from_IServiceProvider(iface);
1218 return IOleClientSite_AddRef(&This->IOleClientSite_iface);
1221 static ULONG WINAPI PHServiceProvider_Release(IServiceProvider *iface)
1223 PluginHost *This = impl_from_IServiceProvider(iface);
1224 return IOleClientSite_Release(&This->IOleClientSite_iface);
1227 static HRESULT WINAPI PHServiceProvider_QueryService(IServiceProvider *iface, REFGUID guidService, REFIID riid, void **ppv)
1229 PluginHost *This = impl_from_IServiceProvider(iface);
1231 if(IsEqualGUID(guidService, &SID_SBindHost)) {
1232 TRACE("SID_SBindHost service\n");
1233 return IOleClientSite_QueryInterface(&This->IOleClientSite_iface, riid, ppv);
1236 TRACE("(%p)->(%s %s %p)\n", This, debugstr_guid(guidService), debugstr_guid(riid), ppv);
1238 if(!This->doc || !This->doc->basedoc.window) {
1240 return E_NOINTERFACE;
1243 return IServiceProvider_QueryService(&This->doc->basedoc.window->base.IServiceProvider_iface,
1244 guidService, riid, ppv);
1247 static const IServiceProviderVtbl ServiceProviderVtbl = {
1248 PHServiceProvider_QueryInterface,
1249 PHServiceProvider_AddRef,
1250 PHServiceProvider_Release,
1251 PHServiceProvider_QueryService
1254 static HRESULT assoc_element(PluginHost *host, HTMLDocumentNode *doc, nsIDOMElement *nselem)
1256 HTMLPluginContainer *container_elem;
1260 hres = get_node(doc, (nsIDOMNode*)nselem, TRUE, &node);
1264 hres = IHTMLDOMNode_QueryInterface(&node->IHTMLDOMNode_iface, &IID_HTMLPluginContainer,
1265 (void**)&container_elem);
1268 ERR("Not an object element\n");
1272 container_elem->plugin_host = host;
1273 host->element = container_elem;
1277 void detach_plugin_host(PluginHost *host)
1281 TRACE("%p\n", host);
1286 if(host->ip_object) {
1288 IOleInPlaceObject_UIDeactivate(host->ip_object);
1289 IOleInPlaceObject_InPlaceDeactivate(host->ip_object);
1292 if(host->plugin_unk) {
1293 IOleObject *ole_obj;
1295 hres = IUnknown_QueryInterface(host->plugin_unk, &IID_IOleObject, (void**)&ole_obj);
1296 if(SUCCEEDED(hres)) {
1297 if(!host->ip_object)
1298 IOleObject_Close(ole_obj, OLECLOSE_NOSAVE);
1299 IOleObject_SetClientSite(ole_obj, NULL);
1300 IOleObject_Release(ole_obj);
1305 host->element->plugin_host = NULL;
1306 host->element = NULL;
1309 list_remove(&host->entry);
1310 list_init(&host->entry);
1314 HRESULT create_plugin_host(HTMLDocumentNode *doc, nsIDOMElement *nselem, IUnknown *unk, const CLSID *clsid, PluginHost **ret)
1319 host = heap_alloc_zero(sizeof(*host));
1321 return E_OUTOFMEMORY;
1323 host->IOleClientSite_iface.lpVtbl = &OleClientSiteVtbl;
1324 host->IAdviseSinkEx_iface.lpVtbl = &AdviseSinkExVtbl;
1325 host->IPropertyNotifySink_iface.lpVtbl = &PropertyNotifySinkVtbl;
1326 host->IDispatch_iface.lpVtbl = &DispatchVtbl;
1327 host->IOleInPlaceSiteEx_iface.lpVtbl = &OleInPlaceSiteExVtbl;
1328 host->IOleControlSite_iface.lpVtbl = &OleControlSiteVtbl;
1329 host->IBindHost_iface.lpVtbl = &BindHostVtbl;
1330 host->IServiceProvider_iface.lpVtbl = &ServiceProviderVtbl;
1334 hres = assoc_element(host, doc, nselem);
1340 IUnknown_AddRef(unk);
1341 host->plugin_unk = unk;
1342 host->clsid = *clsid;
1345 list_add_tail(&doc->plugin_hosts, &host->entry);