mshtml: Share nsselect reference with nsnode.
[wine] / dlls / mshtml / htmlobject.c
1 /*
2  * Copyright 2010 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 <stdarg.h>
20 #include <assert.h>
21
22 #define COBJMACROS
23
24 #include "windef.h"
25 #include "winbase.h"
26 #include "winuser.h"
27 #include "winreg.h"
28 #include "ole2.h"
29
30 #include "wine/debug.h"
31
32 #include "mshtml_private.h"
33 #include "pluginhost.h"
34
35 WINE_DEFAULT_DEBUG_CHANNEL(mshtml);
36
37 typedef struct {
38     HTMLPluginContainer plugin_container;
39
40     IHTMLObjectElement IHTMLObjectElement_iface;
41     IHTMLObjectElement2 IHTMLObjectElement2_iface;
42
43     nsIDOMHTMLObjectElement *nsobject;
44 } HTMLObjectElement;
45
46 static inline HTMLObjectElement *impl_from_IHTMLObjectElement(IHTMLObjectElement *iface)
47 {
48     return CONTAINING_RECORD(iface, HTMLObjectElement, IHTMLObjectElement_iface);
49 }
50
51 static HRESULT WINAPI HTMLObjectElement_QueryInterface(IHTMLObjectElement *iface,
52         REFIID riid, void **ppv)
53 {
54     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
55
56     return IHTMLDOMNode_QueryInterface(&This->plugin_container.element.node.IHTMLDOMNode_iface,
57             riid, ppv);
58 }
59
60 static ULONG WINAPI HTMLObjectElement_AddRef(IHTMLObjectElement *iface)
61 {
62     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
63
64     return IHTMLDOMNode_AddRef(&This->plugin_container.element.node.IHTMLDOMNode_iface);
65 }
66
67 static ULONG WINAPI HTMLObjectElement_Release(IHTMLObjectElement *iface)
68 {
69     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
70
71     return IHTMLDOMNode_Release(&This->plugin_container.element.node.IHTMLDOMNode_iface);
72 }
73
74 static HRESULT WINAPI HTMLObjectElement_GetTypeInfoCount(IHTMLObjectElement *iface, UINT *pctinfo)
75 {
76     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
77     return IDispatchEx_GetTypeInfoCount(&This->plugin_container.element.node.dispex.IDispatchEx_iface,
78             pctinfo);
79 }
80
81 static HRESULT WINAPI HTMLObjectElement_GetTypeInfo(IHTMLObjectElement *iface, UINT iTInfo,
82                                               LCID lcid, ITypeInfo **ppTInfo)
83 {
84     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
85     return IDispatchEx_GetTypeInfo(&This->plugin_container.element.node.dispex.IDispatchEx_iface,
86             iTInfo, lcid, ppTInfo);
87 }
88
89 static HRESULT WINAPI HTMLObjectElement_GetIDsOfNames(IHTMLObjectElement *iface, REFIID riid,
90                                                 LPOLESTR *rgszNames, UINT cNames,
91                                                 LCID lcid, DISPID *rgDispId)
92 {
93     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
94     return IDispatchEx_GetIDsOfNames(&This->plugin_container.element.node.dispex.IDispatchEx_iface,
95             riid, rgszNames, cNames, lcid, rgDispId);
96 }
97
98 static HRESULT WINAPI HTMLObjectElement_Invoke(IHTMLObjectElement *iface, DISPID dispIdMember,
99                             REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS *pDispParams,
100                             VARIANT *pVarResult, EXCEPINFO *pExcepInfo, UINT *puArgErr)
101 {
102     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
103     return IDispatchEx_Invoke(&This->plugin_container.element.node.dispex.IDispatchEx_iface,
104             dispIdMember, riid, lcid, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr);
105 }
106
107 static HRESULT WINAPI HTMLObjectElement_get_object(IHTMLObjectElement *iface, IDispatch **p)
108 {
109     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
110
111     TRACE("(%p)->(%p)\n", This, p);
112
113     return get_plugin_disp(&This->plugin_container, p);
114 }
115
116 static HRESULT WINAPI HTMLObjectElement_get_classid(IHTMLObjectElement *iface, BSTR *p)
117 {
118     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
119
120     TRACE("(%p)->(%p)\n", This, p);
121
122     return IHTMLObjectElement2_get_classid(&This->IHTMLObjectElement_iface, p);
123 }
124
125 static HRESULT WINAPI HTMLObjectElement_get_data(IHTMLObjectElement *iface, BSTR *p)
126 {
127     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
128
129     TRACE("(%p)->(%p)\n", This, p);
130
131     return IHTMLObjectElement2_get_data(&This->IHTMLObjectElement_iface, p);
132 }
133
134 static HRESULT WINAPI HTMLObjectElement_put_recordset(IHTMLObjectElement *iface, IDispatch *v)
135 {
136     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
137     FIXME("(%p)->(%p)\n", This, v);
138     return E_NOTIMPL;
139 }
140
141 static HRESULT WINAPI HTMLObjectElement_get_recordset(IHTMLObjectElement *iface, IDispatch **p)
142 {
143     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
144     FIXME("(%p)->(%p)\n", This, p);
145     return E_NOTIMPL;
146 }
147
148 static HRESULT WINAPI HTMLObjectElement_put_align(IHTMLObjectElement *iface, BSTR v)
149 {
150     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
151     FIXME("(%p)->(%s)\n", This, debugstr_w(v));
152     return E_NOTIMPL;
153 }
154
155 static HRESULT WINAPI HTMLObjectElement_get_align(IHTMLObjectElement *iface, BSTR *p)
156 {
157     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
158     FIXME("(%p)->(%p)\n", This, p);
159     return E_NOTIMPL;
160 }
161
162 static HRESULT WINAPI HTMLObjectElement_put_name(IHTMLObjectElement *iface, BSTR v)
163 {
164     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
165     FIXME("(%p)->(%s)\n", This, debugstr_w(v));
166     return E_NOTIMPL;
167 }
168
169 static HRESULT WINAPI HTMLObjectElement_get_name(IHTMLObjectElement *iface, BSTR *p)
170 {
171     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
172     FIXME("(%p)->(%p)\n", This, p);
173     return E_NOTIMPL;
174 }
175
176 static HRESULT WINAPI HTMLObjectElement_put_codeBase(IHTMLObjectElement *iface, BSTR v)
177 {
178     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
179     FIXME("(%p)->(%s)\n", This, debugstr_w(v));
180     return E_NOTIMPL;
181 }
182
183 static HRESULT WINAPI HTMLObjectElement_get_codeBase(IHTMLObjectElement *iface, BSTR *p)
184 {
185     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
186     FIXME("(%p)->(%p)\n", This, p);
187     return E_NOTIMPL;
188 }
189
190 static HRESULT WINAPI HTMLObjectElement_put_codeType(IHTMLObjectElement *iface, BSTR v)
191 {
192     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
193     FIXME("(%p)->(%s)\n", This, debugstr_w(v));
194     return E_NOTIMPL;
195 }
196
197 static HRESULT WINAPI HTMLObjectElement_get_codeType(IHTMLObjectElement *iface, BSTR *p)
198 {
199     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
200     FIXME("(%p)->(%p)\n", This, p);
201     return E_NOTIMPL;
202 }
203
204 static HRESULT WINAPI HTMLObjectElement_put_code(IHTMLObjectElement *iface, BSTR v)
205 {
206     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
207     FIXME("(%p)->(%s)\n", This, debugstr_w(v));
208     return E_NOTIMPL;
209 }
210
211 static HRESULT WINAPI HTMLObjectElement_get_code(IHTMLObjectElement *iface, BSTR *p)
212 {
213     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
214     FIXME("(%p)->(%p)\n", This, p);
215     return E_NOTIMPL;
216 }
217
218 static HRESULT WINAPI HTMLObjectElement_get_BaseHref(IHTMLObjectElement *iface, BSTR *p)
219 {
220     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
221     FIXME("(%p)->(%p)\n", This, p);
222     return E_NOTIMPL;
223 }
224
225 static HRESULT WINAPI HTMLObjectElement_put_type(IHTMLObjectElement *iface, BSTR v)
226 {
227     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
228     FIXME("(%p)->(%s)\n", This, debugstr_w(v));
229     return E_NOTIMPL;
230 }
231
232 static HRESULT WINAPI HTMLObjectElement_get_type(IHTMLObjectElement *iface, BSTR *p)
233 {
234     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
235     FIXME("(%p)->(%p)\n", This, p);
236     return E_NOTIMPL;
237 }
238
239 static HRESULT WINAPI HTMLObjectElement_get_form(IHTMLObjectElement *iface, IHTMLFormElement **p)
240 {
241     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
242     FIXME("(%p)->(%p)\n", This, p);
243     return E_NOTIMPL;
244 }
245
246 static HRESULT WINAPI HTMLObjectElement_put_width(IHTMLObjectElement *iface, VARIANT v)
247 {
248     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
249     FIXME("(%p)->(%s)\n", This, debugstr_variant(&v));
250     return E_NOTIMPL;
251 }
252
253 static HRESULT WINAPI HTMLObjectElement_get_width(IHTMLObjectElement *iface, VARIANT *p)
254 {
255     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
256     FIXME("(%p)->(%p)\n", This, p);
257     return E_NOTIMPL;
258 }
259
260 static HRESULT WINAPI HTMLObjectElement_put_height(IHTMLObjectElement *iface, VARIANT v)
261 {
262     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
263     FIXME("(%p)->(%s)\n", This, debugstr_variant(&v));
264     return E_NOTIMPL;
265 }
266
267 static HRESULT WINAPI HTMLObjectElement_get_height(IHTMLObjectElement *iface, VARIANT *p)
268 {
269     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
270     FIXME("(%p)->(%p)\n", This, p);
271     return E_NOTIMPL;
272 }
273
274 static HRESULT WINAPI HTMLObjectElement_get_readyState(IHTMLObjectElement *iface, LONG *p)
275 {
276     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
277     FIXME("(%p)->(%p)\n", This, p);
278     return E_NOTIMPL;
279 }
280
281 static HRESULT WINAPI HTMLObjectElement_put_onreadystatechange(IHTMLObjectElement *iface, VARIANT v)
282 {
283     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
284     FIXME("(%p)->(%s)\n", This, debugstr_variant(&v));
285     return E_NOTIMPL;
286 }
287
288 static HRESULT WINAPI HTMLObjectElement_get_onreadystatechange(IHTMLObjectElement *iface, VARIANT *p)
289 {
290     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
291     FIXME("(%p)->(%p)\n", This, p);
292     return E_NOTIMPL;
293 }
294
295 static HRESULT WINAPI HTMLObjectElement_put_onerror(IHTMLObjectElement *iface, VARIANT v)
296 {
297     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
298     FIXME("(%p)->(%s)\n", This, debugstr_variant(&v));
299     return E_NOTIMPL;
300 }
301
302 static HRESULT WINAPI HTMLObjectElement_get_onerror(IHTMLObjectElement *iface, VARIANT *p)
303 {
304     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
305     FIXME("(%p)->(%p)\n", This, p);
306     return E_NOTIMPL;
307 }
308
309 static HRESULT WINAPI HTMLObjectElement_put_altHtml(IHTMLObjectElement *iface, BSTR v)
310 {
311     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
312     FIXME("(%p)->(%s)\n", This, debugstr_w(v));
313     return E_NOTIMPL;
314 }
315
316 static HRESULT WINAPI HTMLObjectElement_get_altHtml(IHTMLObjectElement *iface, BSTR *p)
317 {
318     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
319     FIXME("(%p)->(%p)\n", This, p);
320     return E_NOTIMPL;
321 }
322
323 static HRESULT WINAPI HTMLObjectElement_put_vspace(IHTMLObjectElement *iface, LONG v)
324 {
325     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
326     FIXME("(%p)->(%d)\n", This, v);
327     return E_NOTIMPL;
328 }
329
330 static HRESULT WINAPI HTMLObjectElement_get_vspace(IHTMLObjectElement *iface, LONG *p)
331 {
332     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
333     PRInt32 vspace;
334     nsresult nsres;
335
336     TRACE("(%p)->(%p)\n", This, p);
337
338     nsres = nsIDOMHTMLObjectElement_GetVspace(This->nsobject, &vspace);
339     if(NS_FAILED(nsres)) {
340         ERR("GetVspace failed: %08x\n", nsres);
341         return E_FAIL;
342     }
343
344     *p = vspace;
345     return S_OK;
346 }
347
348 static HRESULT WINAPI HTMLObjectElement_put_hspace(IHTMLObjectElement *iface, LONG v)
349 {
350     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
351     FIXME("(%p)->(%d)\n", This, v);
352     return E_NOTIMPL;
353 }
354
355 static HRESULT WINAPI HTMLObjectElement_get_hspace(IHTMLObjectElement *iface, LONG *p)
356 {
357     HTMLObjectElement *This = impl_from_IHTMLObjectElement(iface);
358     FIXME("(%p)->(%p)\n", This, p);
359     return E_NOTIMPL;
360 }
361
362 static const IHTMLObjectElementVtbl HTMLObjectElementVtbl = {
363     HTMLObjectElement_QueryInterface,
364     HTMLObjectElement_AddRef,
365     HTMLObjectElement_Release,
366     HTMLObjectElement_GetTypeInfoCount,
367     HTMLObjectElement_GetTypeInfo,
368     HTMLObjectElement_GetIDsOfNames,
369     HTMLObjectElement_Invoke,
370     HTMLObjectElement_get_object,
371     HTMLObjectElement_get_classid,
372     HTMLObjectElement_get_data,
373     HTMLObjectElement_put_recordset,
374     HTMLObjectElement_get_recordset,
375     HTMLObjectElement_put_align,
376     HTMLObjectElement_get_align,
377     HTMLObjectElement_put_name,
378     HTMLObjectElement_get_name,
379     HTMLObjectElement_put_codeBase,
380     HTMLObjectElement_get_codeBase,
381     HTMLObjectElement_put_codeType,
382     HTMLObjectElement_get_codeType,
383     HTMLObjectElement_put_code,
384     HTMLObjectElement_get_code,
385     HTMLObjectElement_get_BaseHref,
386     HTMLObjectElement_put_type,
387     HTMLObjectElement_get_type,
388     HTMLObjectElement_get_form,
389     HTMLObjectElement_put_width,
390     HTMLObjectElement_get_width,
391     HTMLObjectElement_put_height,
392     HTMLObjectElement_get_height,
393     HTMLObjectElement_get_readyState,
394     HTMLObjectElement_put_onreadystatechange,
395     HTMLObjectElement_get_onreadystatechange,
396     HTMLObjectElement_put_onerror,
397     HTMLObjectElement_get_onerror,
398     HTMLObjectElement_put_altHtml,
399     HTMLObjectElement_get_altHtml,
400     HTMLObjectElement_put_vspace,
401     HTMLObjectElement_get_vspace,
402     HTMLObjectElement_put_hspace,
403     HTMLObjectElement_get_hspace
404 };
405
406 static inline HTMLObjectElement *impl_from_IHTMLObjectElement2(IHTMLObjectElement2 *iface)
407 {
408     return CONTAINING_RECORD(iface, HTMLObjectElement, IHTMLObjectElement2_iface);
409 }
410
411 static HRESULT WINAPI HTMLObjectElement2_QueryInterface(IHTMLObjectElement2 *iface,
412         REFIID riid, void **ppv)
413 {
414     HTMLObjectElement *This = impl_from_IHTMLObjectElement2(iface);
415
416     return IHTMLDOMNode_QueryInterface(&This->plugin_container.element.node.IHTMLDOMNode_iface,
417             riid, ppv);
418 }
419
420 static ULONG WINAPI HTMLObjectElement2_AddRef(IHTMLObjectElement2 *iface)
421 {
422     HTMLObjectElement *This = impl_from_IHTMLObjectElement2(iface);
423
424     return IHTMLDOMNode_AddRef(&This->plugin_container.element.node.IHTMLDOMNode_iface);
425 }
426
427 static ULONG WINAPI HTMLObjectElement2_Release(IHTMLObjectElement2 *iface)
428 {
429     HTMLObjectElement *This = impl_from_IHTMLObjectElement2(iface);
430
431     return IHTMLDOMNode_Release(&This->plugin_container.element.node.IHTMLDOMNode_iface);
432 }
433
434 static HRESULT WINAPI HTMLObjectElement2_GetTypeInfoCount(IHTMLObjectElement2 *iface, UINT *pctinfo)
435 {
436     HTMLObjectElement *This = impl_from_IHTMLObjectElement2(iface);
437     return IDispatchEx_GetTypeInfoCount(&This->plugin_container.element.node.dispex.IDispatchEx_iface,
438             pctinfo);
439 }
440
441 static HRESULT WINAPI HTMLObjectElement2_GetTypeInfo(IHTMLObjectElement2 *iface, UINT iTInfo,
442                                               LCID lcid, ITypeInfo **ppTInfo)
443 {
444     HTMLObjectElement *This = impl_from_IHTMLObjectElement2(iface);
445     return IDispatchEx_GetTypeInfo(&This->plugin_container.element.node.dispex.IDispatchEx_iface,
446             iTInfo, lcid, ppTInfo);
447 }
448
449 static HRESULT WINAPI HTMLObjectElement2_GetIDsOfNames(IHTMLObjectElement2 *iface, REFIID riid,
450         LPOLESTR *rgszNames, UINT cNames, LCID lcid, DISPID *rgDispId)
451 {
452     HTMLObjectElement *This = impl_from_IHTMLObjectElement2(iface);
453     return IDispatchEx_GetIDsOfNames(&This->plugin_container.element.node.dispex.IDispatchEx_iface,
454             riid, rgszNames, cNames, lcid, rgDispId);
455 }
456
457 static HRESULT WINAPI HTMLObjectElement2_Invoke(IHTMLObjectElement2 *iface, DISPID dispIdMember,
458         REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS *pDispParams, VARIANT *pVarResult,
459         EXCEPINFO *pExcepInfo, UINT *puArgErr)
460 {
461     HTMLObjectElement *This = impl_from_IHTMLObjectElement2(iface);
462     return IDispatchEx_Invoke(&This->plugin_container.element.node.dispex.IDispatchEx_iface,
463             dispIdMember, riid, lcid, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr);
464 }
465
466 static HRESULT WINAPI HTMLObjectElement2_namedRecordset(IHTMLObjectElement2 *iface, BSTR dataMember,
467         VARIANT *hierarchy, IDispatch **ppRecordset)
468 {
469     HTMLObjectElement *This = impl_from_IHTMLObjectElement2(iface);
470     FIXME("(%p)->(%s %p %p)\n", This, debugstr_w(dataMember), hierarchy, ppRecordset);
471     return E_NOTIMPL;
472 }
473
474 static HRESULT WINAPI HTMLObjectElement2_put_classid(IHTMLObjectElement2 *iface, BSTR v)
475 {
476     HTMLObjectElement *This = impl_from_IHTMLObjectElement2(iface);
477     FIXME("(%p)->(%s)\n", This, debugstr_w(v));
478     return E_NOTIMPL;
479 }
480
481 static HRESULT WINAPI HTMLObjectElement2_get_classid(IHTMLObjectElement2 *iface, BSTR *p)
482 {
483     HTMLObjectElement *This = impl_from_IHTMLObjectElement2(iface);
484     FIXME("(%p)->(%p)\n", This, p);
485     return E_NOTIMPL;
486 }
487
488 static HRESULT WINAPI HTMLObjectElement2_put_data(IHTMLObjectElement2 *iface, BSTR v)
489 {
490     HTMLObjectElement *This = impl_from_IHTMLObjectElement2(iface);
491     FIXME("(%p)->(%s)\n", This, debugstr_w(v));
492     return E_NOTIMPL;
493 }
494
495 static HRESULT WINAPI HTMLObjectElement2_get_data(IHTMLObjectElement2 *iface, BSTR *p)
496 {
497     HTMLObjectElement *This = impl_from_IHTMLObjectElement2(iface);
498     FIXME("(%p)->(%p)\n", This, p);
499     return E_NOTIMPL;
500 }
501
502 static const IHTMLObjectElement2Vtbl HTMLObjectElement2Vtbl = {
503     HTMLObjectElement2_QueryInterface,
504     HTMLObjectElement2_AddRef,
505     HTMLObjectElement2_Release,
506     HTMLObjectElement2_GetTypeInfoCount,
507     HTMLObjectElement2_GetTypeInfo,
508     HTMLObjectElement2_GetIDsOfNames,
509     HTMLObjectElement2_Invoke,
510     HTMLObjectElement2_namedRecordset,
511     HTMLObjectElement2_put_classid,
512     HTMLObjectElement2_get_classid,
513     HTMLObjectElement2_put_data,
514     HTMLObjectElement2_get_data
515 };
516
517 static inline HTMLObjectElement *impl_from_HTMLDOMNode(HTMLDOMNode *iface)
518 {
519     return CONTAINING_RECORD(iface, HTMLObjectElement, plugin_container.element.node);
520 }
521
522 static HRESULT HTMLObjectElement_QI(HTMLDOMNode *iface, REFIID riid, void **ppv)
523 {
524     HTMLObjectElement *This = impl_from_HTMLDOMNode(iface);
525
526     if(IsEqualGUID(&IID_IUnknown, riid)) {
527         TRACE("(%p)->(IID_IUnknown %p)\n", This, ppv);
528         *ppv = &This->IHTMLObjectElement_iface;
529     }else if(IsEqualGUID(&IID_IDispatch, riid)) {
530         TRACE("(%p)->(IID_IDispatch %p)\n", This, ppv);
531         *ppv = &This->IHTMLObjectElement_iface;
532     }else if(IsEqualGUID(&IID_IHTMLObjectElement, riid)) {
533         TRACE("(%p)->(IID_IHTMLObjectElement %p)\n", This, ppv);
534         *ppv = &This->IHTMLObjectElement_iface;
535     }else if(IsEqualGUID(&IID_IHTMLObjectElement2, riid)) {
536         TRACE("(%p)->(IID_IHTMLObjectElement2 %p)\n", This, ppv);
537         *ppv = &This->IHTMLObjectElement2_iface;
538     }else if(IsEqualGUID(&IID_HTMLPluginContainer, riid)) {
539         TRACE("(%p)->(IID_HTMLPluginContainer %p)\n", This, ppv);
540         *ppv = &This->plugin_container;
541         return S_OK;
542     }else {
543         HRESULT hres;
544
545         hres = HTMLElement_QI(&This->plugin_container.element.node, riid, ppv);
546         if(hres == E_NOINTERFACE && This->plugin_container.plugin_host && This->plugin_container.plugin_host->plugin_unk) {
547             IUnknown *plugin_iface, *ret;
548
549             hres = IUnknown_QueryInterface(This->plugin_container.plugin_host->plugin_unk, riid, (void**)&plugin_iface);
550             if(hres == S_OK) {
551                 hres = wrap_iface(plugin_iface, (IUnknown*)&This->IHTMLObjectElement_iface, &ret);
552                 IUnknown_Release(plugin_iface);
553                 if(FAILED(hres))
554                     return hres;
555
556                 TRACE("returning plugin iface %p wrapped to %p\n", plugin_iface, ret);
557                 *ppv = ret;
558                 return S_OK;
559             }
560         }
561
562         return hres;
563     }
564
565     IUnknown_AddRef((IUnknown*)*ppv);
566     return S_OK;
567 }
568
569 static void HTMLObjectElement_destructor(HTMLDOMNode *iface)
570 {
571     HTMLObjectElement *This = impl_from_HTMLDOMNode(iface);
572
573     if(This->plugin_container.plugin_host)
574         detach_plugin_host(This->plugin_container.plugin_host);
575
576     HTMLElement_destructor(&This->plugin_container.element.node);
577 }
578
579 static HRESULT HTMLObjectElement_get_readystate(HTMLDOMNode *iface, BSTR *p)
580 {
581     HTMLObjectElement *This = impl_from_HTMLDOMNode(iface);
582     FIXME("(%p)->(%p)\n", This, p);
583     return E_NOTIMPL;
584 }
585
586 static HRESULT HTMLObjectElement_get_dispid(HTMLDOMNode *iface, BSTR name,
587         DWORD grfdex, DISPID *pid)
588 {
589     HTMLObjectElement *This = impl_from_HTMLDOMNode(iface);
590
591     TRACE("(%p)->(%s %x %p)\n", This, debugstr_w(name), grfdex, pid);
592
593     return get_plugin_dispid(&This->plugin_container, name, pid);
594 }
595
596 static HRESULT HTMLObjectElement_invoke(HTMLDOMNode *iface, DISPID id, LCID lcid,
597         WORD flags, DISPPARAMS *params, VARIANT *res, EXCEPINFO *ei, IServiceProvider *caller)
598 {
599     HTMLObjectElement *This = impl_from_HTMLDOMNode(iface);
600
601     TRACE("(%p)->(%d)\n", This, id);
602
603     return invoke_plugin_prop(&This->plugin_container, id, lcid, flags, params, res, ei);
604 }
605
606 static const NodeImplVtbl HTMLObjectElementImplVtbl = {
607     HTMLObjectElement_QI,
608     HTMLObjectElement_destructor,
609     HTMLElement_clone,
610     HTMLElement_get_attr_col,
611     NULL,
612     NULL,
613     NULL,
614     NULL,
615     NULL,
616     NULL,
617     HTMLObjectElement_get_readystate,
618     HTMLObjectElement_get_dispid,
619     HTMLObjectElement_invoke
620 };
621
622 static const tid_t HTMLObjectElement_iface_tids[] = {
623     HTMLELEMENT_TIDS,
624     IHTMLObjectElement_tid,
625     IHTMLObjectElement2_tid,
626     0
627 };
628 static dispex_static_data_t HTMLObjectElement_dispex = {
629     NULL,
630     DispHTMLObjectElement_tid,
631     NULL,
632     HTMLObjectElement_iface_tids
633 };
634
635 HRESULT HTMLObjectElement_Create(HTMLDocumentNode *doc, nsIDOMHTMLElement *nselem, HTMLElement **elem)
636 {
637     HTMLObjectElement *ret;
638     nsresult nsres;
639
640     ret = heap_alloc_zero(sizeof(*ret));
641     if(!ret)
642         return E_OUTOFMEMORY;
643
644     ret->IHTMLObjectElement_iface.lpVtbl = &HTMLObjectElementVtbl;
645     ret->IHTMLObjectElement2_iface.lpVtbl = &HTMLObjectElement2Vtbl;
646     ret->plugin_container.element.node.vtbl = &HTMLObjectElementImplVtbl;
647
648     HTMLElement_Init(&ret->plugin_container.element, doc, nselem, &HTMLObjectElement_dispex);
649
650     nsres = nsIDOMHTMLElement_QueryInterface(nselem, &IID_nsIDOMHTMLObjectElement, (void**)&ret->nsobject);
651
652     /* Share nsobject reference with nsnode */
653     assert(nsres == NS_OK && (nsIDOMNode*)ret->nsobject == ret->plugin_container.element.node.nsnode);
654     nsIDOMNode_Release(ret->plugin_container.element.node.nsnode);
655
656     *elem = &ret->plugin_container.element;
657     return S_OK;
658 }