kernel32: Return error on second attempt to free a module.
[wine] / dlls / mshtml / htmlimg.c
1 /*
2  * Copyright 2008 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
21 #define COBJMACROS
22
23 #include "windef.h"
24 #include "winbase.h"
25 #include "winuser.h"
26 #include "ole2.h"
27
28 #include "wine/debug.h"
29
30 #include "mshtml_private.h"
31
32 WINE_DEFAULT_DEBUG_CHANNEL(mshtml);
33
34 typedef struct {
35     HTMLElement element;
36
37     const IHTMLImgElementVtbl *lpHTMLImgElementVtbl;
38
39     nsIDOMHTMLImageElement *nsimg;
40 } HTMLImgElement;
41
42 #define HTMLIMG(x)  ((IHTMLImgElement*)  &(x)->lpHTMLImgElementVtbl)
43
44 #define HTMLIMG_THIS(iface) DEFINE_THIS(HTMLImgElement, HTMLImgElement, iface)
45
46 static HRESULT WINAPI HTMLImgElement_QueryInterface(IHTMLImgElement *iface, REFIID riid, void **ppv)
47 {
48     HTMLImgElement *This = HTMLIMG_THIS(iface);
49
50     return IHTMLDOMNode_QueryInterface(HTMLDOMNODE(&This->element.node), riid, ppv);
51 }
52
53 static ULONG WINAPI HTMLImgElement_AddRef(IHTMLImgElement *iface)
54 {
55     HTMLImgElement *This = HTMLIMG_THIS(iface);
56
57     return IHTMLDOMNode_AddRef(HTMLDOMNODE(&This->element.node));
58 }
59
60 static ULONG WINAPI HTMLImgElement_Release(IHTMLImgElement *iface)
61 {
62     HTMLImgElement *This = HTMLIMG_THIS(iface);
63
64     return IHTMLDOMNode_Release(HTMLDOMNODE(&This->element.node));
65 }
66
67 static HRESULT WINAPI HTMLImgElement_GetTypeInfoCount(IHTMLImgElement *iface, UINT *pctinfo)
68 {
69     HTMLImgElement *This = HTMLIMG_THIS(iface);
70     return IDispatchEx_GetTypeInfoCount(DISPATCHEX(&This->element.node.dispex), pctinfo);
71 }
72
73 static HRESULT WINAPI HTMLImgElement_GetTypeInfo(IHTMLImgElement *iface, UINT iTInfo,
74                                               LCID lcid, ITypeInfo **ppTInfo)
75 {
76     HTMLImgElement *This = HTMLIMG_THIS(iface);
77     return IDispatchEx_GetTypeInfo(DISPATCHEX(&This->element.node.dispex), iTInfo, lcid, ppTInfo);
78 }
79
80 static HRESULT WINAPI HTMLImgElement_GetIDsOfNames(IHTMLImgElement *iface, REFIID riid,
81                                                 LPOLESTR *rgszNames, UINT cNames,
82                                                 LCID lcid, DISPID *rgDispId)
83 {
84     HTMLImgElement *This = HTMLIMG_THIS(iface);
85     return IDispatchEx_GetIDsOfNames(DISPATCHEX(&This->element.node.dispex), riid, rgszNames, cNames, lcid, rgDispId);
86 }
87
88 static HRESULT WINAPI HTMLImgElement_Invoke(IHTMLImgElement *iface, DISPID dispIdMember,
89                             REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS *pDispParams,
90                             VARIANT *pVarResult, EXCEPINFO *pExcepInfo, UINT *puArgErr)
91 {
92     HTMLImgElement *This = HTMLIMG_THIS(iface);
93     return IDispatchEx_Invoke(DISPATCHEX(&This->element.node.dispex), dispIdMember, riid, lcid, wFlags, pDispParams,
94             pVarResult, pExcepInfo, puArgErr);
95 }
96
97 static HRESULT WINAPI HTMLImgElement_put_isMap(IHTMLImgElement *iface, VARIANT_BOOL v)
98 {
99     HTMLImgElement *This = HTMLIMG_THIS(iface);
100     FIXME("(%p)->(%x)\n", This, v);
101     return E_NOTIMPL;
102 }
103
104 static HRESULT WINAPI HTMLImgElement_get_isMap(IHTMLImgElement *iface, VARIANT_BOOL *p)
105 {
106     HTMLImgElement *This = HTMLIMG_THIS(iface);
107     FIXME("(%p)->(%p)\n", This, p);
108     return E_NOTIMPL;
109 }
110
111 static HRESULT WINAPI HTMLImgElement_put_useMap(IHTMLImgElement *iface, BSTR v)
112 {
113     HTMLImgElement *This = HTMLIMG_THIS(iface);
114     FIXME("(%p)->(%s)\n", This, debugstr_w(v));
115     return E_NOTIMPL;
116 }
117
118 static HRESULT WINAPI HTMLImgElement_get_useMap(IHTMLImgElement *iface, BSTR *p)
119 {
120     HTMLImgElement *This = HTMLIMG_THIS(iface);
121     FIXME("(%p)->(%p)\n", This, p);
122     return E_NOTIMPL;
123 }
124
125 static HRESULT WINAPI HTMLImgElement_get_mimeType(IHTMLImgElement *iface, BSTR *p)
126 {
127     HTMLImgElement *This = HTMLIMG_THIS(iface);
128     FIXME("(%p)->(%p)\n", This, p);
129     return E_NOTIMPL;
130 }
131
132 static HRESULT WINAPI HTMLImgElement_get_fileSize(IHTMLImgElement *iface, BSTR *p)
133 {
134     HTMLImgElement *This = HTMLIMG_THIS(iface);
135     FIXME("(%p)->(%p)\n", This, p);
136     return E_NOTIMPL;
137 }
138
139 static HRESULT WINAPI HTMLImgElement_get_fileCreatedDate(IHTMLImgElement *iface, BSTR *p)
140 {
141     HTMLImgElement *This = HTMLIMG_THIS(iface);
142     FIXME("(%p)->(%p)\n", This, p);
143     return E_NOTIMPL;
144 }
145
146 static HRESULT WINAPI HTMLImgElement_get_fileModifiedDate(IHTMLImgElement *iface, BSTR *p)
147 {
148     HTMLImgElement *This = HTMLIMG_THIS(iface);
149     FIXME("(%p)->(%p)\n", This, p);
150     return E_NOTIMPL;
151 }
152
153 static HRESULT WINAPI HTMLImgElement_get_fileUpdatedDate(IHTMLImgElement *iface, BSTR *p)
154 {
155     HTMLImgElement *This = HTMLIMG_THIS(iface);
156     FIXME("(%p)->(%p)\n", This, p);
157     return E_NOTIMPL;
158 }
159
160 static HRESULT WINAPI HTMLImgElement_get_protocol(IHTMLImgElement *iface, BSTR *p)
161 {
162     HTMLImgElement *This = HTMLIMG_THIS(iface);
163     FIXME("(%p)->(%p)\n", This, p);
164     return E_NOTIMPL;
165 }
166
167 static HRESULT WINAPI HTMLImgElement_get_href(IHTMLImgElement *iface, BSTR *p)
168 {
169     HTMLImgElement *This = HTMLIMG_THIS(iface);
170     FIXME("(%p)->(%p)\n", This, p);
171     return E_NOTIMPL;
172 }
173
174 static HRESULT WINAPI HTMLImgElement_get_nameProp(IHTMLImgElement *iface, BSTR *p)
175 {
176     HTMLImgElement *This = HTMLIMG_THIS(iface);
177     FIXME("(%p)->(%p)\n", This, p);
178     return E_NOTIMPL;
179 }
180
181 static HRESULT WINAPI HTMLImgElement_put_border(IHTMLImgElement *iface, VARIANT v)
182 {
183     HTMLImgElement *This = HTMLIMG_THIS(iface);
184     FIXME("(%p)->()\n", This);
185     return E_NOTIMPL;
186 }
187
188 static HRESULT WINAPI HTMLImgElement_get_border(IHTMLImgElement *iface, VARIANT *p)
189 {
190     HTMLImgElement *This = HTMLIMG_THIS(iface);
191     FIXME("(%p)->(%p)\n", This, p);
192     return E_NOTIMPL;
193 }
194
195 static HRESULT WINAPI HTMLImgElement_put_vspace(IHTMLImgElement *iface, LONG v)
196 {
197     HTMLImgElement *This = HTMLIMG_THIS(iface);
198     FIXME("(%p)->(%d)\n", This, v);
199     return E_NOTIMPL;
200 }
201
202 static HRESULT WINAPI HTMLImgElement_get_vspace(IHTMLImgElement *iface, LONG *p)
203 {
204     HTMLImgElement *This = HTMLIMG_THIS(iface);
205     FIXME("(%p)->(%p)\n", This, p);
206     return E_NOTIMPL;
207 }
208
209 static HRESULT WINAPI HTMLImgElement_put_hspace(IHTMLImgElement *iface, LONG v)
210 {
211     HTMLImgElement *This = HTMLIMG_THIS(iface);
212     FIXME("(%p)->(%d)\n", This, v);
213     return E_NOTIMPL;
214 }
215
216 static HRESULT WINAPI HTMLImgElement_get_hspace(IHTMLImgElement *iface, LONG *p)
217 {
218     HTMLImgElement *This = HTMLIMG_THIS(iface);
219     FIXME("(%p)->(%p)\n", This, p);
220     return E_NOTIMPL;
221 }
222
223 static HRESULT WINAPI HTMLImgElement_put_alt(IHTMLImgElement *iface, BSTR v)
224 {
225     HTMLImgElement *This = HTMLIMG_THIS(iface);
226     nsAString alt_str;
227     nsresult nsres;
228
229     TRACE("(%p)->(%s)\n", This, debugstr_w(v));
230
231     nsAString_Init(&alt_str, v);
232     nsres = nsIDOMHTMLImageElement_SetAlt(This->nsimg, &alt_str);
233     nsAString_Finish(&alt_str);
234     if(NS_FAILED(nsres))
235         ERR("SetAlt failed: %08x\n", nsres);
236
237     return S_OK;
238 }
239
240 static HRESULT WINAPI HTMLImgElement_get_alt(IHTMLImgElement *iface, BSTR *p)
241 {
242     HTMLImgElement *This = HTMLIMG_THIS(iface);
243     nsAString alt_str;
244     nsresult nsres;
245
246     TRACE("(%p)->(%p)\n", This, p);
247
248     nsAString_Init(&alt_str, NULL);
249     nsres = nsIDOMHTMLImageElement_GetAlt(This->nsimg, &alt_str);
250     if(NS_SUCCEEDED(nsres)) {
251         const PRUnichar *alt;
252
253         nsAString_GetData(&alt_str, &alt);
254         *p = *alt ? SysAllocString(alt) : NULL;
255     }else {
256         ERR("GetAlt failed: %08x\n", nsres);
257     }
258     nsAString_Finish(&alt_str);
259
260     return NS_SUCCEEDED(nsres) ? S_OK : E_FAIL;
261 }
262
263 static HRESULT WINAPI HTMLImgElement_put_src(IHTMLImgElement *iface, BSTR v)
264 {
265     HTMLImgElement *This = HTMLIMG_THIS(iface);
266     nsAString src_str;
267     nsresult nsres;
268
269     TRACE("(%p)->(%s)\n", This, debugstr_w(v));
270
271     nsAString_Init(&src_str, v);
272     nsres = nsIDOMHTMLImageElement_SetSrc(This->nsimg, &src_str);
273     nsAString_Finish(&src_str);
274     if(NS_FAILED(nsres))
275         ERR("SetSrc failed: %08x\n", nsres);
276
277     return NS_OK;
278 }
279
280 static HRESULT WINAPI HTMLImgElement_get_src(IHTMLImgElement *iface, BSTR *p)
281 {
282     HTMLImgElement *This = HTMLIMG_THIS(iface);
283     const PRUnichar *src;
284     nsAString src_str;
285     nsresult nsres;
286     HRESULT hres;
287
288     TRACE("(%p)->(%p)\n", This, p);
289
290     nsAString_Init(&src_str, NULL);
291     nsres = nsIDOMHTMLImageElement_GetSrc(This->nsimg, &src_str);
292     if(NS_FAILED(nsres)) {
293         ERR("GetSrc failed: %08x\n", nsres);
294         return E_FAIL;
295     }
296
297     nsAString_GetData(&src_str, &src);
298     hres = nsuri_to_url(src, TRUE, p);
299     nsAString_Finish(&src_str);
300
301     return hres;
302 }
303
304 static HRESULT WINAPI HTMLImgElement_put_lowsrc(IHTMLImgElement *iface, BSTR v)
305 {
306     HTMLImgElement *This = HTMLIMG_THIS(iface);
307     FIXME("(%p)->(%s)\n", This, debugstr_w(v));
308     return E_NOTIMPL;
309 }
310
311 static HRESULT WINAPI HTMLImgElement_get_lowsrc(IHTMLImgElement *iface, BSTR *p)
312 {
313     HTMLImgElement *This = HTMLIMG_THIS(iface);
314     FIXME("(%p)->(%p)\n", This, p);
315     return E_NOTIMPL;
316 }
317
318 static HRESULT WINAPI HTMLImgElement_put_vrml(IHTMLImgElement *iface, BSTR v)
319 {
320     HTMLImgElement *This = HTMLIMG_THIS(iface);
321     FIXME("(%p)->(%s)\n", This, debugstr_w(v));
322     return E_NOTIMPL;
323 }
324
325 static HRESULT WINAPI HTMLImgElement_get_vrml(IHTMLImgElement *iface, BSTR *p)
326 {
327     HTMLImgElement *This = HTMLIMG_THIS(iface);
328     FIXME("(%p)->(%p)\n", This, p);
329     return E_NOTIMPL;
330 }
331
332 static HRESULT WINAPI HTMLImgElement_put_dynsrc(IHTMLImgElement *iface, BSTR v)
333 {
334     HTMLImgElement *This = HTMLIMG_THIS(iface);
335     FIXME("(%p)->(%s)\n", This, debugstr_w(v));
336     return E_NOTIMPL;
337 }
338
339 static HRESULT WINAPI HTMLImgElement_get_dynsrc(IHTMLImgElement *iface, BSTR *p)
340 {
341     HTMLImgElement *This = HTMLIMG_THIS(iface);
342     FIXME("(%p)->(%p)\n", This, p);
343     return E_NOTIMPL;
344 }
345
346 static HRESULT WINAPI HTMLImgElement_get_readyState(IHTMLImgElement *iface, BSTR *p)
347 {
348     HTMLImgElement *This = HTMLIMG_THIS(iface);
349     FIXME("(%p)->(%p)\n", This, p);
350     return E_NOTIMPL;
351 }
352
353 static HRESULT WINAPI HTMLImgElement_get_complete(IHTMLImgElement *iface, VARIANT_BOOL *p)
354 {
355     HTMLImgElement *This = HTMLIMG_THIS(iface);
356     FIXME("(%p)->(%p)\n", This, p);
357     return E_NOTIMPL;
358 }
359
360 static HRESULT WINAPI HTMLImgElement_put_loop(IHTMLImgElement *iface, VARIANT v)
361 {
362     HTMLImgElement *This = HTMLIMG_THIS(iface);
363     FIXME("(%p)->()\n", This);
364     return E_NOTIMPL;
365 }
366
367 static HRESULT WINAPI HTMLImgElement_get_loop(IHTMLImgElement *iface, VARIANT *p)
368 {
369     HTMLImgElement *This = HTMLIMG_THIS(iface);
370     FIXME("(%p)->(%p)\n", This, p);
371     return E_NOTIMPL;
372 }
373
374 static HRESULT WINAPI HTMLImgElement_put_align(IHTMLImgElement *iface, BSTR v)
375 {
376     HTMLImgElement *This = HTMLIMG_THIS(iface);
377     FIXME("(%p)->(%s)\n", This, debugstr_w(v));
378     return E_NOTIMPL;
379 }
380
381 static HRESULT WINAPI HTMLImgElement_get_align(IHTMLImgElement *iface, BSTR *p)
382 {
383     HTMLImgElement *This = HTMLIMG_THIS(iface);
384     FIXME("(%p)->(%p)\n", This, p);
385     return E_NOTIMPL;
386 }
387
388 static HRESULT WINAPI HTMLImgElement_put_onload(IHTMLImgElement *iface, VARIANT v)
389 {
390     HTMLImgElement *This = HTMLIMG_THIS(iface);
391     FIXME("(%p)->()\n", This);
392     return E_NOTIMPL;
393 }
394
395 static HRESULT WINAPI HTMLImgElement_get_onload(IHTMLImgElement *iface, VARIANT *p)
396 {
397     HTMLImgElement *This = HTMLIMG_THIS(iface);
398     FIXME("(%p)->(%p)\n", This, p);
399     return E_NOTIMPL;
400 }
401
402 static HRESULT WINAPI HTMLImgElement_put_onerror(IHTMLImgElement *iface, VARIANT v)
403 {
404     HTMLImgElement *This = HTMLIMG_THIS(iface);
405     FIXME("(%p)->()\n", This);
406     return E_NOTIMPL;
407 }
408
409 static HRESULT WINAPI HTMLImgElement_get_onerror(IHTMLImgElement *iface, VARIANT *p)
410 {
411     HTMLImgElement *This = HTMLIMG_THIS(iface);
412     FIXME("(%p)->(%p)\n", This, p);
413     return E_NOTIMPL;
414 }
415
416 static HRESULT WINAPI HTMLImgElement_put_onabort(IHTMLImgElement *iface, VARIANT v)
417 {
418     HTMLImgElement *This = HTMLIMG_THIS(iface);
419     FIXME("(%p)->()\n", This);
420     return E_NOTIMPL;
421 }
422
423 static HRESULT WINAPI HTMLImgElement_get_onabort(IHTMLImgElement *iface, VARIANT *p)
424 {
425     HTMLImgElement *This = HTMLIMG_THIS(iface);
426     FIXME("(%p)->(%p)\n", This, p);
427     return E_NOTIMPL;
428 }
429
430 static HRESULT WINAPI HTMLImgElement_put_name(IHTMLImgElement *iface, BSTR v)
431 {
432     HTMLImgElement *This = HTMLIMG_THIS(iface);
433     FIXME("(%p)->(%s)\n", This, debugstr_w(v));
434     return E_NOTIMPL;
435 }
436
437 static HRESULT WINAPI HTMLImgElement_get_name(IHTMLImgElement *iface, BSTR *p)
438 {
439     HTMLImgElement *This = HTMLIMG_THIS(iface);
440     nsAString strName;
441     nsresult nsres;
442
443     TRACE("(%p)->(%p)\n", This, p);
444
445     nsAString_Init(&strName, NULL);
446     nsres = nsIDOMHTMLImageElement_GetName(This->nsimg, &strName);
447     if(NS_SUCCEEDED(nsres)) {
448         const PRUnichar *str;
449
450         nsAString_GetData(&strName, &str);
451         *p = *str ? SysAllocString(str) : NULL;
452     }else {
453         ERR("GetName failed: %08x\n", nsres);
454     }
455     nsAString_Finish(&strName);
456
457     return NS_SUCCEEDED(nsres) ? S_OK : E_FAIL;
458 }
459
460 static HRESULT WINAPI HTMLImgElement_put_width(IHTMLImgElement *iface, LONG v)
461 {
462     HTMLImgElement *This = HTMLIMG_THIS(iface);
463     FIXME("(%p)->(%d)\n", This, v);
464     return E_NOTIMPL;
465 }
466
467 static HRESULT WINAPI HTMLImgElement_get_width(IHTMLImgElement *iface, LONG *p)
468 {
469     HTMLImgElement *This = HTMLIMG_THIS(iface);
470     FIXME("(%p)->(%p)\n", This, p);
471     return E_NOTIMPL;
472 }
473
474 static HRESULT WINAPI HTMLImgElement_put_height(IHTMLImgElement *iface, LONG v)
475 {
476     HTMLImgElement *This = HTMLIMG_THIS(iface);
477     FIXME("(%p)->(%d)\n", This, v);
478     return E_NOTIMPL;
479 }
480
481 static HRESULT WINAPI HTMLImgElement_get_height(IHTMLImgElement *iface, LONG *p)
482 {
483     HTMLImgElement *This = HTMLIMG_THIS(iface);
484     FIXME("(%p)->(%p)\n", This, p);
485     return E_NOTIMPL;
486 }
487
488 static HRESULT WINAPI HTMLImgElement_put_start(IHTMLImgElement *iface, BSTR v)
489 {
490     HTMLImgElement *This = HTMLIMG_THIS(iface);
491     FIXME("(%p)->()\n", This);
492     return E_NOTIMPL;
493 }
494
495 static HRESULT WINAPI HTMLImgElement_get_start(IHTMLImgElement *iface, BSTR *p)
496 {
497     HTMLImgElement *This = HTMLIMG_THIS(iface);
498     FIXME("(%p)->(%p)\n", This, p);
499     return E_NOTIMPL;
500 }
501
502 #undef HTMLIMG_THIS
503
504 static const IHTMLImgElementVtbl HTMLImgElementVtbl = {
505     HTMLImgElement_QueryInterface,
506     HTMLImgElement_AddRef,
507     HTMLImgElement_Release,
508     HTMLImgElement_GetTypeInfoCount,
509     HTMLImgElement_GetTypeInfo,
510     HTMLImgElement_GetIDsOfNames,
511     HTMLImgElement_Invoke,
512     HTMLImgElement_put_isMap,
513     HTMLImgElement_get_isMap,
514     HTMLImgElement_put_useMap,
515     HTMLImgElement_get_useMap,
516     HTMLImgElement_get_mimeType,
517     HTMLImgElement_get_fileSize,
518     HTMLImgElement_get_fileCreatedDate,
519     HTMLImgElement_get_fileModifiedDate,
520     HTMLImgElement_get_fileUpdatedDate,
521     HTMLImgElement_get_protocol,
522     HTMLImgElement_get_href,
523     HTMLImgElement_get_nameProp,
524     HTMLImgElement_put_border,
525     HTMLImgElement_get_border,
526     HTMLImgElement_put_vspace,
527     HTMLImgElement_get_vspace,
528     HTMLImgElement_put_hspace,
529     HTMLImgElement_get_hspace,
530     HTMLImgElement_put_alt,
531     HTMLImgElement_get_alt,
532     HTMLImgElement_put_src,
533     HTMLImgElement_get_src,
534     HTMLImgElement_put_lowsrc,
535     HTMLImgElement_get_lowsrc,
536     HTMLImgElement_put_vrml,
537     HTMLImgElement_get_vrml,
538     HTMLImgElement_put_dynsrc,
539     HTMLImgElement_get_dynsrc,
540     HTMLImgElement_get_readyState,
541     HTMLImgElement_get_complete,
542     HTMLImgElement_put_loop,
543     HTMLImgElement_get_loop,
544     HTMLImgElement_put_align,
545     HTMLImgElement_get_align,
546     HTMLImgElement_put_onload,
547     HTMLImgElement_get_onload,
548     HTMLImgElement_put_onerror,
549     HTMLImgElement_get_onerror,
550     HTMLImgElement_put_onabort,
551     HTMLImgElement_get_onabort,
552     HTMLImgElement_put_name,
553     HTMLImgElement_get_name,
554     HTMLImgElement_put_width,
555     HTMLImgElement_get_width,
556     HTMLImgElement_put_height,
557     HTMLImgElement_get_height,
558     HTMLImgElement_put_start,
559     HTMLImgElement_get_start
560 };
561
562 #define HTMLIMG_NODE_THIS(iface) DEFINE_THIS2(HTMLImgElement, element.node, iface)
563
564 static HRESULT HTMLImgElement_QI(HTMLDOMNode *iface, REFIID riid, void **ppv)
565 {
566     HTMLImgElement *This = HTMLIMG_NODE_THIS(iface);
567
568     *ppv = NULL;
569
570     if(IsEqualGUID(&IID_IHTMLImgElement, riid)) {
571         TRACE("(%p)->(IID_IHTMLImgElement %p)\n", This, ppv);
572         *ppv = HTMLIMG(This);
573     }else {
574         return HTMLElement_QI(&This->element.node, riid, ppv);
575     }
576
577     IUnknown_AddRef((IUnknown*)*ppv);
578     return S_OK;
579 }
580
581 static void HTMLImgElement_destructor(HTMLDOMNode *iface)
582 {
583     HTMLImgElement *This = HTMLIMG_NODE_THIS(iface);
584
585     if(This->nsimg)
586         nsIDOMHTMLImageElement_Release(This->nsimg);
587
588     HTMLElement_destructor(&This->element.node);
589 }
590
591 static HRESULT HTMLImgElement_get_readystate(HTMLDOMNode *iface, BSTR *p)
592 {
593     HTMLImgElement *This = HTMLIMG_NODE_THIS(iface);
594
595     return IHTMLImgElement_get_readyState(HTMLIMG(This), p);
596 }
597
598 #undef HTMLIMG_NODE_THIS
599
600 static const NodeImplVtbl HTMLImgElementImplVtbl = {
601     HTMLImgElement_QI,
602     HTMLImgElement_destructor,
603     NULL,
604     NULL,
605     NULL,
606     NULL,
607     NULL,
608     HTMLImgElement_get_readystate
609 };
610
611 static const tid_t HTMLImgElement_iface_tids[] = {
612     IHTMLDOMNode_tid,
613     IHTMLDOMNode2_tid,
614     IHTMLElement_tid,
615     IHTMLElement2_tid,
616     IHTMLElement3_tid,
617     IHTMLImgElement_tid,
618     0
619 };
620 static dispex_static_data_t HTMLImgElement_dispex = {
621     NULL,
622     DispHTMLImg_tid,
623     NULL,
624     HTMLImgElement_iface_tids
625 };
626
627 HTMLElement *HTMLImgElement_Create(HTMLDocumentNode *doc, nsIDOMHTMLElement *nselem)
628 {
629     HTMLImgElement *ret = heap_alloc_zero(sizeof(HTMLImgElement));
630     nsresult nsres;
631
632     ret->lpHTMLImgElementVtbl = &HTMLImgElementVtbl;
633     ret->element.node.vtbl = &HTMLImgElementImplVtbl;
634
635     nsres = nsIDOMHTMLElement_QueryInterface(nselem, &IID_nsIDOMHTMLImageElement, (void**)&ret->nsimg);
636     if(NS_FAILED(nsres))
637         ERR("Could not get nsIDOMHTMLImageElement: %08x\n", nsres);
638
639     HTMLElement_Init(&ret->element, doc, nselem, &HTMLImgElement_dispex);
640
641     return &ret->element;
642 }
643
644 #define HTMLIMGFACTORY_THIS(iface) DEFINE_THIS(HTMLImageElementFactory, HTMLImageElementFactory, iface)
645
646 static HRESULT WINAPI HTMLImageElementFactory_QueryInterface(IHTMLImageElementFactory *iface,
647         REFIID riid, void **ppv)
648 {
649     HTMLImageElementFactory *This = HTMLIMGFACTORY_THIS(iface);
650
651     *ppv = NULL;
652
653     if(IsEqualGUID(&IID_IUnknown, riid)) {
654         TRACE("(%p)->(IID_Unknown %p)\n", This, ppv);
655         *ppv = HTMLIMGFACTORY(This);
656     }else if(IsEqualGUID(&IID_IHTMLImageElementFactory, riid)) {
657         TRACE("(%p)->(IID_IHTMLImageElementFactory %p)\n", This, ppv);
658         *ppv = HTMLIMGFACTORY(This);
659     }else if(dispex_query_interface(&This->dispex, riid, ppv))
660         return *ppv ? S_OK : E_NOINTERFACE;
661
662     if(*ppv) {
663         IUnknown_AddRef((IUnknown*)*ppv);
664         return S_OK;
665     }
666
667     WARN("(%p)->(%s %p)\n", This, debugstr_guid(riid), ppv);
668     return E_NOINTERFACE;
669 }
670
671 static ULONG WINAPI HTMLImageElementFactory_AddRef(IHTMLImageElementFactory *iface)
672 {
673     HTMLImageElementFactory *This = HTMLIMGFACTORY_THIS(iface);
674     LONG ref = InterlockedIncrement(&This->ref);
675
676     TRACE("(%p) ref=%d\n", This, ref);
677
678     return ref;
679 }
680
681 static ULONG WINAPI HTMLImageElementFactory_Release(IHTMLImageElementFactory *iface)
682 {
683     HTMLImageElementFactory *This = HTMLIMGFACTORY_THIS(iface);
684     LONG ref = InterlockedDecrement(&This->ref);
685
686     TRACE("(%p) ref=%d\n", This, ref);
687
688     if(!ref)
689         heap_free(This);
690
691     return ref;
692 }
693
694 static HRESULT WINAPI HTMLImageElementFactory_GetTypeInfoCount(IHTMLImageElementFactory *iface,
695         UINT *pctinfo)
696 {
697     HTMLImageElementFactory *This = HTMLIMGFACTORY_THIS(iface);
698     FIXME("(%p)->(%p)\n", This, pctinfo);
699     return E_NOTIMPL;
700 }
701
702 static HRESULT WINAPI HTMLImageElementFactory_GetTypeInfo(IHTMLImageElementFactory *iface,
703         UINT iTInfo, LCID lcid, ITypeInfo **ppTInfo)
704 {
705     HTMLImageElementFactory *This = HTMLIMGFACTORY_THIS(iface);
706     FIXME("(%p)->(%u %u %p)\n", This, iTInfo, lcid, ppTInfo);
707     return E_NOTIMPL;
708 }
709
710 static HRESULT WINAPI HTMLImageElementFactory_GetIDsOfNames(IHTMLImageElementFactory *iface,
711         REFIID riid, LPOLESTR *rgszNames, UINT cNames, LCID lcid,
712         DISPID *rgDispId)
713 {
714     HTMLImageElementFactory *This = HTMLIMGFACTORY_THIS(iface);
715     FIXME("(%p)->(%s %p %u %u %p)\n", This, debugstr_guid(riid), rgszNames,
716             cNames, lcid, rgDispId);
717     return E_NOTIMPL;
718 }
719
720 static HRESULT WINAPI HTMLImageElementFactory_Invoke(IHTMLImageElementFactory *iface,
721         DISPID dispIdMember, REFIID riid, LCID lcid, WORD wFlags,
722         DISPPARAMS *pDispParams, VARIANT *pVarResult, EXCEPINFO *pExcepInfo,
723         UINT *puArgErr)
724 {
725     HTMLImageElementFactory *This = HTMLIMGFACTORY_THIS(iface);
726     FIXME("(%p)->(%d %s %d %d %p %p %p %p)\n", This, dispIdMember, debugstr_guid(riid),
727             lcid, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr);
728     return E_NOTIMPL;
729 }
730
731 static HRESULT WINAPI HTMLImageElementFactory_create(IHTMLImageElementFactory *iface,
732         VARIANT width, VARIANT height, IHTMLImgElement **img_elem)
733 {
734     HTMLImageElementFactory *This = HTMLIMGFACTORY_THIS(iface);
735     HTMLElement *elem;
736     nsIDOMHTMLElement *nselem;
737     HRESULT hres;
738
739     static const PRUnichar imgW[] = {'I','M','G',0};
740
741     TRACE("(%p)->(%s %s %p)\n", This, debugstr_variant(&width),
742             debugstr_variant(&height), img_elem);
743
744     if(!This->window || !This->window->doc) {
745         WARN("NULL doc\n");
746         return E_UNEXPECTED;
747     }
748
749     *img_elem = NULL;
750
751     hres = create_nselem(This->window->doc, imgW, &nselem);
752     if(FAILED(hres))
753         return hres;
754
755     elem = HTMLElement_Create(This->window->doc, (nsIDOMNode*)nselem, FALSE);
756     if(!elem) {
757         ERR("HTMLElement_Create failed\n");
758         return E_FAIL;
759     }
760
761     hres = IHTMLElement_QueryInterface(HTMLELEM(elem), &IID_IHTMLImgElement, (void**)img_elem);
762     if(FAILED(hres)) {
763         ERR("IHTMLElement_QueryInterface failed: 0x%08x\n", hres);
764         return hres;
765     }
766
767     nsIDOMHTMLElement_Release(nselem);
768
769     if(V_VT(&width) != VT_EMPTY || V_VT(&height) != VT_EMPTY)
770         FIXME("Not setting image dimensions\n");
771
772     return S_OK;
773 }
774
775 static HRESULT HTMLImageElementFactory_value(IUnknown *iface, LCID lcid,
776         WORD flags, DISPPARAMS *params, VARIANT *res, EXCEPINFO *ei,
777         IServiceProvider *caller)
778 {
779     HTMLImageElementFactory *This = HTMLIMGFACTORY_THIS(iface);
780     IHTMLImgElement *img;
781     VARIANT empty, *width, *height;
782     HRESULT hres;
783     int argc = params->cArgs - params->cNamedArgs;
784
785     V_VT(res) = VT_NULL;
786
787     V_VT(&empty) = VT_EMPTY;
788
789     width = argc >= 1 ? params->rgvarg + (params->cArgs - 1) : &empty;
790     height = argc >= 2 ? params->rgvarg + (params->cArgs - 2) : &empty;
791
792     hres = IHTMLImageElementFactory_create(HTMLIMGFACTORY(This), *width, *height, &img);
793     if(FAILED(hres))
794         return hres;
795
796     V_VT(res) = VT_DISPATCH;
797     V_DISPATCH(res) = (IDispatch*)img;
798
799     return S_OK;
800 }
801
802 #undef HTMLIMGFACTORY_THIS
803
804 static const IHTMLImageElementFactoryVtbl HTMLImageElementFactoryVtbl = {
805     HTMLImageElementFactory_QueryInterface,
806     HTMLImageElementFactory_AddRef,
807     HTMLImageElementFactory_Release,
808     HTMLImageElementFactory_GetTypeInfoCount,
809     HTMLImageElementFactory_GetTypeInfo,
810     HTMLImageElementFactory_GetIDsOfNames,
811     HTMLImageElementFactory_Invoke,
812     HTMLImageElementFactory_create
813 };
814
815 static const tid_t HTMLImageElementFactory_iface_tids[] = {
816     IHTMLImageElementFactory_tid,
817     0
818 };
819
820 static const dispex_static_data_vtbl_t HTMLImageElementFactory_dispex_vtbl = {
821     HTMLImageElementFactory_value,
822     NULL,
823     NULL
824 };
825
826 static dispex_static_data_t HTMLImageElementFactory_dispex = {
827     &HTMLImageElementFactory_dispex_vtbl,
828     IHTMLImageElementFactory_tid,
829     NULL,
830     HTMLImageElementFactory_iface_tids
831 };
832
833 HTMLImageElementFactory *HTMLImageElementFactory_Create(HTMLWindow *window)
834 {
835     HTMLImageElementFactory *ret;
836
837     ret = heap_alloc(sizeof(HTMLImageElementFactory));
838
839     ret->lpHTMLImageElementFactoryVtbl = &HTMLImageElementFactoryVtbl;
840     ret->ref = 1;
841     ret->window = window;
842
843     init_dispex(&ret->dispex, (IUnknown*)HTMLIMGFACTORY(ret), &HTMLImageElementFactory_dispex);
844
845     return ret;
846 }