mshtml: Added createDocumentFragment tests.
[wine] / dlls / mshtml / htmlform.c
1 /*
2  * Copyright 2009 Andrew Eikum 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 struct HTMLFormElement {
35     HTMLElement element;
36
37     const IHTMLFormElementVtbl *lpHTMLFormElementVtbl;
38
39     nsIDOMHTMLFormElement *nsform;
40 };
41
42 #define HTMLFORM(x)  (&(x)->lpHTMLFormElementVtbl)
43
44 static HRESULT htmlform_item(HTMLFormElement *This, int i, IDispatch **ret)
45 {
46     nsIDOMHTMLCollection *elements;
47     nsIDOMNode *item;
48     HTMLDOMNode *node;
49     nsresult nsres;
50
51     nsres = nsIDOMHTMLFormElement_GetElements(This->nsform, &elements);
52     if(NS_FAILED(nsres)) {
53         FIXME("GetElements failed: 0x%08x\n", nsres);
54         return E_FAIL;
55     }
56
57     nsres = nsIDOMHTMLCollection_Item(elements, i, &item);
58     nsIDOMHTMLCollection_Release(elements);
59     if(NS_FAILED(nsres)) {
60         FIXME("Item failed: 0x%08x\n", nsres);
61         return E_FAIL;
62     }
63
64     if(item) {
65         node = get_node(This->element.node.doc, item, TRUE);
66         if(!node)
67             return E_OUTOFMEMORY;
68
69         IHTMLDOMNode_AddRef(HTMLDOMNODE(node));
70         nsIDOMNode_Release(item);
71         *ret = (IDispatch*)HTMLDOMNODE(node);
72     }else {
73         *ret = NULL;
74     }
75
76     return S_OK;
77 }
78
79 #define HTMLFORM_THIS(iface) DEFINE_THIS(HTMLFormElement, HTMLFormElement, iface)
80
81 static HRESULT WINAPI HTMLFormElement_QueryInterface(IHTMLFormElement *iface,
82         REFIID riid, void **ppv)
83 {
84     HTMLFormElement *This = HTMLFORM_THIS(iface);
85
86     return IHTMLDOMNode_QueryInterface(HTMLDOMNODE(&This->element.node), riid, ppv);
87 }
88
89 static ULONG WINAPI HTMLFormElement_AddRef(IHTMLFormElement *iface)
90 {
91     HTMLFormElement *This = HTMLFORM_THIS(iface);
92
93     return IHTMLDOMNode_AddRef(HTMLDOMNODE(&This->element.node));
94 }
95
96 static ULONG WINAPI HTMLFormElement_Release(IHTMLFormElement *iface)
97 {
98     HTMLFormElement *This = HTMLFORM_THIS(iface);
99
100     return IHTMLDOMNode_Release(HTMLDOMNODE(&This->element.node));
101 }
102
103 static HRESULT WINAPI HTMLFormElement_GetTypeInfoCount(IHTMLFormElement *iface, UINT *pctinfo)
104 {
105     HTMLFormElement *This = HTMLFORM_THIS(iface);
106     return IDispatchEx_GetTypeInfoCount(DISPATCHEX(&This->element.node.dispex), pctinfo);
107 }
108
109 static HRESULT WINAPI HTMLFormElement_GetTypeInfo(IHTMLFormElement *iface, UINT iTInfo,
110                                               LCID lcid, ITypeInfo **ppTInfo)
111 {
112     HTMLFormElement *This = HTMLFORM_THIS(iface);
113     return IDispatchEx_GetTypeInfo(DISPATCHEX(&This->element.node.dispex), iTInfo, lcid, ppTInfo);
114 }
115
116 static HRESULT WINAPI HTMLFormElement_GetIDsOfNames(IHTMLFormElement *iface, REFIID riid,
117                                                 LPOLESTR *rgszNames, UINT cNames,
118                                                 LCID lcid, DISPID *rgDispId)
119 {
120     HTMLFormElement *This = HTMLFORM_THIS(iface);
121     return IDispatchEx_GetIDsOfNames(DISPATCHEX(&This->element.node.dispex), riid, rgszNames, cNames, lcid, rgDispId);
122 }
123
124 static HRESULT WINAPI HTMLFormElement_Invoke(IHTMLFormElement *iface, DISPID dispIdMember,
125                             REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS *pDispParams,
126                             VARIANT *pVarResult, EXCEPINFO *pExcepInfo, UINT *puArgErr)
127 {
128     HTMLFormElement *This = HTMLFORM_THIS(iface);
129     return IDispatchEx_Invoke(DISPATCHEX(&This->element.node.dispex), dispIdMember, riid, lcid, wFlags, pDispParams,
130             pVarResult, pExcepInfo, puArgErr);
131 }
132
133 static HRESULT WINAPI HTMLFormElement_put_action(IHTMLFormElement *iface, BSTR v)
134 {
135     HTMLFormElement *This = HTMLFORM_THIS(iface);
136     nsAString action_str;
137     nsresult nsres;
138
139     TRACE("(%p)->(%s)\n", This, wine_dbgstr_w(v));
140
141     nsAString_InitDepend(&action_str, v);
142     nsres = nsIDOMHTMLFormElement_SetAction(This->nsform, &action_str);
143     nsAString_Finish(&action_str);
144     if(NS_FAILED(nsres)) {
145         ERR("SetAction failed: %08x\n", nsres);
146         return E_FAIL;
147     }
148
149     return S_OK;
150 }
151
152 static HRESULT WINAPI HTMLFormElement_get_action(IHTMLFormElement *iface, BSTR *p)
153 {
154     HTMLFormElement *This = HTMLFORM_THIS(iface);
155     nsAString action_str;
156     nsresult nsres;
157     HRESULT hres;
158
159     TRACE("(%p)->(%p)\n", This, p);
160
161     nsAString_Init(&action_str, NULL);
162     nsres = nsIDOMHTMLFormElement_GetAction(This->nsform, &action_str);
163     if(NS_SUCCEEDED(nsres)) {
164         const PRUnichar *action;
165         nsAString_GetData(&action_str, &action);
166         hres = nsuri_to_url(action, FALSE, p);
167     }else {
168         ERR("GetAction failed: %08x\n", nsres);
169         hres = E_FAIL;
170     }
171
172     return hres;
173 }
174
175 static HRESULT WINAPI HTMLFormElement_put_dir(IHTMLFormElement *iface, BSTR v)
176 {
177     HTMLFormElement *This = HTMLFORM_THIS(iface);
178     FIXME("(%p)->(%s)\n", This, wine_dbgstr_w(v));
179     return E_NOTIMPL;
180 }
181
182 static HRESULT WINAPI HTMLFormElement_get_dir(IHTMLFormElement *iface, BSTR *p)
183 {
184     HTMLFormElement *This = HTMLFORM_THIS(iface);
185     FIXME("(%p)->(%p)\n", This, p);
186     return E_NOTIMPL;
187 }
188
189 static HRESULT WINAPI HTMLFormElement_put_encoding(IHTMLFormElement *iface, BSTR v)
190 {
191     static const WCHAR urlencodedW[] = {'a','p','p','l','i','c','a','t','i','o','n','/',
192         'x','-','w','w','w','-','f','o','r','m','-','u','r','l','e','n','c','o','d','e','d',0};
193     static const WCHAR dataW[] = {'m','u','l','t','i','p','a','r','t','/',
194         'f','o','r','m','-','d','a','t','a',0};
195     static const WCHAR plainW[] = {'t','e','x','t','/','p','l','a','i','n',0};
196
197     HTMLFormElement *This = HTMLFORM_THIS(iface);
198     nsAString encoding_str;
199     nsresult nsres;
200
201     TRACE("(%p)->(%s)\n", This, wine_dbgstr_w(v));
202
203     if(lstrcmpiW(v, urlencodedW) && lstrcmpiW(v, dataW) && lstrcmpiW(v, plainW)) {
204         WARN("incorrect enctype\n");
205         return E_INVALIDARG;
206     }
207
208     nsAString_InitDepend(&encoding_str, v);
209     nsres = nsIDOMHTMLFormElement_SetEnctype(This->nsform, &encoding_str);
210     nsAString_Finish(&encoding_str);
211     if(NS_FAILED(nsres))
212         return E_FAIL;
213
214     return S_OK;
215 }
216
217 static HRESULT WINAPI HTMLFormElement_get_encoding(IHTMLFormElement *iface, BSTR *p)
218 {
219     HTMLFormElement *This = HTMLFORM_THIS(iface);
220     nsAString encoding_str;
221     nsresult nsres;
222
223     TRACE("(%p)->(%p)\n", This, p);
224
225     nsAString_Init(&encoding_str, NULL);
226     nsres = nsIDOMHTMLFormElement_GetEnctype(This->nsform, &encoding_str);
227     if(NS_SUCCEEDED(nsres)) {
228         const PRUnichar *encoding;
229         nsAString_GetData(&encoding_str, &encoding);
230
231         *p = SysAllocString(encoding);
232         if(!*p)
233             return E_OUTOFMEMORY;
234     }else
235         return E_FAIL;
236
237     return S_OK;
238 }
239
240 static HRESULT WINAPI HTMLFormElement_put_method(IHTMLFormElement *iface, BSTR v)
241 {
242     static const WCHAR postW[] = {'P','O','S','T',0};
243     static const WCHAR getW[] = {'G','E','T',0};
244
245     HTMLFormElement *This = HTMLFORM_THIS(iface);
246     nsAString method_str;
247     nsresult nsres;
248
249     TRACE("(%p)->(%s)\n", This, wine_dbgstr_w(v));
250
251     if(lstrcmpiW(v, postW) && lstrcmpiW(v, getW)) {
252         WARN("unrecognized method\n");
253         return E_INVALIDARG;
254     }
255
256     nsAString_InitDepend(&method_str, v);
257     nsres = nsIDOMHTMLFormElement_SetMethod(This->nsform, &method_str);
258     nsAString_Finish(&method_str);
259     if(NS_FAILED(nsres))
260         return E_FAIL;
261
262     return S_OK;
263 }
264
265 static HRESULT WINAPI HTMLFormElement_get_method(IHTMLFormElement *iface, BSTR *p)
266 {
267     HTMLFormElement *This = HTMLFORM_THIS(iface);
268     nsAString method_str;
269     nsresult nsres;
270
271     TRACE("(%p)->(%p)\n", This, p);
272
273     nsAString_Init(&method_str, NULL);
274     nsres = nsIDOMHTMLFormElement_GetMethod(This->nsform, &method_str);
275     if(NS_SUCCEEDED(nsres)) {
276         const PRUnichar *method;
277         nsAString_GetData(&method_str, &method);
278
279         *p = SysAllocString(method);
280         if(!*p)
281             return E_OUTOFMEMORY;
282     }else
283         return E_FAIL;
284
285     return S_OK;
286 }
287
288 static HRESULT WINAPI HTMLFormElement_get_elements(IHTMLFormElement *iface, IDispatch **p)
289 {
290     HTMLFormElement *This = HTMLFORM_THIS(iface);
291     FIXME("(%p)->(%p)\n", This, p);
292     return E_NOTIMPL;
293 }
294
295 static HRESULT WINAPI HTMLFormElement_put_target(IHTMLFormElement *iface, BSTR v)
296 {
297     HTMLFormElement *This = HTMLFORM_THIS(iface);
298     FIXME("(%p)->(%s)\n", This, wine_dbgstr_w(v));
299     return E_NOTIMPL;
300 }
301
302 static HRESULT WINAPI HTMLFormElement_get_target(IHTMLFormElement *iface, BSTR *p)
303 {
304     HTMLFormElement *This = HTMLFORM_THIS(iface);
305     FIXME("(%p)->(%p)\n", This, p);
306     return E_NOTIMPL;
307 }
308
309 static HRESULT WINAPI HTMLFormElement_put_name(IHTMLFormElement *iface, BSTR v)
310 {
311     HTMLFormElement *This = HTMLFORM_THIS(iface);
312     nsAString name_str;
313     nsresult nsres;
314
315     TRACE("(%p)->(%s)\n", This, wine_dbgstr_w(v));
316
317     nsAString_InitDepend(&name_str, v);
318     nsres = nsIDOMHTMLFormElement_SetName(This->nsform, &name_str);
319     nsAString_Finish(&name_str);
320     if(NS_FAILED(nsres))
321         return E_FAIL;
322
323     return S_OK;
324 }
325
326 static HRESULT WINAPI HTMLFormElement_get_name(IHTMLFormElement *iface, BSTR *p)
327 {
328     HTMLFormElement *This = HTMLFORM_THIS(iface);
329     nsAString name_str;
330     nsresult nsres;
331
332     TRACE("(%p)->(%p)\n", This, p);
333
334     nsAString_Init(&name_str, NULL);
335     nsres = nsIDOMHTMLFormElement_GetName(This->nsform, &name_str);
336     if(NS_SUCCEEDED(nsres)) {
337         const PRUnichar *name;
338         nsAString_GetData(&name_str, &name);
339
340         if(*name) {
341             *p = SysAllocString(name);
342             if(!*p)
343                 return E_OUTOFMEMORY;
344         }else
345             *p = NULL;
346     }else
347         return E_FAIL;
348
349     return S_OK;
350 }
351
352 static HRESULT WINAPI HTMLFormElement_put_onsubmit(IHTMLFormElement *iface, VARIANT v)
353 {
354     HTMLFormElement *This = HTMLFORM_THIS(iface);
355     FIXME("(%p)->(v)\n", This);
356     return E_NOTIMPL;
357 }
358
359 static HRESULT WINAPI HTMLFormElement_get_onsubmit(IHTMLFormElement *iface, VARIANT *p)
360 {
361     HTMLFormElement *This = HTMLFORM_THIS(iface);
362     FIXME("(%p)->(%p)\n", This, p);
363     return E_NOTIMPL;
364 }
365
366 static HRESULT WINAPI HTMLFormElement_put_onreset(IHTMLFormElement *iface, VARIANT v)
367 {
368     HTMLFormElement *This = HTMLFORM_THIS(iface);
369     FIXME("(%p)->(v)\n", This);
370     return E_NOTIMPL;
371 }
372
373 static HRESULT WINAPI HTMLFormElement_get_onreset(IHTMLFormElement *iface, VARIANT *p)
374 {
375     HTMLFormElement *This = HTMLFORM_THIS(iface);
376     FIXME("(%p)->(%p)\n", This, p);
377     return E_NOTIMPL;
378 }
379
380 static HRESULT WINAPI HTMLFormElement_submit(IHTMLFormElement *iface)
381 {
382     HTMLFormElement *This = HTMLFORM_THIS(iface);
383     FIXME("(%p)->()\n", This);
384     return E_NOTIMPL;
385 }
386
387 static HRESULT WINAPI HTMLFormElement_reset(IHTMLFormElement *iface)
388 {
389     HTMLFormElement *This = HTMLFORM_THIS(iface);
390     FIXME("(%p)->()\n", This);
391     return E_NOTIMPL;
392 }
393
394 static HRESULT WINAPI HTMLFormElement_put_length(IHTMLFormElement *iface, LONG v)
395 {
396     HTMLFormElement *This = HTMLFORM_THIS(iface);
397     FIXME("(%p)->(%d)\n", This, v);
398     return E_NOTIMPL;
399 }
400
401 static HRESULT WINAPI HTMLFormElement_get_length(IHTMLFormElement *iface, LONG *p)
402 {
403     HTMLFormElement *This = HTMLFORM_THIS(iface);
404     PRInt32 length;
405     nsresult nsres;
406
407     TRACE("(%p)->(%p)\n", This, p);
408
409     nsres = nsIDOMHTMLFormElement_GetLength(This->nsform, &length);
410     if(NS_FAILED(nsres)) {
411         ERR("GetLength failed: %08x\n", nsres);
412         return E_FAIL;
413     }
414
415     *p = length;
416     return S_OK;
417 }
418
419 static HRESULT WINAPI HTMLFormElement__newEnum(IHTMLFormElement *iface, IUnknown **p)
420 {
421     HTMLFormElement *This = HTMLFORM_THIS(iface);
422     FIXME("(%p)->(%p)\n", This, p);
423     return E_NOTIMPL;
424 }
425
426 static HRESULT WINAPI HTMLFormElement_item(IHTMLFormElement *iface, VARIANT name,
427         VARIANT index, IDispatch **pdisp)
428 {
429     HTMLFormElement *This = HTMLFORM_THIS(iface);
430
431     TRACE("(%p)->(%s %s %p)\n", This, debugstr_variant(&name), debugstr_variant(&index), pdisp);
432
433     if(!pdisp)
434         return E_INVALIDARG;
435     *pdisp = NULL;
436
437     if(V_VT(&name) == VT_I4) {
438         if(V_I4(&name) < 0)
439             return E_INVALIDARG;
440         return htmlform_item(This, V_I4(&name), pdisp);
441     }
442
443     FIXME("Unsupported args\n");
444     return E_NOTIMPL;
445 }
446
447 static HRESULT WINAPI HTMLFormElement_tags(IHTMLFormElement *iface, VARIANT tagName,
448         IDispatch **pdisp)
449 {
450     HTMLFormElement *This = HTMLFORM_THIS(iface);
451     FIXME("(%p)->(v %p)\n", This, pdisp);
452     return E_NOTIMPL;
453 }
454
455 #undef HTMLFORM_THIS
456
457 static const IHTMLFormElementVtbl HTMLFormElementVtbl = {
458     HTMLFormElement_QueryInterface,
459     HTMLFormElement_AddRef,
460     HTMLFormElement_Release,
461     HTMLFormElement_GetTypeInfoCount,
462     HTMLFormElement_GetTypeInfo,
463     HTMLFormElement_GetIDsOfNames,
464     HTMLFormElement_Invoke,
465     HTMLFormElement_put_action,
466     HTMLFormElement_get_action,
467     HTMLFormElement_put_dir,
468     HTMLFormElement_get_dir,
469     HTMLFormElement_put_encoding,
470     HTMLFormElement_get_encoding,
471     HTMLFormElement_put_method,
472     HTMLFormElement_get_method,
473     HTMLFormElement_get_elements,
474     HTMLFormElement_put_target,
475     HTMLFormElement_get_target,
476     HTMLFormElement_put_name,
477     HTMLFormElement_get_name,
478     HTMLFormElement_put_onsubmit,
479     HTMLFormElement_get_onsubmit,
480     HTMLFormElement_put_onreset,
481     HTMLFormElement_get_onreset,
482     HTMLFormElement_submit,
483     HTMLFormElement_reset,
484     HTMLFormElement_put_length,
485     HTMLFormElement_get_length,
486     HTMLFormElement__newEnum,
487     HTMLFormElement_item,
488     HTMLFormElement_tags
489 };
490
491 #define HTMLFORM_NODE_THIS(iface) DEFINE_THIS2(HTMLFormElement, element.node, iface)
492
493 static HRESULT HTMLFormElement_QI(HTMLDOMNode *iface, REFIID riid, void **ppv)
494 {
495     HTMLFormElement *This = HTMLFORM_NODE_THIS(iface);
496
497     *ppv = NULL;
498
499     if(IsEqualGUID(&IID_IUnknown, riid)) {
500         TRACE("(%p)->(IID_IUnknown %p)\n", This, ppv);
501         *ppv = HTMLFORM(This);
502     }else if(IsEqualGUID(&IID_IDispatch, riid)) {
503         TRACE("(%p)->(IID_IDispatch %p)\n", This, ppv);
504         *ppv = HTMLFORM(This);
505     }else if(IsEqualGUID(&IID_IHTMLFormElement, riid)) {
506         TRACE("(%p)->(IID_IHTMLFormElement %p)\n", This, ppv);
507         *ppv = HTMLFORM(This);
508     }
509
510     if(*ppv) {
511         IUnknown_AddRef((IUnknown*)*ppv);
512         return S_OK;
513     }
514
515     return HTMLElement_QI(&This->element.node, riid, ppv);
516 }
517
518 static void HTMLFormElement_destructor(HTMLDOMNode *iface)
519 {
520     HTMLFormElement *This = HTMLFORM_NODE_THIS(iface);
521
522     if(This->nsform)
523         nsIDOMHTMLFormElement_Release(This->nsform);
524
525     HTMLElement_destructor(&This->element.node);
526 }
527
528 static HRESULT HTMLFormElement_get_dispid(HTMLDOMNode *iface,
529         BSTR name, DWORD grfdex, DISPID *pid)
530 {
531     HTMLFormElement *This = HTMLFORM_NODE_THIS(iface);
532     nsIDOMHTMLCollection *elements;
533     nsAString nsname, nsstr;
534     PRUint32 len, i;
535     nsresult nsres;
536     HRESULT hres = DISP_E_UNKNOWNNAME;
537
538     static const PRUnichar nameW[] = {'n','a','m','e',0};
539
540     TRACE("(%p)->(%s %x %p)\n", This, wine_dbgstr_w(name), grfdex, pid);
541
542     nsres = nsIDOMHTMLFormElement_GetElements(This->nsform, &elements);
543     if(NS_FAILED(nsres)) {
544         FIXME("GetElements failed: 0x%08x\n", nsres);
545         return E_FAIL;
546     }
547
548     nsres = nsIDOMHTMLCollection_GetLength(elements, &len);
549     if(NS_FAILED(nsres)) {
550         FIXME("GetLength failed: 0x%08x\n", nsres);
551         nsIDOMHTMLCollection_Release(elements);
552         return E_FAIL;
553     }
554
555     nsAString_InitDepend(&nsname, nameW);
556     nsAString_Init(&nsstr, NULL);
557     for(i = 0; i < len; ++i) {
558         nsIDOMNode *nsitem;
559         nsIDOMHTMLElement *nshtml_elem;
560         const PRUnichar *str;
561
562         nsres = nsIDOMHTMLCollection_Item(elements, i, &nsitem);
563         if(NS_FAILED(nsres)) {
564             FIXME("Item failed: 0x%08x\n", nsres);
565             hres = E_FAIL;
566             break;
567         }
568
569         nsres = nsIDOMNode_QueryInterface(nsitem, &IID_nsIDOMHTMLElement, (void**)&nshtml_elem);
570         nsIDOMNode_Release(nsitem);
571         if(NS_FAILED(nsres)) {
572             FIXME("Failed to get nsIDOMHTMLNode interface: 0x%08x\n", nsres);
573             hres = E_FAIL;
574             break;
575         }
576
577         /* compare by id attr */
578         nsres = nsIDOMHTMLElement_GetId(nshtml_elem, &nsstr);
579         if(NS_FAILED(nsres)) {
580             FIXME("GetId failed: 0x%08x\n", nsres);
581             nsIDOMHTMLElement_Release(nshtml_elem);
582             hres = E_FAIL;
583             break;
584         }
585         nsAString_GetData(&nsstr, &str);
586         if(!strcmpiW(str, name)) {
587             nsIDOMHTMLElement_Release(nshtml_elem);
588             /* FIXME: using index for dispid */
589             *pid = MSHTML_DISPID_CUSTOM_MIN + i;
590             hres = S_OK;
591             break;
592         }
593
594         /* compare by name attr */
595         nsres = nsIDOMHTMLElement_GetAttribute(nshtml_elem, &nsname, &nsstr);
596         nsIDOMHTMLElement_Release(nshtml_elem);
597         nsAString_GetData(&nsstr, &str);
598         if(!strcmpiW(str, name)) {
599             /* FIXME: using index for dispid */
600             *pid = MSHTML_DISPID_CUSTOM_MIN + i;
601             hres = S_OK;
602             break;
603         }
604     }
605     nsAString_Finish(&nsname);
606     nsAString_Finish(&nsstr);
607
608     nsIDOMHTMLCollection_Release(elements);
609
610     return hres;
611 }
612
613 static HRESULT HTMLFormElement_invoke(HTMLDOMNode *iface,
614         DISPID id, LCID lcid, WORD flags, DISPPARAMS *params, VARIANT *res,
615         EXCEPINFO *ei, IServiceProvider *caller)
616 {
617     HTMLFormElement *This = HTMLFORM_NODE_THIS(iface);
618     IDispatch *ret;
619     HRESULT hres;
620
621     TRACE("(%p)->(%x %x %x %p %p %p %p)\n", This, id, lcid, flags, params, res, ei, caller);
622
623     hres = htmlform_item(This, id - MSHTML_DISPID_CUSTOM_MIN, &ret);
624     if(FAILED(hres))
625         return hres;
626
627     if(ret) {
628         V_VT(res) = VT_DISPATCH;
629         V_DISPATCH(res) = ret;
630     }else {
631         V_VT(res) = VT_NULL;
632     }
633     return S_OK;
634 }
635
636 #undef HTMLFORM_NODE_THIS
637
638 static const NodeImplVtbl HTMLFormElementImplVtbl = {
639     HTMLFormElement_QI,
640     HTMLFormElement_destructor,
641     HTMLElement_clone,
642     NULL,
643     NULL,
644     NULL,
645     NULL,
646     NULL,
647     NULL,
648     HTMLFormElement_get_dispid,
649     HTMLFormElement_invoke
650 };
651
652 static const tid_t HTMLFormElement_iface_tids[] = {
653     HTMLELEMENT_TIDS,
654     IHTMLFormElement_tid,
655     0
656 };
657
658 static dispex_static_data_t HTMLFormElement_dispex = {
659     NULL,
660     DispHTMLFormElement_tid,
661     NULL,
662     HTMLFormElement_iface_tids
663 };
664
665 HTMLElement *HTMLFormElement_Create(HTMLDocumentNode *doc, nsIDOMHTMLElement *nselem)
666 {
667     HTMLFormElement *ret = heap_alloc_zero(sizeof(HTMLFormElement));
668     nsresult nsres;
669
670     ret->lpHTMLFormElementVtbl = &HTMLFormElementVtbl;
671     ret->element.node.vtbl = &HTMLFormElementImplVtbl;
672
673     HTMLElement_Init(&ret->element, doc, nselem, &HTMLFormElement_dispex);
674
675     nsres = nsIDOMHTMLElement_QueryInterface(nselem, &IID_nsIDOMHTMLFormElement, (void**)&ret->nsform);
676     if(NS_FAILED(nsres))
677         ERR("Could not get nsIDOMHTMLFormElement interface: %08x\n", nsres);
678
679     return &ret->element;
680 }