wbemprox: Support overriding the CIM to VARIANT type mapping for method parameters.
[wine] / dlls / mshtml / tests / dom.c
1 /*
2  * Copyright 2007-2011 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 #define COBJMACROS
20 #define CONST_VTABLE
21
22 #include <wine/test.h>
23 #include <stdarg.h>
24 #include <stdio.h>
25
26 #include "windef.h"
27 #include "winbase.h"
28 #include "ole2.h"
29 #include "mshtml.h"
30 #include "mshtmcid.h"
31 #include "mshtmhst.h"
32 #include "docobj.h"
33 #include "dispex.h"
34 #include "mshtml_test.h"
35 #include "objsafe.h"
36 #include "htiface.h"
37 #include "tlogstg.h"
38
39 static INT (WINAPI *pLCIDToLocaleName)(LCID,LPWSTR,INT,DWORD);
40 static LANGID (WINAPI *pGetUserDefaultUILanguage)(void);
41
42 static const char doc_blank[] = "<html></html>";
43 static const char doc_str1[] = "<html><body>test</body></html>";
44 static const char range_test_str[] =
45     "<html><body>test \na<font size=\"2\">bc\t123<br /> it's\r\n  \t</font>text<br /></body></html>";
46 static const char range_test2_str[] =
47     "<html><body>abc<hr />123<br /><hr />def</body></html>";
48 static const char elem_test_str[] =
49     "<html><head><title>test</title><style id=\"styleid\">.body { margin-right: 0px; }</style>"
50     "<meta id=\"metaid\" name=\"meta name\" http-equiv=\"Content-Type\" content=\"text/html; charset=utf-8\">"
51     "<body onload=\"Testing()\">text test<!-- a comment -->"
52     "<a id=\"a\" href=\"http://test\" name=\"x\">link</a>"
53     "<input id=\"in\" class=\"testclass\" tabIndex=\"2\" title=\"test title\" />"
54     "<select id=\"s\"><option id=\"x\" value=\"val1\">opt1</option><option id=\"y\">opt2</option></select>"
55     "<textarea id=\"X\">text text</textarea>"
56     "<table id=\"tbl\"><tbody><tr></tr><tr id=\"row2\"><td>td1 text</td><td>td2 text</td></tr></tbody></table>"
57     "<script id=\"sc\" type=\"text/javascript\"><!--\nfunction Testing() {}\n// -->\n</script>"
58     "<test /><object id=\"objid\" vspace=100></object><embed />"
59     "<img id=\"imgid\" name=\"WineImg\"/>"
60     "<iframe src=\"about:blank\" id=\"ifr\"></iframe>"
61     "<form id=\"frm\"></form>"
62     "<div id=\"attr\" attr1=\"attr1\" attr2 attr3=\"attr3\"></div>"
63     "</body></html>";
64 static const char elem_test2_str[] =
65     "<html><head><title>test</title><style>.body { margin-right: 0px; }</style>"
66     "<body><div id=\"divid\" emptyattr=\"\" onclick=\"parseInt();\"></div></body>"
67     "</html>";
68
69 static const char indent_test_str[] =
70     "<html><head><title>test</title></head><body>abc<br /><a href=\"about:blank\">123</a></body></html>";
71 static const char cond_comment_str[] =
72     "<html><head><title>test</title></head><body>"
73     "<!--[if gte IE 4]> <br> <![endif]-->"
74     "</body></html>";
75 static const char frameset_str[] =
76     "<html><head><title>frameset test</title></head><frameset rows=\"25, 25, *\">"
77     "<frame src=\"about:blank\" name=\"nm1\" id=\"fr1\"><frame src=\"about:blank\" name=\"nm2\" id=\"fr2\">"
78     "<frame src=\"about:blank\" id=\"fr3\">"
79     "</frameset></html>";
80 static const char emptydiv_str[] =
81     "<html><head><title>emptydiv test</title></head>"
82     "<body><div id=\"divid\"></div></body></html>";
83 static const char noscript_str[] =
84     "<html><head><title>noscript test</title><noscript><style>.body { margin-right: 0px; }</style></noscript></head>"
85     "<body><noscript><div>test</div></noscript></body></html>";
86
87 static WCHAR characterW[] = {'c','h','a','r','a','c','t','e','r',0};
88 static WCHAR texteditW[] = {'t','e','x','t','e','d','i','t',0};
89 static WCHAR wordW[] = {'w','o','r','d',0};
90
91 typedef enum {
92     ET_NONE,
93     ET_HTML,
94     ET_HEAD,
95     ET_TITLE,
96     ET_BODY,
97     ET_A,
98     ET_INPUT,
99     ET_SELECT,
100     ET_TEXTAREA,
101     ET_OPTION,
102     ET_STYLE,
103     ET_BLOCKQUOTE,
104     ET_P,
105     ET_BR,
106     ET_TABLE,
107     ET_TBODY,
108     ET_SCRIPT,
109     ET_TEST,
110     ET_TESTG,
111     ET_COMMENT,
112     ET_IMG,
113     ET_TR,
114     ET_TD,
115     ET_IFRAME,
116     ET_FORM,
117     ET_FRAME,
118     ET_OBJECT,
119     ET_EMBED,
120     ET_DIV,
121     ET_META,
122     ET_NOSCRIPT
123 } elem_type_t;
124
125 static const IID * const none_iids[] = {
126     &IID_IUnknown,
127     NULL
128 };
129
130 static const IID * const doc_node_iids[] = {
131     &IID_IHTMLDOMNode,
132     &IID_IHTMLDOMNode2,
133     &IID_IHTMLDocument,
134     &IID_IHTMLDocument2,
135     &IID_IHTMLDocument3,
136     &IID_IHTMLDocument4,
137     &IID_IHTMLDocument5,
138     &IID_IDispatchEx,
139     &IID_IConnectionPointContainer,
140     &IID_IInternetHostSecurityManager,
141     &IID_IOleContainer,
142     &IID_IObjectSafety,
143     &IID_IProvideClassInfo,
144     NULL
145 };
146
147 static const IID * const doc_obj_iids[] = {
148     &IID_IHTMLDocument,
149     &IID_IHTMLDocument2,
150     &IID_IHTMLDocument3,
151     &IID_IHTMLDocument4,
152     &IID_IHTMLDocument5,
153     &IID_IDispatchEx,
154     &IID_IConnectionPointContainer,
155     &IID_ICustomDoc,
156     &IID_IOleContainer,
157     &IID_IObjectSafety,
158     &IID_IProvideClassInfo,
159     &IID_ITargetContainer,
160     NULL
161 };
162
163 #define ELEM_IFACES \
164     &IID_IHTMLDOMNode,  \
165     &IID_IHTMLDOMNode2, \
166     &IID_IHTMLElement,  \
167     &IID_IHTMLElement2, \
168     &IID_IHTMLElement3, \
169     &IID_IHTMLElement4, \
170     &IID_IDispatchEx
171
172 static const IID * const elem_iids[] = {
173     ELEM_IFACES,
174     &IID_IConnectionPointContainer,
175     NULL
176 };
177
178 static const IID * const body_iids[] = {
179     ELEM_IFACES,
180     &IID_IHTMLTextContainer,
181     &IID_IHTMLBodyElement,
182     &IID_IConnectionPointContainer,
183     NULL
184 };
185
186 static const IID * const anchor_iids[] = {
187     ELEM_IFACES,
188     &IID_IHTMLAnchorElement,
189     &IID_IConnectionPointContainer,
190     NULL
191 };
192
193 static const IID * const input_iids[] = {
194     ELEM_IFACES,
195     &IID_IHTMLInputElement,
196     &IID_IHTMLInputTextElement,
197     &IID_IConnectionPointContainer,
198     NULL
199 };
200
201 static const IID * const select_iids[] = {
202     ELEM_IFACES,
203     &IID_IHTMLSelectElement,
204     &IID_IConnectionPointContainer,
205     NULL
206 };
207
208 static const IID * const textarea_iids[] = {
209     ELEM_IFACES,
210     &IID_IHTMLTextAreaElement,
211     &IID_IConnectionPointContainer,
212     NULL
213 };
214
215 static const IID * const option_iids[] = {
216     ELEM_IFACES,
217     &IID_IHTMLOptionElement,
218     &IID_IConnectionPointContainer,
219     NULL
220 };
221
222 static const IID * const table_iids[] = {
223     ELEM_IFACES,
224     &IID_IHTMLTable,
225     &IID_IHTMLTable2,
226     &IID_IHTMLTable3,
227     &IID_IConnectionPointContainer,
228     NULL
229 };
230
231 static const IID * const script_iids[] = {
232     ELEM_IFACES,
233     &IID_IHTMLScriptElement,
234     &IID_IConnectionPointContainer,
235     NULL
236 };
237
238 static const IID * const text_iids[] = {
239     &IID_IHTMLDOMNode,
240     &IID_IHTMLDOMNode2,
241     &IID_IHTMLDOMTextNode,
242     NULL
243 };
244
245 static const IID * const attr_iids[] = {
246     &IID_IHTMLDOMAttribute,
247     &IID_IDispatchEx,
248     NULL
249 };
250
251 static const IID * const location_iids[] = {
252     &IID_IDispatch,
253     &IID_IHTMLLocation,
254     NULL
255 };
256
257 static const IID * const window_iids[] = {
258     &IID_IDispatch,
259     &IID_IHTMLWindow2,
260     &IID_IHTMLWindow3,
261     &IID_IDispatchEx,
262     &IID_IServiceProvider,
263     NULL
264 };
265
266 static const IID * const comment_iids[] = {
267     ELEM_IFACES,
268     &IID_IHTMLCommentElement,
269     &IID_IConnectionPointContainer,
270     NULL
271 };
272
273 static const IID * const img_iids[] = {
274     ELEM_IFACES,
275     &IID_IHTMLImgElement,
276     &IID_IConnectionPointContainer,
277     NULL
278 };
279
280 static const IID * const tr_iids[] = {
281     ELEM_IFACES,
282     &IID_IHTMLTableRow,
283     &IID_IConnectionPointContainer,
284     NULL
285 };
286
287 static const IID * const td_iids[] = {
288     ELEM_IFACES,
289     &IID_IHTMLTableCell,
290     &IID_IConnectionPointContainer,
291     NULL
292 };
293
294 static const IID * const frame_iids[] = {
295     ELEM_IFACES,
296     &IID_IHTMLFrameBase,
297     &IID_IHTMLFrameBase2,
298     &IID_IConnectionPointContainer,
299     NULL
300 };
301
302 static const IID * const head_iids[] = {
303     ELEM_IFACES,
304     &IID_IHTMLHeadElement,
305     &IID_IConnectionPointContainer,
306     NULL
307 };
308
309 static const IID * const title_iids[] = {
310     ELEM_IFACES,
311     &IID_IHTMLTitleElement,
312     &IID_IConnectionPointContainer,
313     NULL
314 };
315
316 static const IID * const meta_iids[] = {
317     ELEM_IFACES,
318     &IID_IHTMLMetaElement,
319     &IID_IConnectionPointContainer,
320     NULL
321 };
322
323 static const IID * const object_iids[] = {
324     ELEM_IFACES,
325     &IID_IHTMLObjectElement,
326     &IID_IHTMLObjectElement2,
327     /* FIXME: No IConnectionPointContainer */
328     NULL
329 };
330
331 static const IID * const embed_iids[] = {
332     ELEM_IFACES,
333     &IID_IHTMLEmbedElement,
334     /* FIXME: No IConnectionPointContainer */
335     NULL
336 };
337
338 static const IID * const iframe_iids[] = {
339     ELEM_IFACES,
340     &IID_IHTMLFrameBase,
341     &IID_IHTMLFrameBase2,
342     &IID_IHTMLIFrameElement,
343     &IID_IHTMLIFrameElement2,
344     &IID_IConnectionPointContainer,
345     NULL
346 };
347
348 static const IID * const form_iids[] = {
349     ELEM_IFACES,
350     &IID_IHTMLFormElement,
351     &IID_IConnectionPointContainer,
352     NULL
353 };
354
355 static const IID * const styleelem_iids[] = {
356     ELEM_IFACES,
357     &IID_IHTMLStyleElement,
358     &IID_IConnectionPointContainer,
359     NULL
360 };
361
362 static const IID * const generic_iids[] = {
363     ELEM_IFACES,
364     &IID_IHTMLGenericElement,
365     &IID_IConnectionPointContainer,
366     NULL
367 };
368
369 static const IID * const style_iids[] = {
370     &IID_IUnknown,
371     &IID_IDispatch,
372     &IID_IDispatchEx,
373     &IID_IHTMLStyle,
374     &IID_IHTMLStyle2,
375     &IID_IHTMLStyle3,
376     &IID_IHTMLStyle4,
377     NULL
378 };
379
380 static const IID * const cstyle_iids[] = {
381     &IID_IUnknown,
382     &IID_IDispatch,
383     &IID_IDispatchEx,
384     &IID_IHTMLCurrentStyle,
385     &IID_IHTMLCurrentStyle2,
386     &IID_IHTMLCurrentStyle3,
387     NULL
388 };
389
390 static const IID * const img_factory_iids[] = {
391     &IID_IUnknown,
392     &IID_IDispatch,
393     &IID_IDispatchEx,
394     &IID_IHTMLImageElementFactory,
395     NULL
396 };
397
398 typedef struct {
399     const char *tag;
400     REFIID *iids;
401     const IID *dispiid;
402 } elem_type_info_t;
403
404 static const elem_type_info_t elem_type_infos[] = {
405     {"",          none_iids,        NULL},
406     {"HTML",      elem_iids,        NULL},
407     {"HEAD",      head_iids,        &DIID_DispHTMLHeadElement},
408     {"TITLE",     title_iids,       &DIID_DispHTMLTitleElement},
409     {"BODY",      body_iids,        &DIID_DispHTMLBody},
410     {"A",         anchor_iids,      &DIID_DispHTMLAnchorElement},
411     {"INPUT",     input_iids,       &DIID_DispHTMLInputElement},
412     {"SELECT",    select_iids,      &DIID_DispHTMLSelectElement},
413     {"TEXTAREA",  textarea_iids,    &DIID_DispHTMLTextAreaElement},
414     {"OPTION",    option_iids,      &DIID_DispHTMLOptionElement},
415     {"STYLE",     styleelem_iids,   &DIID_DispHTMLStyleElement},
416     {"BLOCKQUOTE",elem_iids,        NULL},
417     {"P",         elem_iids,        NULL},
418     {"BR",        elem_iids,        NULL},
419     {"TABLE",     table_iids,       &DIID_DispHTMLTable},
420     {"TBODY",     elem_iids,        NULL},
421     {"SCRIPT",    script_iids,      &DIID_DispHTMLScriptElement},
422     {"TEST",      elem_iids,        &DIID_DispHTMLUnknownElement},
423     {"TEST",      generic_iids,     &DIID_DispHTMLGenericElement},
424     {"!",         comment_iids,     &DIID_DispHTMLCommentElement},
425     {"IMG",       img_iids,         &DIID_DispHTMLImg},
426     {"TR",        tr_iids,          &DIID_DispHTMLTableRow},
427     {"TD",        td_iids,          &DIID_DispHTMLTableCell},
428     {"IFRAME",    iframe_iids,      &DIID_DispHTMLIFrame},
429     {"FORM",      form_iids,        &DIID_DispHTMLFormElement},
430     {"FRAME",     frame_iids,       &DIID_DispHTMLFrameElement},
431     {"OBJECT",    object_iids,      &DIID_DispHTMLObjectElement},
432     {"EMBED",     embed_iids,       &DIID_DispHTMLEmbed},
433     {"DIV",       elem_iids,        NULL},
434     {"META",      meta_iids,        &DIID_DispHTMLMetaElement},
435     {"NOSCRIPT",  elem_iids,        NULL /*&DIID_DispHTMLNoShowElement*/}
436 };
437
438 static const char *dbgstr_guid(REFIID riid)
439 {
440     static char buf[50];
441
442     sprintf(buf, "{%08X-%04X-%04X-%02X%02X-%02X%02X%02X%02X%02X%02X}",
443             riid->Data1, riid->Data2, riid->Data3, riid->Data4[0],
444             riid->Data4[1], riid->Data4[2], riid->Data4[3], riid->Data4[4],
445             riid->Data4[5], riid->Data4[6], riid->Data4[7]);
446
447     return buf;
448 }
449
450 static int strcmp_wa(LPCWSTR strw, const char *stra)
451 {
452     CHAR buf[512];
453     WideCharToMultiByte(CP_ACP, 0, strw, -1, buf, sizeof(buf), NULL, NULL);
454     return lstrcmpA(stra, buf);
455 }
456
457 static BSTR a2bstr(const char *str)
458 {
459     BSTR ret;
460     int len;
461
462     if(!str)
463         return NULL;
464
465     len = MultiByteToWideChar(CP_ACP, 0, str, -1, NULL, 0);
466     ret = SysAllocStringLen(NULL, len);
467     MultiByteToWideChar(CP_ACP, 0, str, -1, ret, len);
468
469     return ret;
470 }
471
472 static BOOL iface_cmp(IUnknown *iface1, IUnknown *iface2)
473 {
474     IUnknown *unk1, *unk2;
475
476     if(iface1 == iface2)
477         return TRUE;
478
479     IUnknown_QueryInterface(iface1, &IID_IUnknown, (void**)&unk1);
480     IUnknown_Release(unk1);
481     IUnknown_QueryInterface(iface2, &IID_IUnknown, (void**)&unk2);
482     IUnknown_Release(unk2);
483
484     return unk1 == unk2;
485 }
486
487 static IHTMLDocument2 *create_document(void)
488 {
489     IHTMLDocument2 *doc;
490     IHTMLDocument5 *doc5;
491     HRESULT hres;
492
493     hres = CoCreateInstance(&CLSID_HTMLDocument, NULL, CLSCTX_INPROC_SERVER|CLSCTX_INPROC_HANDLER,
494             &IID_IHTMLDocument2, (void**)&doc);
495     ok(hres == S_OK, "CoCreateInstance failed: %08x\n", hres);
496     if(FAILED(hres))
497         return NULL;
498
499     hres = IHTMLDocument2_QueryInterface(doc, &IID_IHTMLDocument5, (void**)&doc5);
500     if(FAILED(hres)) {
501         win_skip("Could not get IHTMLDocument5, probably too old IE\n");
502         IHTMLDocument2_Release(doc);
503         return NULL;
504     }
505
506     IHTMLDocument5_Release(doc5);
507     return doc;
508 }
509
510 #define get_dispex_iface(u) _get_dispex_iface(__LINE__,u)
511 static IDispatchEx *_get_dispex_iface(unsigned line, IUnknown *unk)
512 {
513     IDispatchEx *dispex;
514     HRESULT hres;
515
516     hres = IUnknown_QueryInterface(unk, &IID_IDispatchEx, (void**)&dispex);
517     ok_(__FILE__,line) (hres == S_OK, "Could not get IDispatchEx: %08x\n", hres);
518     return dispex;
519 }
520
521 #define test_ifaces(i,ids) _test_ifaces(__LINE__,i,ids)
522 static void _test_ifaces(unsigned line, IUnknown *iface, REFIID *iids)
523 {
524     const IID * const *piid;
525     IUnknown *unk;
526     HRESULT hres;
527
528      for(piid = iids; *piid; piid++) {
529         hres = IUnknown_QueryInterface(iface, *piid, (void**)&unk);
530         ok_(__FILE__,line) (hres == S_OK, "Could not get %s interface: %08x\n", dbgstr_guid(*piid), hres);
531         if(SUCCEEDED(hres))
532             IUnknown_Release(unk);
533     }
534 }
535
536 #define test_no_iface(a,b) _test_no_iface(__LINE__,a,b)
537 static void _test_no_iface(unsigned line, IUnknown *iface, REFIID iid)
538 {
539     IUnknown *unk;
540     HRESULT hres;
541
542     unk = (void*)0xdeadbeef;
543     hres = IUnknown_QueryInterface(iface, iid, (void**)&unk);
544     ok_(__FILE__,line)(hres == E_NOINTERFACE, "hres = %08x, expected E_NOINTERFACE\n", hres);
545     ok_(__FILE__,line)(!unk, "unk = %p\n", unk);
546 }
547
548 #define test_get_dispid(u,id) _test_get_dispid(__LINE__,u,id)
549 static BOOL _test_get_dispid(unsigned line, IUnknown *unk, IID *iid)
550 {
551     IDispatchEx *dispex = _get_dispex_iface(line, unk);
552     ITypeInfo *typeinfo;
553     BOOL ret = FALSE;
554     UINT ticnt;
555     HRESULT hres;
556
557     ticnt = 0xdeadbeef;
558     hres = IDispatchEx_GetTypeInfoCount(dispex, &ticnt);
559     ok_(__FILE__,line) (hres == S_OK, "GetTypeInfoCount failed: %08x\n", hres);
560     ok_(__FILE__,line) (ticnt == 1, "ticnt=%u\n", ticnt);
561
562     hres = IDispatchEx_GetTypeInfo(dispex, 0, 0, &typeinfo);
563     ok_(__FILE__,line) (hres == S_OK, "GetTypeInfo failed: %08x\n", hres);
564
565     if(SUCCEEDED(hres)) {
566         TYPEATTR *type_attr;
567
568         hres = ITypeInfo_GetTypeAttr(typeinfo, &type_attr);
569         ok_(__FILE__,line) (hres == S_OK, "GetTypeAttr failed: %08x\n", hres);
570         if(hres == S_OK) {
571             *iid = type_attr->guid;
572             ret = TRUE;
573         }
574
575         ITypeInfo_ReleaseTypeAttr(typeinfo, type_attr);
576         ITypeInfo_Release(typeinfo);
577     }
578
579     IDispatchEx_Release(dispex);
580     return ret;
581 }
582
583 #define test_disp_value(u) _test_disp_value(__LINE__,u,v)
584 static void _test_disp_value(unsigned line, IUnknown *unk, const char *val)
585 {
586     IDispatchEx *dispex = _get_dispex_iface(line, unk);
587     DISPPARAMS dp  = {NULL,NULL,0,0};
588     EXCEPINFO ei;
589     VARIANT var;
590     HRESULT hres;
591
592     hres = IDispatchEx_InvokeEx(dispex, DISPID_VALUE, 0, DISPATCH_PROPERTYGET, &dp, &var, &ei, NULL);
593     IDispatchEx_Release(dispex);
594     ok_(__FILE__,line)(hres == S_OK, "InvokeEx(DISPID_VALUE) returned: %08x\n", hres);
595
596     ok_(__FILE__,line)(V_VT(&var) == VT_BSTR, "V_VT(value) = %d\n", V_VT(&var));
597     ok_(__FILE__,line)(!strcmp_wa(V_BSTR(&var), val), "value = %s, expected %s\n", wine_dbgstr_w(V_BSTR(&var)), val);
598     VariantClear(&var);
599 }
600
601 #define test_disp(u,id,v) _test_disp(__LINE__,u,id,v)
602 static void _test_disp(unsigned line, IUnknown *unk, const IID *diid, const char *val)
603 {
604     IID iid;
605
606     if(_test_get_dispid(line, unk, &iid))
607         ok_(__FILE__,line) (IsEqualGUID(&iid, diid), "unexpected guid %s\n", dbgstr_guid(&iid));
608
609     if(val)
610         _test_disp_value(line, unk, val);
611 }
612
613 #define test_disp2(u,id,id2,v) _test_disp2(__LINE__,u,id,id2,v)
614 static void _test_disp2(unsigned line, IUnknown *unk, const IID *diid, const IID *diid2, const char *val)
615 {
616     IID iid;
617
618     if(_test_get_dispid(line, unk, &iid))
619         ok_(__FILE__,line) (IsEqualGUID(&iid, diid) || broken(IsEqualGUID(&iid, diid2)),
620                 "unexpected guid %s\n", dbgstr_guid(&iid));
621
622     if(val)
623         _test_disp_value(line, unk, val);
624 }
625
626 #define test_class_info(u) _test_class_info(__LINE__,u)
627 static void _test_class_info(unsigned line, IUnknown *unk)
628 {
629     IProvideClassInfo *classinfo;
630     ITypeInfo *typeinfo;
631     TYPEATTR *type_attr;
632     HRESULT hres;
633
634     hres = IUnknown_QueryInterface(unk, &IID_IProvideClassInfo, (void**)&classinfo);
635     ok_(__FILE__,line)(hres == S_OK, "Could not get IProvideClassInfo interface: %08x\n", hres);
636     if(FAILED(hres))
637         return;
638
639     hres = IProvideClassInfo_GetClassInfo(classinfo, &typeinfo);
640     ok_(__FILE__,line)(hres == S_OK, "Could not get ITypeInfo interface: %08x\n", hres);
641     if(FAILED(hres))
642     {
643         IProvideClassInfo_Release(classinfo);
644         return;
645     }
646
647     hres = ITypeInfo_GetTypeAttr(typeinfo, &type_attr);
648     ok_(__FILE__,line)(hres == S_OK, "GetTypeAttr failed: %08x\n", hres);
649     if(SUCCEEDED(hres))
650     {
651         ok_(__FILE__,line)(IsEqualGUID(&type_attr->guid, &CLSID_HTMLDocument),
652                 "unexpected guid %s\n", dbgstr_guid(&type_attr->guid));
653         ok_(__FILE__,line)(type_attr->typekind == TKIND_COCLASS,
654                 "unexpected typekind %d\n", type_attr->typekind);
655         ITypeInfo_ReleaseTypeAttr(typeinfo, type_attr);
656     }
657
658     ITypeInfo_Release(typeinfo);
659     IProvideClassInfo_Release(classinfo);
660 }
661
662 #define set_dispex_value(a,b,c) _set_dispex_value(__LINE__,a,b,c)
663 static void _set_dispex_value(unsigned line, IUnknown *unk, const char *name, VARIANT *val)
664 {
665     IDispatchEx *dispex = _get_dispex_iface(line, unk);
666     DISPPARAMS dp = {val, NULL, 1, 0};
667     EXCEPINFO ei;
668     DISPID id;
669     BSTR str;
670     HRESULT hres;
671
672     str = a2bstr(name);
673     hres = IDispatchEx_GetDispID(dispex, str, fdexNameEnsure|fdexNameCaseInsensitive, &id);
674     SysFreeString(str);
675     ok_(__FILE__,line)(hres == S_OK, "GetDispID failed: %08x\n", hres);
676
677     memset(&ei, 0, sizeof(ei));
678     hres = IDispatchEx_InvokeEx(dispex, id, LOCALE_NEUTRAL, INVOKE_PROPERTYPUT, &dp, NULL, &ei, NULL);
679     ok_(__FILE__,line)(hres == S_OK, "InvokeEx failed: %08x\n", hres);
680
681 }
682
683 #define get_elem_iface(u) _get_elem_iface(__LINE__,u)
684 static IHTMLElement *_get_elem_iface(unsigned line, IUnknown *unk)
685 {
686     IHTMLElement *elem;
687     HRESULT hres;
688
689     hres = IUnknown_QueryInterface(unk, &IID_IHTMLElement, (void**)&elem);
690     ok_(__FILE__,line) (hres == S_OK, "Could not get IHTMLElement: %08x\n", hres);
691     return elem;
692 }
693
694 #define get_elem2_iface(u) _get_elem2_iface(__LINE__,u)
695 static IHTMLElement2 *_get_elem2_iface(unsigned line, IUnknown *unk)
696 {
697     IHTMLElement2 *elem;
698     HRESULT hres;
699
700     hres = IUnknown_QueryInterface(unk, &IID_IHTMLElement2, (void**)&elem);
701     ok_(__FILE__,line) (hres == S_OK, "Could not get IHTMLElement2: %08x\n", hres);
702     return elem;
703 }
704
705 #define get_elem3_iface(u) _get_elem3_iface(__LINE__,u)
706 static IHTMLElement3 *_get_elem3_iface(unsigned line, IUnknown *unk)
707 {
708     IHTMLElement3 *elem;
709     HRESULT hres;
710
711     hres = IUnknown_QueryInterface(unk, &IID_IHTMLElement3, (void**)&elem);
712     ok_(__FILE__,line) (hres == S_OK, "Could not get IHTMLElement3: %08x\n", hres);
713     return elem;
714 }
715
716 #define get_elem4_iface(u) _get_elem4_iface(__LINE__,u)
717 static IHTMLElement4 *_get_elem4_iface(unsigned line, IUnknown *unk)
718 {
719     IHTMLElement4 *elem;
720     HRESULT hres;
721
722     hres = IUnknown_QueryInterface(unk, &IID_IHTMLElement4, (void**)&elem);
723     ok_(__FILE__,line) (hres == S_OK, "Could not get IHTMLElement4: %08x\n", hres);
724     return elem;
725 }
726
727 #define get_node_iface(u) _get_node_iface(__LINE__,u)
728 static IHTMLDOMNode *_get_node_iface(unsigned line, IUnknown *unk)
729 {
730     IHTMLDOMNode *node;
731     HRESULT hres;
732
733     hres = IUnknown_QueryInterface(unk, &IID_IHTMLDOMNode, (void**)&node);
734     ok_(__FILE__,line) (hres == S_OK, "Could not get IHTMLDOMNode: %08x\n", hres);
735     return node;
736 }
737
738 #define get_node2_iface(u) _get_node2_iface(__LINE__,u)
739 static IHTMLDOMNode2 *_get_node2_iface(unsigned line, IUnknown *unk)
740 {
741     IHTMLDOMNode2 *node;
742     HRESULT hres;
743
744     hres = IUnknown_QueryInterface(unk, &IID_IHTMLDOMNode2, (void**)&node);
745     ok_(__FILE__,line) (hres == S_OK, "Could not get IHTMLDOMNode2: %08x\n", hres);
746     return node;
747 }
748
749 #define get_img_iface(u) _get_img_iface(__LINE__,u)
750 static IHTMLImgElement *_get_img_iface(unsigned line, IUnknown *unk)
751 {
752     IHTMLImgElement *img;
753     HRESULT hres;
754
755     hres = IUnknown_QueryInterface(unk, &IID_IHTMLImgElement, (void**)&img);
756     ok_(__FILE__,line) (hres == S_OK, "Could not get IHTMLImgElement: %08x\n", hres);
757     return img;
758 }
759
760 #define get_anchor_iface(u) _get_anchor_iface(__LINE__,u)
761 static IHTMLAnchorElement *_get_anchor_iface(unsigned line, IUnknown *unk)
762 {
763     IHTMLAnchorElement *anchor;
764     HRESULT hres;
765
766     hres = IUnknown_QueryInterface(unk, &IID_IHTMLAnchorElement, (void**)&anchor);
767     ok_(__FILE__,line) (hres == S_OK, "Could not get IHTMLAnchorElement: %08x\n", hres);
768     return anchor;
769 }
770
771 #define get_textarea_iface(u) _get_textarea_iface(__LINE__,u)
772 static IHTMLTextAreaElement *_get_textarea_iface(unsigned line, IUnknown *unk)
773 {
774     IHTMLTextAreaElement *textarea;
775     HRESULT hres;
776
777     hres = IUnknown_QueryInterface(unk, &IID_IHTMLTextAreaElement, (void**)&textarea);
778     ok_(__FILE__,line) (hres == S_OK, "Could not get IHTMLTextAreaElement: %08x\n", hres);
779     return textarea;
780 }
781
782 #define get_select_iface(u) _get_select_iface(__LINE__,u)
783 static IHTMLSelectElement *_get_select_iface(unsigned line, IUnknown *unk)
784 {
785     IHTMLSelectElement *select;
786     HRESULT hres;
787
788     hres = IUnknown_QueryInterface(unk, &IID_IHTMLSelectElement, (void**)&select);
789     ok_(__FILE__,line) (hres == S_OK, "Could not get IHTMLSelectElement: %08x\n", hres);
790     return select;
791 }
792
793 #define get_form_iface(u) _get_form_iface(__LINE__,u)
794 static IHTMLFormElement *_get_form_iface(unsigned line, IUnknown *unk)
795 {
796     IHTMLFormElement *form;
797     HRESULT hres;
798
799     hres = IUnknown_QueryInterface(unk, &IID_IHTMLFormElement, (void**)&form);
800     ok_(__FILE__,line) (hres == S_OK, "Could not get IHTMLFormElement: %08x\n", hres);
801     return form;
802 }
803
804 #define get_text_iface(u) _get_text_iface(__LINE__,u)
805 static IHTMLDOMTextNode *_get_text_iface(unsigned line, IUnknown *unk)
806 {
807     IHTMLDOMTextNode *text;
808     HRESULT hres;
809
810     hres = IUnknown_QueryInterface(unk, &IID_IHTMLDOMTextNode, (void**)&text);
811     ok_(__FILE__,line) (hres == S_OK, "Could not get IHTMLDOMTextNode: %08x\n", hres);
812     return text;
813 }
814
815 #define get_comment_iface(u) _get_comment_iface(__LINE__,u)
816 static IHTMLCommentElement *_get_comment_iface(unsigned line, IUnknown *unk)
817 {
818     IHTMLCommentElement *comment;
819     HRESULT hres;
820
821     hres = IUnknown_QueryInterface(unk, &IID_IHTMLCommentElement, (void**)&comment);
822     ok_(__FILE__,line) (hres == S_OK, "Could not get IHTMLCommentElement: %08x\n", hres);
823     return comment;
824 }
825
826 #define get_object_iface(u) _get_object_iface(__LINE__,u)
827 static IHTMLObjectElement *_get_object_iface(unsigned line, IUnknown *unk)
828 {
829     IHTMLObjectElement *obj;
830     HRESULT hres;
831
832     hres = IUnknown_QueryInterface(unk, &IID_IHTMLObjectElement, (void**)&obj);
833     ok_(__FILE__,line) (hres == S_OK, "Could not get IHTMLObjectElement: %08x\n", hres);
834     return obj;
835 }
836
837 #define get_style_iface(u) _get_style_iface(__LINE__,u)
838 static IHTMLStyleElement *_get_style_iface(unsigned line, IUnknown *unk)
839 {
840     IHTMLStyleElement *obj;
841     HRESULT hres;
842
843     hres = IUnknown_QueryInterface(unk, &IID_IHTMLStyleElement, (void**)&obj);
844     ok_(__FILE__,line) (hres == S_OK, "Could not get IHTMLStyleElement: %08x\n", hres);
845     return obj;
846 }
847
848 #define get_metaelem_iface(u) _get_metaelem_iface(__LINE__,u)
849 static IHTMLMetaElement *_get_metaelem_iface(unsigned line, IUnknown *unk)
850 {
851     IHTMLMetaElement *ret;
852     HRESULT hres;
853
854     hres = IUnknown_QueryInterface(unk, &IID_IHTMLMetaElement, (void**)&ret);
855     ok_(__FILE__,line) (hres == S_OK, "Could not get IHTMLMetaElement: %08x\n", hres);
856     return ret;
857 }
858
859 #define test_node_name(u,n) _test_node_name(__LINE__,u,n)
860 static void _test_node_name(unsigned line, IUnknown *unk, const char *exname)
861 {
862     IHTMLDOMNode *node = _get_node_iface(line, unk);
863     BSTR name;
864     HRESULT hres;
865
866     hres = IHTMLDOMNode_get_nodeName(node, &name);
867     IHTMLDOMNode_Release(node);
868     ok_(__FILE__, line) (hres == S_OK, "get_nodeName failed: %08x\n", hres);
869     ok_(__FILE__, line) (!strcmp_wa(name, exname), "got name: %s, expected %s\n", wine_dbgstr_w(name), exname);
870
871     SysFreeString(name);
872 }
873
874 #define get_owner_doc(u) _get_owner_doc(__LINE__,u)
875 static IHTMLDocument2 *_get_owner_doc(unsigned line, IUnknown *unk)
876 {
877     IHTMLDOMNode2 *node = _get_node2_iface(line, unk);
878     IDispatch *disp = (void*)0xdeadbeef;
879     IHTMLDocument2 *doc = NULL;
880     HRESULT hres;
881
882     hres = IHTMLDOMNode2_get_ownerDocument(node, &disp);
883     IHTMLDOMNode2_Release(node);
884     ok_(__FILE__,line)(hres == S_OK, "get_ownerDocument failed: %08x\n", hres);
885
886     if(disp) {
887         hres = IDispatch_QueryInterface(disp, &IID_IHTMLDocument2, (void**)&doc);
888         IDispatch_Release(disp);
889         ok_(__FILE__,line)(hres == S_OK, "Could not get IHTMLDocument2 iface: %08x\n", hres);
890     }
891
892     return doc;
893 }
894
895 #define get_doc_window(d) _get_doc_window(__LINE__,d)
896 static IHTMLWindow2 *_get_doc_window(unsigned line, IHTMLDocument2 *doc)
897 {
898     IHTMLWindow2 *window;
899     HRESULT hres;
900
901     window = NULL;
902     hres = IHTMLDocument2_get_parentWindow(doc, &window);
903     ok_(__FILE__,line)(hres == S_OK, "get_parentWindow failed: %08x\n", hres);
904     ok_(__FILE__,line)(window != NULL, "window == NULL\n");
905
906     return window;
907 }
908
909 #define clone_node(n,d) _clone_node(__LINE__,n,d)
910 static IHTMLDOMNode *_clone_node(unsigned line, IUnknown *unk, VARIANT_BOOL deep)
911 {
912     IHTMLDOMNode *node = _get_node_iface(line, unk);
913     IHTMLDOMNode *ret = NULL;
914     HRESULT hres;
915
916     hres = IHTMLDOMNode_cloneNode(node, deep, &ret);
917     IHTMLDOMNode_Release(node);
918     ok_(__FILE__,line)(hres == S_OK, "cloneNode failed: %08x\n", hres);
919     ok_(__FILE__,line)(ret != NULL, "ret == NULL\n");
920
921     return ret;
922
923 }
924
925 #define test_elem_tag(u,n) _test_elem_tag(__LINE__,u,n)
926 static void _test_elem_tag(unsigned line, IUnknown *unk, const char *extag)
927 {
928     IHTMLElement *elem = _get_elem_iface(line, unk);
929     BSTR tag;
930     HRESULT hres;
931
932     hres = IHTMLElement_get_tagName(elem, &tag);
933     IHTMLElement_Release(elem);
934     ok_(__FILE__, line) (hres == S_OK, "get_tagName failed: %08x\n", hres);
935     ok_(__FILE__, line) (!strcmp_wa(tag, extag), "got tag: %s, expected %s\n", wine_dbgstr_w(tag), extag);
936
937     SysFreeString(tag);
938 }
939
940 #define test_elem_type(ifc,t) _test_elem_type(__LINE__,ifc,t)
941 static void _test_elem_type(unsigned line, IUnknown *unk, elem_type_t type)
942 {
943     _test_elem_tag(line, unk, elem_type_infos[type].tag);
944     _test_ifaces(line, unk, elem_type_infos[type].iids);
945
946     if(elem_type_infos[type].dispiid && type != ET_A)
947         _test_disp(line, unk, elem_type_infos[type].dispiid, "[object]");
948 }
949
950 #define get_node_type(n) _get_node_type(__LINE__,n)
951 static LONG _get_node_type(unsigned line, IUnknown *unk)
952 {
953     IHTMLDOMNode *node = _get_node_iface(line, unk);
954     LONG type = -1;
955     HRESULT hres;
956
957     hres = IHTMLDOMNode_get_nodeType(node, &type);
958     ok(hres == S_OK, "get_nodeType failed: %08x\n", hres);
959
960     IHTMLDOMNode_Release(node);
961
962     return type;
963 }
964
965 #define get_child_nodes(u) _get_child_nodes(__LINE__,u)
966 static IHTMLDOMChildrenCollection *_get_child_nodes(unsigned line, IUnknown *unk)
967 {
968     IHTMLDOMNode *node = _get_node_iface(line, unk);
969     IHTMLDOMChildrenCollection *col = NULL;
970     IDispatch *disp;
971     HRESULT hres;
972
973     hres = IHTMLDOMNode_get_childNodes(node, &disp);
974     IHTMLDOMNode_Release(node);
975     ok_(__FILE__,line) (hres == S_OK, "get_childNodes failed: %08x\n", hres);
976     if(FAILED(hres))
977         return NULL;
978
979     hres = IDispatch_QueryInterface(disp, &IID_IHTMLDOMChildrenCollection, (void**)&col);
980     IDispatch_Release(disp);
981     ok_(__FILE__,line) (hres == S_OK, "Could not get IHTMLDOMChildrenCollection: %08x\n", hres);
982
983     return col;
984 }
985
986 #define get_child_item(c,i) _get_child_item(__LINE__,c,i)
987 static IHTMLDOMNode *_get_child_item(unsigned line, IHTMLDOMChildrenCollection *col, LONG idx)
988 {
989     IHTMLDOMNode *node = NULL;
990     IDispatch *disp;
991     HRESULT hres;
992
993     hres = IHTMLDOMChildrenCollection_item(col, idx, &disp);
994     ok(hres == S_OK, "item failed: %08x\n", hres);
995
996     node = _get_node_iface(line, (IUnknown*)disp);
997     IDispatch_Release(disp);
998
999     return node;
1000 }
1001
1002 #define test_elem_attr(e,n,v) _test_elem_attr(__LINE__,e,n,v)
1003 static void _test_elem_attr(unsigned line, IHTMLElement *elem, const char *name, const char *exval)
1004 {
1005     VARIANT value;
1006     BSTR tmp;
1007     HRESULT hres;
1008
1009     VariantInit(&value);
1010
1011     tmp = a2bstr(name);
1012     hres = IHTMLElement_getAttribute(elem, tmp, 0, &value);
1013     SysFreeString(tmp);
1014     ok_(__FILE__,line) (hres == S_OK, "getAttribute failed: %08x\n", hres);
1015
1016     if(exval) {
1017         ok_(__FILE__,line) (V_VT(&value) == VT_BSTR, "vt=%d\n", V_VT(&value));
1018         ok_(__FILE__,line) (!strcmp_wa(V_BSTR(&value), exval), "unexpected value %s\n", wine_dbgstr_w(V_BSTR(&value)));
1019     }else {
1020         ok_(__FILE__,line) (V_VT(&value) == VT_NULL, "vt=%d\n", V_VT(&value));
1021     }
1022
1023     VariantClear(&value);
1024 }
1025
1026 #define test_elem_offset(a,b) _test_elem_offset(__LINE__,a,b)
1027 static void _test_elem_offset(unsigned line, IUnknown *unk, const char *parent_tag)
1028 {
1029     IHTMLElement *elem = _get_elem_iface(line, unk);
1030     IHTMLElement *off_parent;
1031     LONG l;
1032     HRESULT hres;
1033
1034     hres = IHTMLElement_get_offsetTop(elem, &l);
1035     ok_(__FILE__,line) (hres == S_OK, "get_offsetTop failed: %08x\n", hres);
1036
1037     hres = IHTMLElement_get_offsetHeight(elem, &l);
1038     ok_(__FILE__,line) (hres == S_OK, "get_offsetHeight failed: %08x\n", hres);
1039
1040     hres = IHTMLElement_get_offsetWidth(elem, &l);
1041     ok_(__FILE__,line) (hres == S_OK, "get_offsetWidth failed: %08x\n", hres);
1042
1043     hres = IHTMLElement_get_offsetLeft(elem, &l);
1044     ok_(__FILE__,line) (hres == S_OK, "get_offsetLeft failed: %08x\n", hres);
1045
1046     hres = IHTMLElement_get_offsetParent(elem, &off_parent);
1047     ok_(__FILE__,line) (hres == S_OK, "get_offsetParent failed: %08x\n", hres);
1048
1049     _test_elem_tag(line, (IUnknown*)off_parent, parent_tag);
1050     IHTMLElement_Release(off_parent);
1051
1052     IHTMLElement_Release(elem);
1053 }
1054
1055 #define get_doc_node(d) _get_doc_node(__LINE__,d)
1056 static IHTMLDocument2 *_get_doc_node(unsigned line, IHTMLDocument2 *doc)
1057 {
1058     IHTMLWindow2 *window;
1059     IHTMLDocument2 *ret;
1060     HRESULT hres;
1061
1062     hres = IHTMLDocument2_get_parentWindow(doc, &window);
1063     ok_(__FILE__,line)(hres == S_OK, "get_parentWindow failed: %08x\n", hres);
1064
1065     hres = IHTMLWindow2_get_document(window, &ret);
1066     ok_(__FILE__,line)(hres == S_OK, "get_document failed: %08x\n", hres);
1067     ok_(__FILE__,line)(ret != NULL, "document = NULL\n");
1068
1069     return ret;
1070 }
1071
1072 #define test_window_name(d,e) _test_window_name(__LINE__,d,e)
1073 static void _test_window_name(unsigned line, IHTMLWindow2 *window, const char *exname)
1074 {
1075     BSTR name;
1076     HRESULT hres;
1077
1078     hres = IHTMLWindow2_get_name(window, &name);
1079     ok_(__FILE__,line)(hres == S_OK, "get_name failed: %08x\n", hres);
1080     if(exname)
1081         ok_(__FILE__,line)(!strcmp_wa(name, exname), "name = %s\n", wine_dbgstr_w(name));
1082     else
1083         ok_(__FILE__,line)(!name, "name = %s\n", wine_dbgstr_w(name));
1084     SysFreeString(name);
1085 }
1086
1087 #define set_window_name(w,n) _set_window_name(__LINE__,w,n)
1088 static void _set_window_name(unsigned line, IHTMLWindow2 *window, const char *name)
1089 {
1090     BSTR str;
1091     HRESULT hres;
1092
1093     str = a2bstr(name);
1094     hres = IHTMLWindow2_put_name(window, str);
1095     SysFreeString(str);
1096     ok_(__FILE__,line)(hres == S_OK, "put_name failed: %08x\n", hres);
1097
1098     _test_window_name(line, window, name);
1099 }
1100
1101 #define test_window_status(d) _test_window_status(__LINE__,d)
1102 static void _test_window_status(unsigned line, IHTMLWindow2 *window)
1103 {
1104     BSTR status;
1105     HRESULT hres;
1106
1107     status = (void*)0xdeadbeef;
1108     hres = IHTMLWindow2_get_status(window, &status);
1109     ok_(__FILE__,line)(hres == S_OK, "get_status failed: %08x\n", hres);
1110     ok_(__FILE__,line)(!status, "status = %s\n", wine_dbgstr_w(status));
1111     SysFreeString(status);
1112 }
1113
1114 #define set_window_status(w,n) _set_window_status(__LINE__,w,n)
1115 static void _set_window_status(unsigned line, IHTMLWindow2 *window, const char *status)
1116 {
1117     BSTR str;
1118     HRESULT hres;
1119
1120     str = a2bstr(status);
1121     hres = IHTMLWindow2_put_status(window, str);
1122     SysFreeString(str);
1123     ok_(__FILE__,line)(hres == S_OK, "put_status failed: %08x\n", hres);
1124 }
1125
1126 #define test_window_length(w,l) _test_window_length(__LINE__,w,l)
1127 static void _test_window_length(unsigned line, IHTMLWindow2 *window, LONG exlen)
1128 {
1129     LONG length = -1;
1130     HRESULT hres;
1131
1132     hres = IHTMLWindow2_get_length(window, &length);
1133     ok_(__FILE__,line)(hres == S_OK, "get_length failed: %08x\n", hres);
1134     ok_(__FILE__,line)(length == exlen, "length = %d, expected %d\n", length, exlen);
1135 }
1136
1137 #define get_frame_content_window(e) _get_frame_content_window(__LINE__,e)
1138 static IHTMLWindow2 *_get_frame_content_window(unsigned line, IUnknown *elem)
1139 {
1140     IHTMLFrameBase2 *base2;
1141     IHTMLWindow2 *window;
1142     HRESULT hres;
1143
1144     hres = IUnknown_QueryInterface(elem, &IID_IHTMLFrameBase2, (void**)&base2);
1145     ok(hres == S_OK, "Could not get IHTMFrameBase2 iface: %08x\n", hres);
1146
1147     window = NULL;
1148     hres = IHTMLFrameBase2_get_contentWindow(base2, &window);
1149     IHTMLFrameBase2_Release(base2);
1150     ok(hres == S_OK, "get_contentWindow failed: %08x\n", hres);
1151     ok(window != NULL, "contentWindow = NULL\n");
1152
1153     return window;
1154 }
1155
1156 static void test_get_set_attr(IHTMLDocument2 *doc)
1157 {
1158     IHTMLElement *elem;
1159     IHTMLDocument3 *doc3;
1160     HRESULT hres;
1161     BSTR bstr;
1162     VARIANT val;
1163
1164     /* grab an element to test with */
1165     hres = IHTMLDocument2_QueryInterface(doc, &IID_IHTMLDocument3, (void**)&doc3);
1166     ok(hres == S_OK, "QueryInterface(IID_IHTMLDocument3) failed: %08x\n", hres);
1167
1168     hres = IHTMLDocument3_get_documentElement(doc3, &elem);
1169     IHTMLDocument3_Release(doc3);
1170     ok(hres == S_OK, "get_documentElement failed: %08x\n", hres);
1171
1172     /* get a non-present attribute */
1173     bstr = a2bstr("notAnAttribute");
1174     hres = IHTMLElement_getAttribute(elem, bstr, 0, &val);
1175     ok(hres == S_OK, "getAttribute failed: %08x\n", hres);
1176     ok(V_VT(&val) == VT_NULL, "variant type should have been VT_NULL (0x%x), was: 0x%x\n", VT_NULL, V_VT(&val));
1177     VariantClear(&val);
1178     SysFreeString(bstr);
1179
1180     /* get a present attribute */
1181     bstr = a2bstr("scrollHeight");
1182     hres = IHTMLElement_getAttribute(elem, bstr, 0, &val);
1183     ok(hres == S_OK, "getAttribute failed: %08x\n", hres);
1184     ok(V_VT(&val) == VT_I4, "variant type should have been VT_I4 (0x%x), was: 0x%x\n", VT_I4, V_VT(&val));
1185     VariantClear(&val);
1186     SysFreeString(bstr);
1187
1188     /* create a new BSTR attribute */
1189     bstr = a2bstr("newAttribute");
1190
1191     V_VT(&val) = VT_BSTR;
1192     V_BSTR(&val) = a2bstr("the value");
1193     hres = IHTMLElement_setAttribute(elem, bstr, val, 0);
1194     ok(hres == S_OK, "setAttribute failed: %08x\n", hres);
1195     VariantClear(&val);
1196
1197     hres = IHTMLElement_getAttribute(elem, bstr, 0, &val);
1198     ok(hres == S_OK, "getAttribute failed: %08x\n", hres);
1199     ok(V_VT(&val) == VT_BSTR, "variant type should have been VT_BSTR (0x%x), was: 0x%x\n", VT_BSTR, V_VT(&val));
1200     ok(strcmp_wa(V_BSTR(&val), "the value") == 0, "variant value should have been L\"the value\", was %s\n", wine_dbgstr_w(V_BSTR(&val)));
1201     VariantClear(&val);
1202
1203     /* overwrite the attribute with a BOOL */
1204     V_VT(&val) = VT_BOOL;
1205     V_BOOL(&val) = VARIANT_TRUE;
1206     hres = IHTMLElement_setAttribute(elem, bstr, val, 0);
1207     ok(hres == S_OK, "setAttribute failed: %08x\n", hres);
1208     VariantClear(&val);
1209
1210     hres = IHTMLElement_getAttribute(elem, bstr, 0, &val);
1211     ok(hres == S_OK, "getAttribute failed: %08x\n", hres);
1212     ok(V_VT(&val) == VT_BOOL, "variant type should have been VT_BOOL (0x%x), was: 0x%x\n", VT_BOOL, V_VT(&val));
1213     ok(V_BOOL(&val) == VARIANT_TRUE, "variant value should have been VARIANT_TRUE (0x%x), was %d\n", VARIANT_TRUE, V_BOOL(&val));
1214     VariantClear(&val);
1215
1216     SysFreeString(bstr);
1217
1218     /* case-insensitive */
1219     bstr = a2bstr("newattribute");
1220     hres = IHTMLElement_getAttribute(elem, bstr, 0, &val);
1221     ok(hres == S_OK, "getAttribute failed: %08x\n", hres);
1222     ok(V_VT(&val) == VT_BOOL, "variant type should have been VT_BOOL (0x%x), was: 0x%x\n", VT_BOOL, V_VT(&val));
1223     ok(V_BOOL(&val) == VARIANT_TRUE, "variant value should have been VARIANT_TRUE (0x%x), was %d\n", VARIANT_TRUE, V_BOOL(&val));
1224     VariantClear(&val);
1225     SysFreeString(bstr);
1226
1227     IHTMLElement_Release(elem);
1228 }
1229
1230 #define get_doc_elem(d) _get_doc_elem(__LINE__,d)
1231 static IHTMLElement *_get_doc_elem(unsigned line, IHTMLDocument2 *doc)
1232 {
1233     IHTMLElement *elem;
1234     IHTMLDocument3 *doc3;
1235     HRESULT hres;
1236
1237     hres = IHTMLDocument2_QueryInterface(doc, &IID_IHTMLDocument3, (void**)&doc3);
1238     ok_(__FILE__,line) (hres == S_OK, "Could not get IHTMLDocument3 interface: %08x\n", hres);
1239     hres = IHTMLDocument3_get_documentElement(doc3, &elem);
1240     ok_(__FILE__,line) (hres == S_OK, "get_documentElement failed: %08x\n", hres);
1241     IHTMLDocument3_Release(doc3);
1242
1243     return elem;
1244 }
1245
1246 #define test_anchor_href(a,h) _test_anchor_href(__LINE__,a,h)
1247 static void _test_anchor_href(unsigned line, IUnknown *unk, const char *exhref)
1248 {
1249     IHTMLAnchorElement *anchor = _get_anchor_iface(line, unk);
1250     BSTR str;
1251     HRESULT hres;
1252
1253     hres = IHTMLAnchorElement_get_href(anchor, &str);
1254     ok_(__FILE__,line)(hres == S_OK, "get_href failed: %08x\n", hres);
1255     ok_(__FILE__,line)(!strcmp_wa(str, exhref), "href = %s, expected %s\n", wine_dbgstr_w(str), exhref);
1256     SysFreeString(str);
1257
1258     _test_disp_value(line, unk, exhref);
1259 }
1260
1261 #define test_anchor_put_href(a,h) _test_anchor_put_href(__LINE__,a,h)
1262 static void _test_anchor_put_href(unsigned line, IUnknown *unk, const char *exhref)
1263 {
1264     IHTMLAnchorElement *anchor = _get_anchor_iface(line, unk);
1265     BSTR str;
1266     HRESULT hres;
1267
1268     str = a2bstr(exhref);
1269     hres = IHTMLAnchorElement_put_href(anchor, str);
1270     ok_(__FILE__,line)(hres == S_OK, "get_href failed: %08x\n", hres);
1271     SysFreeString(str);
1272
1273     _test_disp_value(line, unk, exhref);
1274 }
1275
1276 #define test_anchor_get_target(a,h) _test_anchor_get_target(__LINE__,a,h)
1277 static void _test_anchor_get_target(unsigned line, IUnknown *unk, const char *target)
1278 {
1279     IHTMLAnchorElement *anchor = _get_anchor_iface(line, unk);
1280     BSTR str;
1281     HRESULT hres;
1282
1283     hres = IHTMLAnchorElement_get_target(anchor, &str);
1284     ok_(__FILE__,line)(hres == S_OK, "get_target failed: %08x\n", hres);
1285     if(target)
1286         ok_(__FILE__,line)(!strcmp_wa(str, target), "target = %s, expected %s\n", wine_dbgstr_w(str), target);
1287     else
1288         ok_(__FILE__,line)(str == NULL, "target = %s, expected NULL\n", wine_dbgstr_w(str));
1289     SysFreeString(str);
1290 }
1291
1292 #define test_anchor_put_target(a,h) _test_anchor_put_target(__LINE__,a,h)
1293 static void _test_anchor_put_target(unsigned line, IUnknown *unk, const char *target)
1294 {
1295     IHTMLAnchorElement *anchor = _get_anchor_iface(line, unk);
1296     BSTR str;
1297     HRESULT hres;
1298
1299     str = target ? a2bstr(target) : NULL;
1300     hres = IHTMLAnchorElement_put_target(anchor, str);
1301     ok_(__FILE__,line)(hres == S_OK, "put_target failed: %08x\n", hres);
1302     SysFreeString(str);
1303 }
1304
1305 #define test_anchor_name(a,h) _test_anchor_name(__LINE__,a,h)
1306 static void _test_anchor_name(unsigned line, IUnknown *unk, const char *name)
1307 {
1308     IHTMLAnchorElement *anchor = _get_anchor_iface(line, unk);
1309     BSTR str;
1310     HRESULT hres;
1311
1312     hres = IHTMLAnchorElement_get_name(anchor, &str);
1313     ok_(__FILE__,line)(hres == S_OK, "get_name failed: %08x\n", hres);
1314     if(name)
1315         ok_(__FILE__,line)(!strcmp_wa(str, name), "name = %s, expected %s\n", wine_dbgstr_w(str), name);
1316     else
1317         ok_(__FILE__,line)(str == NULL, "name = %s, expected NULL\n", wine_dbgstr_w(str));
1318     SysFreeString(str);
1319 }
1320
1321 #define test_anchor_put_name(a,h) _test_anchor_put_name(__LINE__,a,h)
1322 static void _test_anchor_put_name(unsigned line, IUnknown *unk, const char *name)
1323 {
1324     IHTMLAnchorElement *anchor = _get_anchor_iface(line, unk);
1325     BSTR str;
1326     HRESULT hres;
1327
1328     str = name ? a2bstr(name) : NULL;
1329     hres = IHTMLAnchorElement_put_name(anchor, str);
1330     ok_(__FILE__,line)(hres == S_OK, "put_name failed: %08x\n", hres);
1331     SysFreeString(str);
1332
1333     _test_anchor_name(line, unk, name);
1334 }
1335
1336
1337 #define test_option_text(o,t) _test_option_text(__LINE__,o,t)
1338 static void _test_option_text(unsigned line, IHTMLOptionElement *option, const char *text)
1339 {
1340     BSTR bstr;
1341     HRESULT hres;
1342
1343     hres = IHTMLOptionElement_get_text(option, &bstr);
1344     ok_(__FILE__,line) (hres == S_OK, "get_text failed: %08x\n", hres);
1345     ok_(__FILE__,line) (!strcmp_wa(bstr, text), "text=%s\n", wine_dbgstr_w(bstr));
1346     SysFreeString(bstr);
1347 }
1348
1349 #define test_option_put_text(o,t) _test_option_put_text(__LINE__,o,t)
1350 static void _test_option_put_text(unsigned line, IHTMLOptionElement *option, const char *text)
1351 {
1352     BSTR bstr;
1353     HRESULT hres;
1354
1355     bstr = a2bstr(text);
1356     hres = IHTMLOptionElement_put_text(option, bstr);
1357     SysFreeString(bstr);
1358     ok(hres == S_OK, "put_text failed: %08x\n", hres);
1359
1360     _test_option_text(line, option, text);
1361 }
1362
1363 #define test_option_value(o,t) _test_option_value(__LINE__,o,t)
1364 static void _test_option_value(unsigned line, IHTMLOptionElement *option, const char *value)
1365 {
1366     BSTR bstr;
1367     HRESULT hres;
1368
1369     hres = IHTMLOptionElement_get_value(option, &bstr);
1370     ok_(__FILE__,line) (hres == S_OK, "get_value failed: %08x\n", hres);
1371     ok_(__FILE__,line) (!strcmp_wa(bstr, value), "value=%s\n", wine_dbgstr_w(bstr));
1372     SysFreeString(bstr);
1373 }
1374
1375 #define test_option_put_value(o,t) _test_option_put_value(__LINE__,o,t)
1376 static void _test_option_put_value(unsigned line, IHTMLOptionElement *option, const char *value)
1377 {
1378     BSTR bstr;
1379     HRESULT hres;
1380
1381     bstr = a2bstr(value);
1382     hres = IHTMLOptionElement_put_value(option, bstr);
1383     SysFreeString(bstr);
1384     ok(hres == S_OK, "put_value failed: %08x\n", hres);
1385
1386     _test_option_value(line, option, value);
1387 }
1388
1389 #define test_option_selected(o,s) _test_option_selected(__LINE__,o,s)
1390 static void _test_option_selected(unsigned line, IHTMLOptionElement *option, VARIANT_BOOL ex)
1391 {
1392     VARIANT_BOOL b = 0x100;
1393     HRESULT hres;
1394
1395     hres = IHTMLOptionElement_get_selected(option, &b);
1396     ok_(__FILE__,line)(hres == S_OK, "get_selected failed: %08x\n", hres);
1397     ok_(__FILE__,line)(b == ex, "selected = %x, expected %x\n", b, ex);
1398 }
1399
1400 #define test_option_put_selected(o,s) _test_option_put_selected(__LINE__,o,s)
1401 static void _test_option_put_selected(unsigned line, IHTMLOptionElement *option, VARIANT_BOOL b)
1402 {
1403     HRESULT hres;
1404
1405     hres = IHTMLOptionElement_put_selected(option, b);
1406     ok_(__FILE__,line)(hres == S_OK, "put_selected failed: %08x\n", hres);
1407     _test_option_selected(line, option, b);
1408 }
1409
1410 #define test_textarea_value(t,v) _test_textarea_value(__LINE__,t,v)
1411 static void _test_textarea_value(unsigned line, IUnknown *unk, const char *exval)
1412 {
1413     IHTMLTextAreaElement *textarea = _get_textarea_iface(line, unk);
1414     BSTR value = (void*)0xdeadbeef;
1415     HRESULT hres;
1416
1417     hres = IHTMLTextAreaElement_get_value(textarea, &value);
1418     IHTMLTextAreaElement_Release(textarea);
1419     ok_(__FILE__,line)(hres == S_OK, "get_value failed: %08x\n", hres);
1420     if(exval)
1421         ok_(__FILE__,line)(!strcmp_wa(value, exval), "value = %s, expected %s\n", wine_dbgstr_w(value), exval);
1422     else
1423         ok_(__FILE__,line)(!value, "value = %p\n", value);
1424     SysFreeString(value);
1425 }
1426
1427 #define test_textarea_put_value(t,v) _test_textarea_put_value(__LINE__,t,v)
1428 static void _test_textarea_put_value(unsigned line, IUnknown *unk, const char *value)
1429 {
1430     IHTMLTextAreaElement *textarea = _get_textarea_iface(line, unk);
1431     BSTR tmp = a2bstr(value);
1432     HRESULT hres;
1433
1434     hres = IHTMLTextAreaElement_put_value(textarea, tmp);
1435     IHTMLTextAreaElement_Release(textarea);
1436     ok_(__FILE__,line)(hres == S_OK, "put_value failed: %08x\n", hres);
1437     SysFreeString(tmp);
1438
1439     _test_textarea_value(line, unk, value);
1440 }
1441
1442 #define test_textarea_readonly(t,v) _test_textarea_readonly(__LINE__,t,v)
1443 static void _test_textarea_readonly(unsigned line, IUnknown *unk, VARIANT_BOOL ex)
1444 {
1445     IHTMLTextAreaElement *textarea = _get_textarea_iface(line, unk);
1446     VARIANT_BOOL b = 0x100;
1447     HRESULT hres;
1448
1449     hres = IHTMLTextAreaElement_get_readOnly(textarea, &b);
1450     IHTMLTextAreaElement_Release(textarea);
1451     ok_(__FILE__,line)(hres == S_OK, "get_readOnly failed: %08x\n", hres);
1452     ok_(__FILE__,line)(b == ex, "readOnly = %x, expected %x\n", b, ex);
1453 }
1454
1455 #define test_textarea_put_readonly(t,v) _test_textarea_put_readonly(__LINE__,t,v)
1456 static void _test_textarea_put_readonly(unsigned line, IUnknown *unk, VARIANT_BOOL b)
1457 {
1458     IHTMLTextAreaElement *textarea = _get_textarea_iface(line, unk);
1459     HRESULT hres;
1460
1461     hres = IHTMLTextAreaElement_put_readOnly(textarea, b);
1462     IHTMLTextAreaElement_Release(textarea);
1463     ok_(__FILE__,line)(hres == S_OK, "put_readOnly failed: %08x\n", hres);
1464
1465     _test_textarea_readonly(line, unk, b);
1466 }
1467
1468 #define test_textarea_type(t) _test_textarea_type(__LINE__,t)
1469 static void _test_textarea_type(unsigned line, IUnknown *unk)
1470 {
1471     IHTMLTextAreaElement *textarea = _get_textarea_iface(line, unk);
1472     BSTR type = (void*)0xdeadbeef;
1473     HRESULT hres;
1474
1475     hres = IHTMLTextAreaElement_get_type(textarea, &type);
1476     IHTMLTextAreaElement_Release(textarea);
1477     ok_(__FILE__,line)(hres == S_OK, "get_type failed: %08x\n", hres);
1478     ok_(__FILE__,line)(!strcmp_wa(type, "textarea"), "type = %s, expected textarea\n", wine_dbgstr_w(type));
1479     SysFreeString(type);
1480 }
1481
1482 #define test_comment_text(c,t) _test_comment_text(__LINE__,c,t)
1483 static void _test_comment_text(unsigned line, IUnknown *unk, const char *extext)
1484 {
1485     IHTMLCommentElement *comment = _get_comment_iface(__LINE__,unk);
1486     BSTR text;
1487     HRESULT hres;
1488
1489     text = a2bstr(extext);
1490     hres = IHTMLCommentElement_get_text(comment, &text);
1491     ok_(__FILE__,line)(hres == S_OK, "get_text failed: %08x\n", hres);
1492     ok_(__FILE__,line)(!strcmp_wa(text, extext), "text = \"%s\", expected \"%s\"\n", wine_dbgstr_w(text), extext);
1493
1494     IHTMLCommentElement_Release(comment);
1495     SysFreeString(text);
1496 }
1497
1498 #define test_comment_attrs(c) _test_comment_attrs(__LINE__,c)
1499 static void _test_comment_attrs(unsigned line, IUnknown *unk)
1500 {
1501     IHTMLCommentElement *comment = _get_comment_iface(__LINE__,unk);
1502     IHTMLElement *elem = _get_elem_iface(__LINE__,unk);
1503     IHTMLElement4 *elem4 = _get_elem4_iface(__LINE__,unk);
1504     IHTMLDOMAttribute *attr;
1505     BSTR name = a2bstr("test");
1506     VARIANT val;
1507     HRESULT hres;
1508
1509     hres = IHTMLElement4_getAttributeNode(elem4, name, &attr);
1510     ok(hres == S_OK, "getAttributeNode failed: %08x\n", hres);
1511     ok(attr == NULL, "attr != NULL\n");
1512
1513     V_VT(&val) = VT_I4;
1514     V_I4(&val) = 1234;
1515     hres = IHTMLElement_setAttribute(elem, name, val, 0);
1516     ok(hres == S_OK, "setAttribute failed: %08x\n", hres);
1517
1518     hres = IHTMLElement4_getAttributeNode(elem4, name, &attr);
1519     ok(hres == S_OK, "getAttributeNode failed: %08x\n", hres);
1520     ok(attr != NULL, "attr == NULL\n");
1521
1522     IHTMLDOMAttribute_Release(attr);
1523     IHTMLCommentElement_Release(comment);
1524     IHTMLElement_Release(elem);
1525     IHTMLElement4_Release(elem4);
1526     SysFreeString(name);
1527 }
1528
1529 #define test_object_vspace(u,s) _test_object_vspace(__LINE__,u,s)
1530 static void _test_object_vspace(unsigned line, IUnknown *unk, LONG exl)
1531 {
1532     IHTMLObjectElement *object = _get_object_iface(line, unk);
1533     LONG l;
1534     HRESULT hres;
1535
1536     l = 0xdeadbeef;
1537     hres = IHTMLObjectElement_get_vspace(object, &l);
1538     ok_(__FILE__,line)(hres == S_OK, "get_vspace failed: %08x\n", hres);
1539     ok_(__FILE__,line)(l == exl, "vspace=%d, expected %d\n", l, exl);
1540     IHTMLObjectElement_Release(object);
1541 }
1542
1543 #define create_option_elem(d,t,v) _create_option_elem(__LINE__,d,t,v)
1544 static IHTMLOptionElement *_create_option_elem(unsigned line, IHTMLDocument2 *doc,
1545         const char *txt, const char *val)
1546 {
1547     IHTMLOptionElementFactory *factory;
1548     IHTMLOptionElement *option;
1549     IHTMLWindow2 *window;
1550     VARIANT text, value, empty;
1551     HRESULT hres;
1552
1553     hres = IHTMLDocument2_get_parentWindow(doc, &window);
1554     ok_(__FILE__,line) (hres == S_OK, "get_parentElement failed: %08x\n", hres);
1555
1556     hres = IHTMLWindow2_get_Option(window, &factory);
1557     IHTMLWindow2_Release(window);
1558     ok_(__FILE__,line) (hres == S_OK, "get_Option failed: %08x\n", hres);
1559
1560     V_VT(&text) = VT_BSTR;
1561     V_BSTR(&text) = a2bstr(txt);
1562     V_VT(&value) = VT_BSTR;
1563     V_BSTR(&value) = a2bstr(val);
1564     V_VT(&empty) = VT_EMPTY;
1565
1566     hres = IHTMLOptionElementFactory_create(factory, text, value, empty, empty, &option);
1567     ok_(__FILE__,line) (hres == S_OK, "create failed: %08x\n", hres);
1568
1569     IHTMLOptionElementFactory_Release(factory);
1570     VariantClear(&text);
1571     VariantClear(&value);
1572
1573     _test_option_text(line, option, txt);
1574     _test_option_value(line, option, val);
1575     _test_option_selected(line, option, VARIANT_FALSE);
1576
1577     return option;
1578 }
1579
1580 #define test_img_width(o,w) _test_img_width(__LINE__,o,w)
1581 static void _test_img_width(unsigned line, IHTMLImgElement *img, const LONG exp)
1582 {
1583     LONG found = -1;
1584     HRESULT hres;
1585
1586     hres = IHTMLImgElement_get_width(img, &found);
1587     ok_(__FILE__,line) (hres == S_OK, "get_width failed: %08x\n", hres);
1588     ok_(__FILE__,line) (found == exp, "width=%d\n", found);
1589 }
1590
1591 #define test_img_put_width(o,w) _test_img_put_width(__LINE__,o,w)
1592 static void _test_img_put_width(unsigned line, IHTMLImgElement *img, const LONG width)
1593 {
1594     HRESULT hres;
1595
1596     hres = IHTMLImgElement_put_width(img, width);
1597     ok(hres == S_OK, "put_width failed: %08x\n", hres);
1598
1599     _test_img_width(line, img, width);
1600 }
1601
1602 #define test_img_height(o,h) _test_img_height(__LINE__,o,h)
1603 static void _test_img_height(unsigned line, IHTMLImgElement *img, const LONG exp)
1604 {
1605     LONG found = -1;
1606     HRESULT hres;
1607
1608     hres = IHTMLImgElement_get_height(img, &found);
1609     ok_(__FILE__,line) (hres == S_OK, "get_height failed: %08x\n", hres);
1610     ok_(__FILE__,line) (found == exp, "height=%d\n", found);
1611 }
1612
1613 #define test_img_put_height(o,w) _test_img_put_height(__LINE__,o,w)
1614 static void _test_img_put_height(unsigned line, IHTMLImgElement *img, const LONG height)
1615 {
1616     HRESULT hres;
1617
1618     hres = IHTMLImgElement_put_height(img, height);
1619     ok(hres == S_OK, "put_height failed: %08x\n", hres);
1620
1621     _test_img_height(line, img, height);
1622 }
1623
1624 #define create_img_elem(d,t,v) _create_img_elem(__LINE__,d,t,v)
1625 static IHTMLImgElement *_create_img_elem(unsigned line, IHTMLDocument2 *doc,
1626         LONG wdth, LONG hght)
1627 {
1628     IHTMLImageElementFactory *factory;
1629     IHTMLImgElement *img;
1630     IHTMLWindow2 *window;
1631     VARIANT width, height;
1632     char buf[16];
1633     HRESULT hres;
1634
1635     hres = IHTMLDocument2_get_parentWindow(doc, &window);
1636     ok_(__FILE__,line) (hres == S_OK, "get_parentElement failed: %08x\n", hres);
1637
1638     hres = IHTMLWindow2_get_Image(window, &factory);
1639     IHTMLWindow2_Release(window);
1640     ok_(__FILE__,line) (hres == S_OK, "get_Image failed: %08x\n", hres);
1641
1642     test_ifaces((IUnknown*)factory, img_factory_iids);
1643     test_disp((IUnknown*)factory, &IID_IHTMLImageElementFactory, "[object]");
1644
1645     if(wdth >= 0){
1646         snprintf(buf, 16, "%d", wdth);
1647         V_VT(&width) = VT_BSTR;
1648         V_BSTR(&width) = a2bstr(buf);
1649     }else{
1650         V_VT(&width) = VT_EMPTY;
1651         wdth = 0;
1652     }
1653
1654     if(hght >= 0){
1655         snprintf(buf, 16, "%d", hght);
1656         V_VT(&height) = VT_BSTR;
1657         V_BSTR(&height) = a2bstr(buf);
1658     }else{
1659         V_VT(&height) = VT_EMPTY;
1660         hght = 0;
1661     }
1662
1663     hres = IHTMLImageElementFactory_create(factory, width, height, &img);
1664     ok_(__FILE__,line) (hres == S_OK, "create failed: %08x\n", hres);
1665
1666     IHTMLImageElementFactory_Release(factory);
1667     VariantClear(&width);
1668     VariantClear(&height);
1669
1670     if(SUCCEEDED(hres)) {
1671         _test_img_width(line, img, wdth);
1672         _test_img_height(line, img, hght);
1673         return img;
1674     }
1675
1676     return NULL;
1677 }
1678
1679 #define test_select_length(s,l) _test_select_length(__LINE__,s,l)
1680 static void _test_select_length(unsigned line, IHTMLSelectElement *select, LONG length)
1681 {
1682     LONG len = 0xdeadbeef;
1683     HRESULT hres;
1684
1685     hres = IHTMLSelectElement_get_length(select, &len);
1686     ok_(__FILE__,line) (hres == S_OK, "get_length failed: %08x\n", hres);
1687     ok_(__FILE__,line) (len == length, "len=%d, expected %d\n", len, length);
1688 }
1689
1690 #define test_select_put_length(s,l) _test_select_put_length(__LINE__,s,l)
1691 static void _test_select_put_length(unsigned line, IUnknown *unk, LONG length)
1692 {
1693     IHTMLSelectElement *select = _get_select_iface(line, unk);
1694     HRESULT hres;
1695
1696     hres = IHTMLSelectElement_put_length(select, length);
1697     ok_(__FILE__,line) (hres == S_OK, "put_length failed: %08x\n", hres);
1698     _test_select_length(line, select, length);
1699     IHTMLSelectElement_Release(select);
1700 }
1701
1702 #define test_select_selidx(s,i) _test_select_selidx(__LINE__,s,i)
1703 static void _test_select_selidx(unsigned line, IHTMLSelectElement *select, LONG index)
1704 {
1705     LONG idx = 0xdeadbeef;
1706     HRESULT hres;
1707
1708     hres = IHTMLSelectElement_get_selectedIndex(select, &idx);
1709     ok_(__FILE__,line) (hres == S_OK, "get_selectedIndex failed: %08x\n", hres);
1710     ok_(__FILE__,line) (idx == index, "idx=%d, expected %d\n", idx, index);
1711 }
1712
1713 #define test_select_put_selidx(s,i) _test_select_put_selidx(__LINE__,s,i)
1714 static void _test_select_put_selidx(unsigned line, IHTMLSelectElement *select, LONG index)
1715 {
1716     HRESULT hres;
1717
1718     hres = IHTMLSelectElement_put_selectedIndex(select, index);
1719     ok_(__FILE__,line) (hres == S_OK, "get_selectedIndex failed: %08x\n", hres);
1720     _test_select_selidx(line, select, index);
1721 }
1722
1723 #define test_select_value(s,v) _test_select_value(__LINE__,s,v)
1724 static void _test_select_value(unsigned line, IHTMLSelectElement *select, const char *exval)
1725 {
1726     BSTR val;
1727     HRESULT hres;
1728
1729     hres = IHTMLSelectElement_get_value(select, &val);
1730     ok_(__FILE__,line) (hres == S_OK, "get_value failed: %08x\n", hres);
1731     if(exval)
1732         ok_(__FILE__,line) (!strcmp_wa(val, exval), "unexpected value %s\n", wine_dbgstr_w(val));
1733     else
1734         ok_(__FILE__,line) (val == NULL, "val=%s, expected NULL\n", wine_dbgstr_w(val));
1735     SysFreeString(val);
1736 }
1737
1738 #define test_select_set_value(s,v) _test_select_set_value(__LINE__,s,v)
1739 static void _test_select_set_value(unsigned line, IHTMLSelectElement *select, const char *val)
1740 {
1741     BSTR bstr;
1742     HRESULT hres;
1743
1744     bstr = a2bstr(val);
1745     hres = IHTMLSelectElement_put_value(select, bstr);
1746     SysFreeString(bstr);
1747     ok_(__FILE__,line) (hres == S_OK, "put_value failed: %08x\n", hres);
1748 }
1749
1750 #define test_select_type(s,t) _test_select_type(__LINE__,s,t)
1751 static void _test_select_type(unsigned line, IHTMLSelectElement *select, const char *extype)
1752 {
1753     BSTR type;
1754     HRESULT hres;
1755
1756     hres = IHTMLSelectElement_get_type(select, &type);
1757     ok_(__FILE__,line) (hres == S_OK, "get_type failed: %08x\n", hres);
1758     ok_(__FILE__,line) (!strcmp_wa(type, extype), "type=%s, expected %s\n", wine_dbgstr_w(type), extype);
1759     SysFreeString(type);
1760 }
1761
1762 #define test_range_text(r,t) _test_range_text(__LINE__,r,t)
1763 static void _test_range_text(unsigned line, IHTMLTxtRange *range, const char *extext)
1764 {
1765     BSTR text;
1766     HRESULT hres;
1767
1768     hres = IHTMLTxtRange_get_text(range, &text);
1769     ok_(__FILE__, line) (hres == S_OK, "get_text failed: %08x\n", hres);
1770
1771     if(extext) {
1772         ok_(__FILE__, line) (text != NULL, "text == NULL\n");
1773         ok_(__FILE__, line) (!strcmp_wa(text, extext), "text=%s, expected %s\n", wine_dbgstr_w(text), extext);
1774     }else {
1775         ok_(__FILE__, line) (text == NULL, "text=%s, expected NULL\n", wine_dbgstr_w(text));
1776     }
1777
1778     SysFreeString(text);
1779
1780 }
1781
1782 #define test_range_collapse(r,b) _test_range_collapse(__LINE__,r,b)
1783 static void _test_range_collapse(unsigned line, IHTMLTxtRange *range, BOOL b)
1784 {
1785     HRESULT hres;
1786
1787     hres = IHTMLTxtRange_collapse(range, b);
1788     ok_(__FILE__, line) (hres == S_OK, "collapse failed: %08x\n", hres);
1789     _test_range_text(line, range, NULL);
1790 }
1791
1792 #define test_range_expand(r,u,b,t) _test_range_expand(__LINE__,r,u,b,t)
1793 static void _test_range_expand(unsigned line, IHTMLTxtRange *range, LPWSTR unit,
1794         VARIANT_BOOL exb, const char *extext)
1795 {
1796     VARIANT_BOOL b = 0xe0e0;
1797     HRESULT hres;
1798
1799     hres = IHTMLTxtRange_expand(range, unit, &b);
1800     ok_(__FILE__,line) (hres == S_OK, "expand failed: %08x\n", hres);
1801     ok_(__FILE__,line) (b == exb, "b=%x, expected %x\n", b, exb);
1802     _test_range_text(line, range, extext);
1803 }
1804
1805 #define test_range_move(r,u,c,e) _test_range_move(__LINE__,r,u,c,e)
1806 static void _test_range_move(unsigned line, IHTMLTxtRange *range, LPWSTR unit, LONG cnt, LONG excnt)
1807 {
1808     LONG c = 0xdeadbeef;
1809     HRESULT hres;
1810
1811     hres = IHTMLTxtRange_move(range, unit, cnt, &c);
1812     ok_(__FILE__,line) (hres == S_OK, "move failed: %08x\n", hres);
1813     ok_(__FILE__,line) (c == excnt, "count=%d, expected %d\n", c, excnt);
1814     _test_range_text(line, range, NULL);
1815 }
1816
1817 #define test_range_movestart(r,u,c,e) _test_range_movestart(__LINE__,r,u,c,e)
1818 static void _test_range_movestart(unsigned line, IHTMLTxtRange *range,
1819         LPWSTR unit, LONG cnt, LONG excnt)
1820 {
1821     LONG c = 0xdeadbeef;
1822     HRESULT hres;
1823
1824     hres = IHTMLTxtRange_moveStart(range, unit, cnt, &c);
1825     ok_(__FILE__,line) (hres == S_OK, "move failed: %08x\n", hres);
1826     ok_(__FILE__,line) (c == excnt, "count=%d, expected %d\n", c, excnt);
1827 }
1828
1829 #define test_range_moveend(r,u,c,e) _test_range_moveend(__LINE__,r,u,c,e)
1830 static void _test_range_moveend(unsigned line, IHTMLTxtRange *range, LPWSTR unit, LONG cnt, LONG excnt)
1831 {
1832     LONG c = 0xdeadbeef;
1833     HRESULT hres;
1834
1835     hres = IHTMLTxtRange_moveEnd(range, unit, cnt, &c);
1836     ok_(__FILE__,line) (hres == S_OK, "move failed: %08x\n", hres);
1837     ok_(__FILE__,line) (c == excnt, "count=%d, expected %d\n", c, excnt);
1838 }
1839
1840 #define test_range_put_text(r,t) _test_range_put_text(__LINE__,r,t)
1841 static void _test_range_put_text(unsigned line, IHTMLTxtRange *range, const char *text)
1842 {
1843     HRESULT hres;
1844     BSTR bstr = a2bstr(text);
1845
1846     hres = IHTMLTxtRange_put_text(range, bstr);
1847     ok_(__FILE__,line) (hres == S_OK, "put_text failed: %08x\n", hres);
1848     SysFreeString(bstr);
1849     _test_range_text(line, range, NULL);
1850 }
1851
1852 #define test_range_inrange(r1,r2,b) _test_range_inrange(__LINE__,r1,r2,b)
1853 static void _test_range_inrange(unsigned line, IHTMLTxtRange *range1, IHTMLTxtRange *range2, VARIANT_BOOL exb)
1854 {
1855     VARIANT_BOOL b;
1856     HRESULT hres;
1857
1858     b = 0xe0e0;
1859     hres = IHTMLTxtRange_inRange(range1, range2, &b);
1860     ok_(__FILE__,line) (hres == S_OK, "(1->2) isEqual failed: %08x\n", hres);
1861     ok_(__FILE__,line) (b == exb, "(1->2) b=%x, expected %x\n", b, exb);
1862 }
1863
1864 #define test_range_isequal(r1,r2,b) _test_range_isequal(__LINE__,r1,r2,b)
1865 static void _test_range_isequal(unsigned line, IHTMLTxtRange *range1, IHTMLTxtRange *range2, VARIANT_BOOL exb)
1866 {
1867     VARIANT_BOOL b;
1868     HRESULT hres;
1869
1870     b = 0xe0e0;
1871     hres = IHTMLTxtRange_isEqual(range1, range2, &b);
1872     ok_(__FILE__,line) (hres == S_OK, "(1->2) isEqual failed: %08x\n", hres);
1873     ok_(__FILE__,line) (b == exb, "(1->2) b=%x, expected %x\n", b, exb);
1874
1875     b = 0xe0e0;
1876     hres = IHTMLTxtRange_isEqual(range2, range1, &b);
1877     ok_(__FILE__,line) (hres == S_OK, "(2->1) isEqual failed: %08x\n", hres);
1878     ok_(__FILE__,line) (b == exb, "(2->1) b=%x, expected %x\n", b, exb);
1879
1880     if(exb) {
1881         test_range_inrange(range1, range2, VARIANT_TRUE);
1882         test_range_inrange(range2, range1, VARIANT_TRUE);
1883     }
1884 }
1885
1886 #define test_range_parent(r,t) _test_range_parent(__LINE__,r,t)
1887 static void _test_range_parent(unsigned line, IHTMLTxtRange *range, elem_type_t type)
1888 {
1889     IHTMLElement *elem;
1890     HRESULT hres;
1891
1892     hres = IHTMLTxtRange_parentElement(range, &elem);
1893     ok_(__FILE__,line) (hres == S_OK, "parentElement failed: %08x\n", hres);
1894
1895     _test_elem_type(line, (IUnknown*)elem, type);
1896
1897     IHTMLElement_Release(elem);
1898 }
1899
1900 #define test_elem_collection(c,t,l) _test_elem_collection(__LINE__,c,t,l)
1901 static void _test_elem_collection(unsigned line, IUnknown *unk,
1902         const elem_type_t *elem_types, LONG exlen)
1903 {
1904     IHTMLElementCollection *col;
1905     LONG len;
1906     DWORD i;
1907     VARIANT name, index;
1908     IDispatch *disp, *disp2;
1909     HRESULT hres;
1910
1911     hres = IUnknown_QueryInterface(unk, &IID_IHTMLElementCollection, (void**)&col);
1912     ok_(__FILE__,line) (hres == S_OK, "Could not get IHTMLElementCollection: %08x\n", hres);
1913
1914     test_disp((IUnknown*)col, &DIID_DispHTMLElementCollection, "[object]");
1915
1916     hres = IHTMLElementCollection_get_length(col, &len);
1917     ok_(__FILE__,line) (hres == S_OK, "get_length failed: %08x\n", hres);
1918     ok_(__FILE__,line) (len == exlen, "len=%d, expected %d\n", len, exlen);
1919
1920     if(len > exlen)
1921         len = exlen;
1922
1923     V_VT(&index) = VT_EMPTY;
1924
1925     for(i=0; i<len; i++) {
1926         V_VT(&name) = VT_I4;
1927         V_I4(&name) = i;
1928         disp = (void*)0xdeadbeef;
1929         hres = IHTMLElementCollection_item(col, name, index, &disp);
1930         ok_(__FILE__,line) (hres == S_OK, "item(%d) failed: %08x\n", i, hres);
1931         ok_(__FILE__,line) (disp != NULL, "item returned NULL\n");
1932         if(FAILED(hres) || !disp)
1933             continue;
1934
1935         _test_elem_type(line, (IUnknown*)disp, elem_types[i]);
1936
1937         if(!i) {
1938             V_VT(&name) = VT_UINT;
1939             V_I4(&name) = 0;
1940             disp2 = (void*)0xdeadbeef;
1941             hres = IHTMLElementCollection_item(col, name, index, &disp2);
1942             ok_(__FILE__,line) (hres == S_OK, "item(%d) failed: %08x\n", i, hres);
1943             ok_(__FILE__,line) (iface_cmp((IUnknown*)disp, (IUnknown*)disp2), "disp != disp2\n");
1944             if(disp2)
1945                 IDispatch_Release(disp2);
1946         }
1947
1948         IDispatch_Release(disp);
1949     }
1950
1951     V_VT(&name) = VT_I4;
1952     V_I4(&name) = len;
1953     disp = (void*)0xdeadbeef;
1954     hres = IHTMLElementCollection_item(col, name, index, &disp);
1955     ok_(__FILE__,line) (hres == S_OK, "item failed: %08x\n", hres);
1956     ok_(__FILE__,line) (disp == NULL, "disp != NULL\n");
1957
1958     V_VT(&name) = VT_I4;
1959     V_I4(&name) = -1;
1960     disp = (void*)0xdeadbeef;
1961     hres = IHTMLElementCollection_item(col, name, index, &disp);
1962     ok_(__FILE__,line) (hres == E_INVALIDARG, "item failed: %08x, expected E_INVALIDARG\n", hres);
1963     ok_(__FILE__,line) (disp == NULL, "disp != NULL\n");
1964
1965     IHTMLElementCollection_Release(col);
1966 }
1967
1968 #define test_elem_all(c,t,l) _test_elem_all(__LINE__,c,t,l)
1969 static void _test_elem_all(unsigned line, IUnknown *unk, const elem_type_t *elem_types, LONG exlen)
1970 {
1971     IHTMLElement *elem = _get_elem_iface(line, unk);
1972     IDispatch *disp;
1973     HRESULT hres;
1974
1975     hres = IHTMLElement_get_all(elem, &disp);
1976     IHTMLElement_Release(elem);
1977     ok_(__FILE__,line)(hres == S_OK, "get_all failed: %08x\n", hres);
1978
1979     _test_elem_collection(line, (IUnknown*)disp, elem_types, exlen);
1980     IDispatch_Release(disp);
1981 }
1982
1983 #define test_doc_all(a,b,c) _test_doc_all(__LINE__,a,b,c)
1984 static void _test_doc_all(unsigned line, IHTMLDocument2 *doc, const elem_type_t *elem_types, LONG exlen)
1985 {
1986     IHTMLElementCollection *col;
1987     HRESULT hres;
1988
1989     hres = IHTMLDocument2_get_all(doc, &col);
1990     ok_(__FILE__,line)(hres == S_OK, "get_all failed: %08x\n", hres);
1991
1992     _test_elem_collection(line, (IUnknown*)col, elem_types, exlen);
1993     IHTMLElementCollection_Release(col);
1994 }
1995
1996 #define test_elem_getelembytag(a,b,c,d) _test_elem_getelembytag(__LINE__,a,b,c,d)
1997 static void _test_elem_getelembytag(unsigned line, IUnknown *unk, elem_type_t type, LONG exlen, IHTMLElement **ret)
1998 {
1999     IHTMLElement2 *elem = _get_elem2_iface(line, unk);
2000     IHTMLElementCollection *col = NULL;
2001     elem_type_t *types = NULL;
2002     BSTR tmp;
2003     int i;
2004     HRESULT hres;
2005
2006     tmp = a2bstr(elem_type_infos[type].tag);
2007     hres = IHTMLElement2_getElementsByTagName(elem, tmp, &col);
2008     SysFreeString(tmp);
2009     IHTMLElement2_Release(elem);
2010     ok_(__FILE__,line) (hres == S_OK, "getElementByTagName failed: %08x\n", hres);
2011     ok_(__FILE__,line) (col != NULL, "col == NULL\n");
2012
2013     if(exlen) {
2014         types = HeapAlloc(GetProcessHeap(), 0, exlen*sizeof(elem_type_t));
2015         for(i=0; i<exlen; i++)
2016             types[i] = type;
2017     }
2018
2019     _test_elem_collection(line, (IUnknown*)col, types, exlen);
2020
2021     HeapFree(GetProcessHeap(), 0, types);
2022
2023     if(ret) {
2024         IDispatch *disp;
2025         VARIANT v;
2026
2027         V_VT(&v) = VT_I4;
2028         V_I4(&v) = 0;
2029         disp = NULL;
2030         hres = IHTMLElementCollection_item(col, v, v, &disp);
2031         ok(hres == S_OK, "item failed: %08x\n", hres);
2032         ok(disp != NULL, "disp == NULL\n");
2033         *ret = _get_elem_iface(line, (IUnknown*)disp);
2034         IDispatch_Release(disp);
2035     }
2036
2037     IHTMLElementCollection_Release(col);
2038 }
2039
2040 #define test_elem_innertext(e,t) _test_elem_innertext(__LINE__,e,t)
2041 static void _test_elem_innertext(unsigned line, IHTMLElement *elem, const char *extext)
2042 {
2043     BSTR text = NULL;
2044     HRESULT hres;
2045
2046     hres = IHTMLElement_get_innerText(elem, &text);
2047     ok_(__FILE__,line) (hres == S_OK, "get_innerText failed: %08x\n", hres);
2048     if(extext)
2049         ok_(__FILE__,line) (!strcmp_wa(text, extext), "get_innerText returned %s expected %s\n",
2050                             wine_dbgstr_w(text), extext);
2051     else
2052         ok_(__FILE__,line) (!text, "get_innerText returned %s expected NULL\n", wine_dbgstr_w(text));
2053     SysFreeString(text);
2054 }
2055
2056 #define test_elem_set_innertext(e,t) _test_elem_set_innertext(__LINE__,e,t)
2057 static void _test_elem_set_innertext(unsigned line, IHTMLElement *elem, const char *text)
2058 {
2059     IHTMLDOMChildrenCollection *col;
2060     BSTR str;
2061     HRESULT hres;
2062
2063     str = a2bstr(text);
2064     hres = IHTMLElement_put_innerText(elem, str);
2065     ok_(__FILE__,line) (hres == S_OK, "put_innerText failed: %08x\n", hres);
2066     SysFreeString(str);
2067
2068     _test_elem_innertext(line, elem, text);
2069
2070
2071     col = _get_child_nodes(line, (IUnknown*)elem);
2072     ok(col != NULL, "col == NULL\n");
2073     if(col) {
2074         LONG length = 0, type;
2075         IHTMLDOMNode *node;
2076
2077         hres = IHTMLDOMChildrenCollection_get_length(col, &length);
2078         ok(hres == S_OK, "get_length failed: %08x\n", hres);
2079         ok(length == 1, "length = %d\n", length);
2080
2081         node = _get_child_item(line, col, 0);
2082         ok(node != NULL, "node == NULL\n");
2083         if(node) {
2084             type = _get_node_type(line, (IUnknown*)node);
2085             ok(type == 3, "type=%d\n", type);
2086             IHTMLDOMNode_Release(node);
2087         }
2088
2089         IHTMLDOMChildrenCollection_Release(col);
2090     }
2091
2092 }
2093
2094 #define test_elem_innerhtml(e,t) _test_elem_innerhtml(__LINE__,e,t)
2095 static void _test_elem_innerhtml(unsigned line, IUnknown *unk, const char *inner_html)
2096 {
2097     IHTMLElement *elem = _get_elem_iface(line, unk);
2098     BSTR html;
2099     HRESULT hres;
2100
2101     hres = IHTMLElement_get_innerHTML(elem, &html);
2102     ok_(__FILE__,line)(hres == S_OK, "get_innerHTML failed: %08x\n", hres);
2103     if(inner_html)
2104         ok_(__FILE__,line)(!strcmp_wa(html, inner_html), "unexpected innerHTML: %s\n", wine_dbgstr_w(html));
2105     else
2106         ok_(__FILE__,line)(!html, "innerHTML = %s\n", wine_dbgstr_w(html));
2107
2108     IHTMLElement_Release(elem);
2109     SysFreeString(html);
2110 }
2111
2112 #define test_elem_set_innerhtml(e,t) _test_elem_set_innerhtml(__LINE__,e,t)
2113 static void _test_elem_set_innerhtml(unsigned line, IUnknown *unk, const char *inner_html)
2114 {
2115     IHTMLElement *elem = _get_elem_iface(line, unk);
2116     BSTR html;
2117     HRESULT hres;
2118
2119     html = a2bstr(inner_html);
2120     hres = IHTMLElement_put_innerHTML(elem, html);
2121     ok_(__FILE__,line)(hres == S_OK, "put_innerHTML failed: %08x\n", hres);
2122
2123     IHTMLElement_Release(elem);
2124     SysFreeString(html);
2125 }
2126
2127 #define test_elem_set_outerhtml(e,t) _test_elem_set_outerhtml(__LINE__,e,t)
2128 static void _test_elem_set_outerhtml(unsigned line, IUnknown *unk, const char *outer_html)
2129 {
2130     IHTMLElement *elem = _get_elem_iface(line, unk);
2131     BSTR html;
2132     HRESULT hres;
2133
2134     html = a2bstr(outer_html);
2135     hres = IHTMLElement_put_outerHTML(elem, html);
2136     ok_(__FILE__,line)(hres == S_OK, "put_outerHTML failed: %08x\n", hres);
2137
2138     IHTMLElement_Release(elem);
2139     SysFreeString(html);
2140 }
2141
2142 #define test_elem_outerhtml(e,t) _test_elem_outerhtml(__LINE__,e,t)
2143 static void _test_elem_outerhtml(unsigned line, IUnknown *unk, const char *outer_html)
2144 {
2145     IHTMLElement *elem = _get_elem_iface(line, unk);
2146     BSTR html;
2147     HRESULT hres;
2148
2149     hres = IHTMLElement_get_outerHTML(elem, &html);
2150     ok_(__FILE__,line)(hres == S_OK, "get_outerHTML failed: %08x\n", hres);
2151     ok_(__FILE__,line)(!strcmp_wa(html, outer_html), "outerHTML = '%s', expected '%s'\n", wine_dbgstr_w(html), outer_html);
2152
2153     IHTMLElement_Release(elem);
2154     SysFreeString(html);
2155 }
2156
2157 #define test_elem_contains(a,b,c) _test_elem_contains(__LINE__,a,b,c)
2158 static void _test_elem_contains(unsigned line, IHTMLElement *elem, IHTMLElement *elem2, VARIANT_BOOL exval)
2159 {
2160     VARIANT_BOOL b;
2161     HRESULT hres;
2162
2163     b = 100;
2164     hres = IHTMLElement_contains(elem, elem2, &b);
2165     ok_(__FILE__,line)(hres == S_OK, "contains failed: %08x\n", hres);
2166     ok_(__FILE__,line)(b == exval, "contains returned %x, expected %x\n", b, exval);
2167 }
2168
2169 #define get_first_child(n) _get_first_child(__LINE__,n)
2170 static IHTMLDOMNode *_get_first_child(unsigned line, IUnknown *unk)
2171 {
2172     IHTMLDOMNode *node = _get_node_iface(line, unk);
2173     IHTMLDOMNode *child = NULL;
2174     HRESULT hres;
2175
2176     hres = IHTMLDOMNode_get_firstChild(node, &child);
2177     IHTMLDOMNode_Release(node);
2178     ok_(__FILE__,line) (hres == S_OK, "get_firstChild failed: %08x\n", hres);
2179
2180     return child;
2181 }
2182
2183 #define test_node_has_child(u,b) _test_node_has_child(__LINE__,u,b)
2184 static void _test_node_has_child(unsigned line, IUnknown *unk, VARIANT_BOOL exb)
2185 {
2186     IHTMLDOMNode *node = _get_node_iface(line, unk);
2187     VARIANT_BOOL b = 0xdead;
2188     HRESULT hres;
2189
2190     hres = IHTMLDOMNode_hasChildNodes(node, &b);
2191     ok_(__FILE__,line) (hres == S_OK, "hasChildNodes failed: %08x\n", hres);
2192     ok_(__FILE__,line) (b == exb, "hasChildNodes=%x, expected %x\n", b, exb);
2193
2194     IHTMLDOMNode_Release(node);
2195 }
2196
2197 #define test_node_get_parent(u) _test_node_get_parent(__LINE__,u)
2198 static IHTMLDOMNode *_test_node_get_parent(unsigned line, IUnknown *unk)
2199 {
2200     IHTMLDOMNode *node = _get_node_iface(line, unk);
2201     IHTMLDOMNode *parent;
2202     HRESULT hres;
2203
2204     hres = IHTMLDOMNode_get_parentNode(node, &parent);
2205     IHTMLDOMNode_Release(node);
2206     ok_(__FILE__,line) (hres == S_OK, "get_parentNode failed: %08x\n", hres);
2207
2208     return parent;
2209 }
2210
2211 #define node_get_next(u) _node_get_next(__LINE__,u)
2212 static IHTMLDOMNode *_node_get_next(unsigned line, IUnknown *unk)
2213 {
2214     IHTMLDOMNode *node = _get_node_iface(line, unk);
2215     IHTMLDOMNode *next;
2216     HRESULT hres;
2217
2218     hres = IHTMLDOMNode_get_nextSibling(node, &next);
2219     IHTMLDOMNode_Release(node);
2220     ok_(__FILE__,line) (hres == S_OK, "get_nextSiblibg failed: %08x\n", hres);
2221
2222     return next;
2223 }
2224
2225 #define node_get_prev(u) _node_get_prev(__LINE__,u)
2226 static IHTMLDOMNode *_node_get_prev(unsigned line, IUnknown *unk)
2227 {
2228     IHTMLDOMNode *node = _get_node_iface(line, unk);
2229     IHTMLDOMNode *prev;
2230     HRESULT hres;
2231
2232     hres = IHTMLDOMNode_get_previousSibling(node, &prev);
2233     IHTMLDOMNode_Release(node);
2234     ok_(__FILE__,line) (hres == S_OK, "get_previousSibling failed: %08x\n", hres);
2235
2236     return prev;
2237 }
2238
2239 #define test_elem_get_parent(u) _test_elem_get_parent(__LINE__,u)
2240 static IHTMLElement *_test_elem_get_parent(unsigned line, IUnknown *unk)
2241 {
2242     IHTMLElement *elem = _get_elem_iface(line, unk);
2243     IHTMLElement *parent;
2244     HRESULT hres;
2245
2246     hres = IHTMLElement_get_parentElement(elem, &parent);
2247     IHTMLElement_Release(elem);
2248     ok_(__FILE__,line) (hres == S_OK, "get_parentElement failed: %08x\n", hres);
2249
2250     return parent;
2251 }
2252
2253 #define test_elem3_get_disabled(i,b) _test_elem3_get_disabled(__LINE__,i,b)
2254 static void _test_elem3_get_disabled(unsigned line, IUnknown *unk, VARIANT_BOOL exb)
2255 {
2256     IHTMLElement3 *elem3 = _get_elem3_iface(line, unk);
2257     VARIANT_BOOL disabled = 100;
2258     HRESULT hres;
2259
2260     if (!elem3) return;
2261     hres = IHTMLElement3_get_disabled(elem3, &disabled);
2262     ok_(__FILE__,line) (hres == S_OK, "get_disabled failed: %08x\n", hres);
2263     ok_(__FILE__,line) (disabled == exb, "disabled=%x, expected %x\n", disabled, exb);
2264     IHTMLElement3_Release(elem3);
2265 }
2266
2267 #define test_elem3_set_disabled(i,b) _test_elem3_set_disabled(__LINE__,i,b)
2268 static void _test_elem3_set_disabled(unsigned line, IUnknown *unk, VARIANT_BOOL b)
2269 {
2270     IHTMLElement3 *elem3 = _get_elem3_iface(line, unk);
2271     HRESULT hres;
2272
2273     if (!elem3) return;
2274     hres = IHTMLElement3_put_disabled(elem3, b);
2275     ok_(__FILE__,line) (hres == S_OK, "get_disabled failed: %08x\n", hres);
2276
2277     IHTMLElement3_Release(elem3);
2278     _test_elem3_get_disabled(line, unk, b);
2279 }
2280
2281 #define test_select_get_disabled(i,b) _test_select_get_disabled(__LINE__,i,b)
2282 static void _test_select_get_disabled(unsigned line, IHTMLSelectElement *select, VARIANT_BOOL exb)
2283 {
2284     VARIANT_BOOL disabled = 100;
2285     HRESULT hres;
2286
2287     hres = IHTMLSelectElement_get_disabled(select, &disabled);
2288     ok_(__FILE__,line) (hres == S_OK, "get_disabled failed: %08x\n", hres);
2289     ok_(__FILE__,line) (disabled == exb, "disabled=%x, expected %x\n", disabled, exb);
2290
2291     _test_elem3_get_disabled(line, (IUnknown*)select, exb);
2292 }
2293
2294 #define test_text_length(u,l) _test_text_length(__LINE__,u,l)
2295 static void _test_text_length(unsigned line, IUnknown *unk, LONG l)
2296 {
2297     IHTMLDOMTextNode *text = _get_text_iface(line, unk);
2298     LONG length;
2299     HRESULT hres;
2300
2301     hres = IHTMLDOMTextNode_get_length(text, &length);
2302     ok_(__FILE__,line)(hres == S_OK, "get_length failed: %08x\n", hres);
2303     ok_(__FILE__,line)(length == l, "length = %d, expected %d\n", length, l);
2304     IHTMLDOMTextNode_Release(text);
2305 }
2306
2307 #define test_select_set_disabled(i,b) _test_select_set_disabled(__LINE__,i,b)
2308 static void _test_select_set_disabled(unsigned line, IHTMLSelectElement *select, VARIANT_BOOL b)
2309 {
2310     HRESULT hres;
2311
2312     hres = IHTMLSelectElement_put_disabled(select, b);
2313     ok_(__FILE__,line) (hres == S_OK, "get_disabled failed: %08x\n", hres);
2314
2315     _test_select_get_disabled(line, select, b);
2316 }
2317
2318 #define elem_get_scroll_height(u) _elem_get_scroll_height(__LINE__,u)
2319 static LONG _elem_get_scroll_height(unsigned line, IUnknown *unk)
2320 {
2321     IHTMLElement2 *elem = _get_elem2_iface(line, unk);
2322     IHTMLTextContainer *txtcont;
2323     LONG l = -1, l2 = -1;
2324     HRESULT hres;
2325
2326     hres = IHTMLElement2_get_scrollHeight(elem, &l);
2327     ok_(__FILE__,line) (hres == S_OK, "get_scrollHeight failed: %08x\n", hres);
2328     IHTMLElement2_Release(elem);
2329
2330     hres = IUnknown_QueryInterface(unk, &IID_IHTMLTextContainer, (void**)&txtcont);
2331     ok_(__FILE__,line) (hres == S_OK, "Could not get IHTMLTextContainer: %08x\n", hres);
2332
2333     hres = IHTMLTextContainer_get_scrollHeight(txtcont, &l2);
2334     IHTMLTextContainer_Release(txtcont);
2335     ok_(__FILE__,line) (hres == S_OK, "IHTMLTextContainer::get_scrollHeight failed: %d\n", l2);
2336     ok_(__FILE__,line) (l == l2, "unexpected height %d, expected %d\n", l2, l);
2337
2338     return l;
2339 }
2340
2341 #define elem_get_scroll_width(u) _elem_get_scroll_width(__LINE__,u)
2342 static LONG _elem_get_scroll_width(unsigned line, IUnknown *unk)
2343 {
2344     IHTMLElement2 *elem = _get_elem2_iface(line, unk);
2345     IHTMLTextContainer *txtcont;
2346     LONG l = -1, l2 = -1;
2347     HRESULT hres;
2348
2349     hres = IHTMLElement2_get_scrollWidth(elem, &l);
2350     ok_(__FILE__,line) (hres == S_OK, "get_scrollWidth failed: %08x\n", hres);
2351     IHTMLElement2_Release(elem);
2352
2353     hres = IUnknown_QueryInterface(unk, &IID_IHTMLTextContainer, (void**)&txtcont);
2354     ok_(__FILE__,line) (hres == S_OK, "Could not get IHTMLTextContainer: %08x\n", hres);
2355
2356     hres = IHTMLTextContainer_get_scrollWidth(txtcont, &l2);
2357     IHTMLTextContainer_Release(txtcont);
2358     ok_(__FILE__,line) (hres == S_OK, "IHTMLTextContainer::get_scrollWidth failed: %d\n", l2);
2359     ok_(__FILE__,line) (l == l2, "unexpected width %d, expected %d\n", l2, l);
2360
2361     return l;
2362 }
2363
2364 #define elem_get_scroll_top(u) _elem_get_scroll_top(__LINE__,u)
2365 static LONG _elem_get_scroll_top(unsigned line, IUnknown *unk)
2366 {
2367     IHTMLElement2 *elem = _get_elem2_iface(line, unk);
2368     IHTMLTextContainer *txtcont;
2369     LONG l = -1, l2 = -1;
2370     HRESULT hres;
2371
2372     hres = IHTMLElement2_get_scrollTop(elem, &l);
2373     ok_(__FILE__,line) (hres == S_OK, "get_scrollTop failed: %08x\n", hres);
2374     IHTMLElement2_Release(elem);
2375
2376     hres = IUnknown_QueryInterface(unk, &IID_IHTMLTextContainer, (void**)&txtcont);
2377     ok_(__FILE__,line) (hres == S_OK, "Could not get IHTMLTextContainer: %08x\n", hres);
2378
2379     hres = IHTMLTextContainer_get_scrollTop(txtcont, &l2);
2380     IHTMLTextContainer_Release(txtcont);
2381     ok_(__FILE__,line) (hres == S_OK, "IHTMLTextContainer::get_scrollTop failed: %d\n", l2);
2382     ok_(__FILE__,line) (l == l2, "unexpected top %d, expected %d\n", l2, l);
2383
2384     return l;
2385 }
2386
2387 #define elem_get_scroll_left(u) _elem_get_scroll_left(__LINE__,u)
2388 static void _elem_get_scroll_left(unsigned line, IUnknown *unk)
2389 {
2390     IHTMLElement2 *elem = _get_elem2_iface(line, unk);
2391     IHTMLTextContainer *txtcont;
2392     LONG l = -1, l2 = -1;
2393     HRESULT hres;
2394
2395     hres = IHTMLElement2_get_scrollLeft(elem, NULL);
2396     ok(hres == E_INVALIDARG, "expect E_INVALIDARG got 0x%08x\n", hres);
2397
2398     hres = IHTMLElement2_get_scrollLeft(elem, &l);
2399     ok(hres == S_OK, "get_scrollTop failed: %08x\n", hres);
2400     IHTMLElement2_Release(elem);
2401
2402     hres = IUnknown_QueryInterface(unk, &IID_IHTMLTextContainer, (void**)&txtcont);
2403     ok(hres == S_OK, "Could not get IHTMLTextContainer: %08x\n", hres);
2404
2405     hres = IHTMLTextContainer_get_scrollLeft(txtcont, &l2);
2406     IHTMLTextContainer_Release(txtcont);
2407     ok(hres == S_OK, "IHTMLTextContainer::get_scrollLeft failed: %d\n", l2);
2408     ok(l == l2, "unexpected left %d, expected %d\n", l2, l);
2409 }
2410
2411 #define test_img_src(a,b,c) _test_img_src(__LINE__,a,b,c)
2412 static void _test_img_src(unsigned line, IUnknown *unk, const char *exsrc, const char *broken_src)
2413 {
2414     IHTMLImgElement *img = _get_img_iface(line, unk);
2415     BSTR src;
2416     HRESULT hres;
2417
2418     hres = IHTMLImgElement_get_src(img, &src);
2419     IHTMLImgElement_Release(img);
2420     ok_(__FILE__,line) (hres == S_OK, "get_src failed: %08x\n", hres);
2421     ok_(__FILE__,line) (!strcmp_wa(src, exsrc) || (broken_src && broken(!strcmp_wa(src, broken_src))),
2422         "get_src returned %s expected %s\n", wine_dbgstr_w(src), exsrc);
2423     SysFreeString(src);
2424 }
2425
2426 #define test_img_set_src(u,s) _test_img_set_src(__LINE__,u,s)
2427 static void _test_img_set_src(unsigned line, IUnknown *unk, const char *src)
2428 {
2429     IHTMLImgElement *img = _get_img_iface(line, unk);
2430     BSTR tmp;
2431     HRESULT hres;
2432
2433     tmp = a2bstr(src);
2434     hres = IHTMLImgElement_put_src(img, tmp);
2435     IHTMLImgElement_Release(img);
2436     SysFreeString(tmp);
2437     ok_(__FILE__,line) (hres == S_OK, "put_src failed: %08x\n", hres);
2438 }
2439
2440 #define test_img_alt(u,a) _test_img_alt(__LINE__,u,a)
2441 static void _test_img_alt(unsigned line, IUnknown *unk, const char *exalt)
2442 {
2443     IHTMLImgElement *img = _get_img_iface(line, unk);
2444     BSTR alt;
2445     HRESULT hres;
2446
2447     hres = IHTMLImgElement_get_alt(img, &alt);
2448     ok_(__FILE__,line) (hres == S_OK, "get_alt failed: %08x\n", hres);
2449     if(exalt)
2450         ok_(__FILE__,line) (!strcmp_wa(alt, exalt), "inexopected alt %s\n", wine_dbgstr_w(alt));
2451     else
2452         ok_(__FILE__,line) (!alt, "alt != NULL\n");
2453     SysFreeString(alt);
2454 }
2455
2456 #define test_img_set_alt(u,a) _test_img_set_alt(__LINE__,u,a)
2457 static void _test_img_set_alt(unsigned line, IUnknown *unk, const char *alt)
2458 {
2459     IHTMLImgElement *img = _get_img_iface(line, unk);
2460     BSTR tmp;
2461     HRESULT hres;
2462
2463     tmp = a2bstr(alt);
2464     hres = IHTMLImgElement_put_alt(img, tmp);
2465     ok_(__FILE__,line) (hres == S_OK, "get_alt failed: %08x\n", hres);
2466     SysFreeString(tmp);
2467
2468     _test_img_alt(line, unk, alt);
2469 }
2470
2471 #define test_img_name(u, c) _test_img_name(__LINE__,u, c)
2472 static void _test_img_name(unsigned line, IUnknown *unk, const char *pValue)
2473 {
2474     IHTMLImgElement *img = _get_img_iface(line, unk);
2475     BSTR sName;
2476     HRESULT hres;
2477
2478     hres = IHTMLImgElement_get_name(img, &sName);
2479     ok_(__FILE__,line) (hres == S_OK, "get_Name failed: %08x\n", hres);
2480     ok_(__FILE__,line) (!strcmp_wa (sName, pValue), "expected '%s' got '%s'\n", pValue, wine_dbgstr_w(sName));
2481     IHTMLImgElement_Release(img);
2482     SysFreeString(sName);
2483 }
2484
2485 static void test_dynamic_properties(IHTMLElement *elem)
2486 {
2487     static const WCHAR attr1W[] = {'a','t','t','r','1',0};
2488     IDispatchEx *dispex;
2489     BSTR name, attr1 = SysAllocString(attr1W);
2490     VARIANT_BOOL succ;
2491     VARIANT val;
2492     int checked_no = 0;
2493     DISPID id = DISPID_STARTENUM;
2494     HRESULT hres;
2495
2496     hres = IHTMLElement_QueryInterface(elem, &IID_IDispatchEx, (void**)&dispex);
2497     ok(hres == S_OK, "QueryInterface failed: %08x\n", hres);
2498
2499     hres = IHTMLElement_removeAttribute(elem, attr1, 0, &succ);
2500     ok(hres == S_OK, "removeAttribute failed: %08x\n", hres);
2501     ok(succ, "removeAttribute set succ to FALSE\n");
2502
2503     while(1) {
2504         hres = IDispatchEx_GetNextDispID(dispex, fdexEnumAll, id, &id);
2505         ok(hres==S_OK || hres==S_FALSE, "GetNextDispID failed: %08x\n", hres);
2506         if(hres != S_OK)
2507             break;
2508
2509         hres = IDispatchEx_GetMemberName(dispex, id, &name);
2510         ok(hres == S_OK, "GetMemberName failed: %08x\n", hres);
2511
2512         if(!strcmp_wa(name, "attr1"))
2513             ok(0, "attr1 should be removed\n");
2514         else if(!strcmp_wa(name, "attr2") || !strcmp_wa(name, "attr3"))
2515             checked_no++;
2516         SysFreeString(name);
2517     }
2518     ok(checked_no == 2, "checked_no=%d, expected 2\n", checked_no);
2519     IDispatchEx_Release(dispex);
2520
2521     V_VT(&val) = VT_BSTR;
2522     V_BSTR(&val) = attr1;
2523     hres = IHTMLElement_setAttribute(elem, attr1, val, 0);
2524     ok(hres == S_OK, "setAttribute failed: %08x\n", hres);
2525     SysFreeString(attr1);
2526 }
2527
2528 static void test_attr_collection_disp(IDispatch *disp)
2529 {
2530     IDispatchEx *dispex;
2531     IHTMLDOMAttribute *attr;
2532     DISPPARAMS dp = {NULL, NULL, 0, 0};
2533     VARIANT var;
2534     EXCEPINFO ei;
2535     DISPID id;
2536     BSTR bstr;
2537     HRESULT hres;
2538
2539     hres = IDispatch_QueryInterface(disp, &IID_IDispatchEx, (void**)&dispex);
2540     ok(hres == S_OK, "QueryInterface failed: %08x\n", hres);
2541
2542     bstr = a2bstr("0");
2543     hres = IDispatchEx_GetDispID(dispex, bstr, fdexNameCaseSensitive, &id);
2544     ok(hres == S_OK, "GetDispID failed: %08x\n", hres);
2545     SysFreeString(bstr);
2546
2547     VariantInit(&var);
2548     hres = IDispatchEx_InvokeEx(dispex, id, LOCALE_NEUTRAL, INVOKE_PROPERTYGET, &dp, &var, &ei, NULL);
2549     ok(hres == S_OK, "InvokeEx failed: %08x\n", hres);
2550     ok(V_VT(&var) == VT_DISPATCH, "V_VT(var)=%d\n", V_VT(&var));
2551     ok(V_DISPATCH(&var) != NULL, "V_DISPATCH(var) == NULL\n");
2552     VariantClear(&var);
2553
2554     bstr = a2bstr("attr1");
2555     hres = IDispatchEx_GetDispID(dispex, bstr, fdexNameCaseSensitive, &id);
2556     ok(hres == S_OK, "GetDispID failed: %08x\n", hres);
2557     SysFreeString(bstr);
2558
2559     VariantInit(&var);
2560     hres = IDispatchEx_InvokeEx(dispex, id, LOCALE_NEUTRAL, INVOKE_PROPERTYGET, &dp, &var, &ei, NULL);
2561     ok(hres == S_OK, "InvokeEx failed: %08x\n", hres);
2562     ok(V_VT(&var) == VT_DISPATCH, "V_VT(var)=%d\n", V_VT(&var));
2563     ok(V_DISPATCH(&var) != NULL, "V_DISPATCH(var) == NULL\n");
2564     hres = IDispatch_QueryInterface(V_DISPATCH(&var), &IID_IHTMLDOMAttribute, (void**)&attr);
2565     ok(hres == S_OK, "QueryInterface failed: %08x\n", hres);
2566     hres = IHTMLDOMAttribute_get_nodeName(attr, &bstr);
2567     ok(hres == S_OK, "get_nodeName failed: %08x\n", hres);
2568     ok(!strcmp_wa(bstr, "attr1"), "node name is %s, expected attr1\n", wine_dbgstr_w(bstr));
2569     SysFreeString(bstr);
2570     IHTMLDOMAttribute_Release(attr);
2571     VariantClear(&var);
2572
2573     IDispatchEx_Release(dispex);
2574 }
2575
2576 static void test_attr_collection(IHTMLElement *elem)
2577 {
2578     static const WCHAR testW[] = {'t','e','s','t',0};
2579
2580     IHTMLDOMNode *node;
2581     IDispatch *disp, *attr;
2582     IHTMLDOMAttribute *dom_attr;
2583     IHTMLAttributeCollection *attr_col;
2584     BSTR name = SysAllocString(testW);
2585     VARIANT id, val;
2586     LONG i, len, checked;
2587     HRESULT hres;
2588
2589     hres = IHTMLElement_QueryInterface(elem, &IID_IHTMLDOMNode, (void**)&node);
2590     ok(hres == S_OK, "QueryInterface failed: %08x\n", hres);
2591
2592     hres = IHTMLDOMNode_get_attributes(node, &disp);
2593     ok(hres == S_OK, "get_attributes failed: %08x\n", hres);
2594
2595     hres = IHTMLDOMNode_get_attributes(node, &attr);
2596     ok(hres == S_OK, "get_attributes failed: %08x\n", hres);
2597     ok(iface_cmp((IUnknown*)disp, (IUnknown*)attr), "disp != attr\n");
2598     IDispatch_Release(attr);
2599     IHTMLDOMNode_Release(node);
2600
2601     hres = IDispatch_QueryInterface(disp, &IID_IHTMLAttributeCollection, (void**)&attr_col);
2602     ok(hres == S_OK, "QueryInterface failed: %08x\n", hres);
2603
2604     hres = IHTMLAttributeCollection_get_length(attr_col, &i);
2605     ok(hres == S_OK, "get_length failed: %08x\n", hres);
2606
2607     V_VT(&val) = VT_I4;
2608     V_I4(&val) = 1;
2609     hres = IHTMLElement_setAttribute(elem, name, val, 0);
2610     ok(hres == S_OK, "setAttribute failed: %08x\n", hres);
2611     SysFreeString(name);
2612
2613     hres = IHTMLAttributeCollection_get_length(attr_col, &len);
2614     ok(hres == S_OK, "get_length failed: %08x\n", hres);
2615     ok(len == i+1, "get_length returned %d, expected %d\n", len, i+1);
2616
2617     checked = 0;
2618     for(i=0; i<len; i++) {
2619         V_VT(&id) = VT_I4;
2620         V_I4(&id) = i;
2621         hres = IHTMLAttributeCollection_item(attr_col, &id, &attr);
2622         ok(hres == S_OK, "%d) item failed: %08x\n", i, hres);
2623
2624         hres = IDispatch_QueryInterface(attr, &IID_IHTMLDOMAttribute, (void**)&dom_attr);
2625         ok(hres == S_OK, "%d) QueryInterface failed: %08x\n", i, hres);
2626         IDispatch_Release(attr);
2627
2628         hres = IHTMLDOMAttribute_get_nodeName(dom_attr, &name);
2629         ok(hres == S_OK, "%d) get_nodeName failed: %08x\n", i, hres);
2630
2631         if(!strcmp_wa(name, "id")) {
2632             checked++;
2633             hres = IHTMLDOMAttribute_get_nodeValue(dom_attr, &val);
2634             ok(hres == S_OK, "%d) get_nodeValue failed: %08x\n", i, hres);
2635             ok(V_VT(&val) == VT_BSTR, "id: V_VT(&val) = %d\n", V_VT(&val));
2636             ok(!strcmp_wa(V_BSTR(&val), "attr"), "id: V_BSTR(&val) = %s\n", wine_dbgstr_w(V_BSTR(&val)));
2637         } else if(!strcmp_wa(name, "attr1")) {
2638             checked++;
2639             hres = IHTMLDOMAttribute_get_nodeValue(dom_attr, &val);
2640             ok(hres == S_OK, "%d) get_nodeValue failed: %08x\n", i, hres);
2641             ok(V_VT(&val) == VT_BSTR, "attr1: V_VT(&val) = %d\n", V_VT(&val));
2642             ok(!strcmp_wa(V_BSTR(&val), "attr1"), "attr1: V_BSTR(&val) = %s\n", wine_dbgstr_w(V_BSTR(&val)));
2643         } else if(!strcmp_wa(name, "attr2")) {
2644             checked++;
2645             hres = IHTMLDOMAttribute_get_nodeValue(dom_attr, &val);
2646             ok(hres == S_OK, "%d) get_nodeValue failed: %08x\n", i, hres);
2647             ok(V_VT(&val) == VT_BSTR, "attr2: V_VT(&val) = %d\n", V_VT(&val));
2648             ok(!V_BSTR(&val), "attr2: V_BSTR(&val) != NULL\n");
2649         } else if(!strcmp_wa(name, "attr3")) {
2650             checked++;
2651             hres = IHTMLDOMAttribute_get_nodeValue(dom_attr, &val);
2652             ok(hres == S_OK, "%d) get_nodeValue failed: %08x\n", i, hres);
2653             ok(V_VT(&val) == VT_BSTR, "attr3: V_VT(&val) = %d\n", V_VT(&val));
2654             ok(!strcmp_wa(V_BSTR(&val), "attr3"), "attr3: V_BSTR(&val) = %s\n", wine_dbgstr_w(V_BSTR(&val)));
2655         } else if(!strcmp_wa(name, "test")) {
2656             checked++;
2657             hres = IHTMLDOMAttribute_get_nodeValue(dom_attr, &val);
2658             ok(hres == S_OK, "%d) get_nodeValue failed: %08x\n", i, hres);
2659             ok(V_VT(&val) == VT_I4, "test: V_VT(&val) = %d\n", V_VT(&val));
2660             ok(V_I4(&val) == 1, "test: V_I4(&val) = %d\n", V_I4(&val));
2661         }
2662
2663         IHTMLDOMAttribute_Release(dom_attr);
2664         SysFreeString(name);
2665         VariantClear(&val);
2666     }
2667     ok(checked==5, "invalid number of specified attributes (%d)\n", checked);
2668
2669     V_I4(&id) = len;
2670     hres = IHTMLAttributeCollection_item(attr_col, &id, &attr);
2671     ok(hres == E_INVALIDARG, "item failed: %08x\n", hres);
2672
2673     V_VT(&id) = VT_BSTR;
2674     V_BSTR(&id) = a2bstr("nonexisting");
2675     hres = IHTMLAttributeCollection_item(attr_col, &id, &attr);
2676     ok(hres == E_INVALIDARG, "item failed: %08x\n", hres);
2677     VariantClear(&id);
2678
2679     test_attr_collection_disp(disp);
2680
2681     IDispatch_Release(disp);
2682     IHTMLAttributeCollection_Release(attr_col);
2683 }
2684
2685 #define test_attr_specified(a,b) _test_attr_specified(__LINE__,a,b)
2686 static void _test_attr_specified(unsigned line, IHTMLDOMAttribute *attr, VARIANT_BOOL expected)
2687 {
2688     VARIANT_BOOL specified;
2689     HRESULT hres;
2690
2691     hres = IHTMLDOMAttribute_get_specified(attr, &specified);
2692     ok_(__FILE__,line)(hres == S_OK, "get_specified failed: %08x\n", hres);
2693     ok_(__FILE__,line)(specified == expected, "specified = %x, expected %x\n", specified, expected);
2694 }
2695
2696 #define test_input_type(i,t) _test_input_type(__LINE__,i,t)
2697 static void _test_input_type(unsigned line, IHTMLInputElement *input, const char *extype)
2698 {
2699     BSTR type;
2700     HRESULT hres;
2701
2702     hres = IHTMLInputElement_get_type(input, &type);
2703     ok_(__FILE__,line) (hres == S_OK, "get_type failed: %08x\n", hres);
2704     ok_(__FILE__,line) (!strcmp_wa(type, extype), "type=%s, expected %s\n", wine_dbgstr_w(type), extype);
2705     SysFreeString(type);
2706 }
2707
2708 #define test_input_name(u, c) _test_input_name(__LINE__,u, c)
2709 static void _test_input_name(unsigned line, IHTMLInputElement *input, const char *exname)
2710 {
2711     BSTR name = (BSTR)0xdeadbeef;
2712     HRESULT hres;
2713
2714     hres = IHTMLInputElement_get_name(input, &name);
2715     ok_(__FILE__,line) (hres == S_OK, "get_name failed: %08x\n", hres);
2716     if(exname)
2717         ok_(__FILE__,line) (!strcmp_wa (name, exname), "name=%s, expected %s\n", wine_dbgstr_w(name), exname);
2718     else
2719         ok_(__FILE__,line) (!name, "name=%p, expected NULL\n", name);
2720     SysFreeString(name);
2721 }
2722
2723 #define test_input_set_name(u, c) _test_input_set_name(__LINE__,u, c)
2724 static void _test_input_set_name(unsigned line, IHTMLInputElement *input, const char *name)
2725 {
2726     BSTR tmp = a2bstr(name);
2727     HRESULT hres;
2728
2729     hres = IHTMLInputElement_put_name(input, tmp);
2730     ok_(__FILE__,line) (hres == S_OK, "put_name failed: %08x\n", hres);
2731     SysFreeString(tmp);
2732
2733     _test_input_name(line, input, name);
2734 }
2735
2736 #define test_input_get_disabled(i,b) _test_input_get_disabled(__LINE__,i,b)
2737 static void _test_input_get_disabled(unsigned line, IHTMLInputElement *input, VARIANT_BOOL exb)
2738 {
2739     VARIANT_BOOL disabled = 100;
2740     HRESULT hres;
2741
2742     hres = IHTMLInputElement_get_disabled(input, &disabled);
2743     ok_(__FILE__,line) (hres == S_OK, "get_disabled failed: %08x\n", hres);
2744     ok_(__FILE__,line) (disabled == exb, "disabled=%x, expected %x\n", disabled, exb);
2745
2746     _test_elem3_get_disabled(line, (IUnknown*)input, exb);
2747 }
2748
2749 #define test_input_set_disabled(i,b) _test_input_set_disabled(__LINE__,i,b)
2750 static void _test_input_set_disabled(unsigned line, IHTMLInputElement *input, VARIANT_BOOL b)
2751 {
2752     HRESULT hres;
2753
2754     hres = IHTMLInputElement_put_disabled(input, b);
2755     ok_(__FILE__,line) (hres == S_OK, "get_disabled failed: %08x\n", hres);
2756
2757     _test_input_get_disabled(line, input, b);
2758 }
2759
2760 #define test_input_get_defaultchecked(i,b) _test_input_get_defaultchecked(__LINE__,i,b)
2761 static void _test_input_get_defaultchecked(unsigned line, IHTMLInputElement *input, VARIANT_BOOL exb)
2762 {
2763     VARIANT_BOOL checked = 100;
2764     HRESULT hres;
2765
2766     hres = IHTMLInputElement_get_defaultChecked(input, &checked);
2767     ok_(__FILE__,line) (hres == S_OK, "get_defaultChecked failed: %08x\n", hres);
2768     ok_(__FILE__,line) (checked == exb, "checked=%x, expected %x\n", checked, exb);
2769 }
2770
2771 #define test_input_set_defaultchecked(i,b) _test_input_set_defaultchecked(__LINE__,i,b)
2772 static void _test_input_set_defaultchecked(unsigned line, IHTMLInputElement *input, VARIANT_BOOL b)
2773 {
2774     HRESULT hres;
2775
2776     hres = IHTMLInputElement_put_defaultChecked(input, b);
2777     ok_(__FILE__,line) (hres == S_OK, "get_defaultChecked failed: %08x\n", hres);
2778
2779     _test_input_get_defaultchecked(line, input, b);
2780 }
2781
2782 #define test_input_get_checked(i,b) _test_input_get_checked(__LINE__,i,b)
2783 static void _test_input_get_checked(unsigned line, IHTMLInputElement *input, VARIANT_BOOL exb)
2784 {
2785     VARIANT_BOOL checked = 100;
2786     HRESULT hres;
2787
2788     hres = IHTMLInputElement_get_checked(input, &checked);
2789     ok_(__FILE__,line) (hres == S_OK, "get_checked failed: %08x\n", hres);
2790     ok_(__FILE__,line) (checked == exb, "checked=%x, expected %x\n", checked, exb);
2791 }
2792
2793 #define test_input_set_checked(i,b) _test_input_set_checked(__LINE__,i,b)
2794 static void _test_input_set_checked(unsigned line, IHTMLInputElement *input, VARIANT_BOOL b)
2795 {
2796     HRESULT hres;
2797
2798     hres = IHTMLInputElement_put_checked(input, b);
2799     ok_(__FILE__,line) (hres == S_OK, "put_checked failed: %08x\n", hres);
2800
2801     _test_input_get_checked(line, input, b);
2802 }
2803
2804 #define test_input_maxlength(i,b) _test_input_maxlength(__LINE__,i,b)
2805 static void _test_input_maxlength(unsigned line, IHTMLInputElement *input, LONG exl)
2806 {
2807     LONG maxlength = 0xdeadbeef;
2808     HRESULT hres;
2809
2810     hres = IHTMLInputElement_get_maxLength(input, &maxlength);
2811     ok_(__FILE__,line) (hres == S_OK, "get_maxLength failed: %08x\n", hres);
2812     ok_(__FILE__,line) (maxlength == exl, "maxLength=%x, expected %d\n", maxlength, exl);
2813 }
2814
2815 #define test_input_set_maxlength(i,b) _test_input_set_maxlength(__LINE__,i,b)
2816 static void _test_input_set_maxlength(unsigned line, IHTMLInputElement *input, LONG l)
2817 {
2818     HRESULT hres;
2819
2820     hres = IHTMLInputElement_put_maxLength(input, l);
2821     ok_(__FILE__,line) (hres == S_OK, "put_maxLength failed: %08x\n", hres);
2822
2823     _test_input_maxlength(line, input, l);
2824 }
2825
2826 #define test_input_value(o,t) _test_input_value(__LINE__,o,t)
2827 static void _test_input_value(unsigned line, IUnknown *unk, const char *exval)
2828 {
2829     IHTMLInputElement *input;
2830     BSTR bstr;
2831     HRESULT hres;
2832
2833     hres = IUnknown_QueryInterface(unk, &IID_IHTMLInputElement, (void**)&input);
2834     ok_(__FILE__,line) (hres == S_OK, "Could not get IHTMLInputElement: %08x\n", hres);
2835     if(FAILED(hres))
2836         return;
2837
2838     hres = IHTMLInputElement_get_value(input, &bstr);
2839     ok_(__FILE__,line) (hres == S_OK, "get_value failed: %08x\n", hres);
2840     if(exval)
2841         ok_(__FILE__,line) (!strcmp_wa(bstr, exval), "value=%s\n", wine_dbgstr_w(bstr));
2842     else
2843         ok_(__FILE__,line) (!bstr, "exval != NULL\n");
2844     SysFreeString(bstr);
2845     IHTMLInputElement_Release(input);
2846 }
2847
2848 #define test_input_put_value(o,v) _test_input_put_value(__LINE__,o,v)
2849 static void _test_input_put_value(unsigned line, IUnknown *unk, const char *val)
2850 {
2851     IHTMLInputElement *input;
2852     BSTR bstr;
2853     HRESULT hres;
2854
2855     hres = IUnknown_QueryInterface(unk, &IID_IHTMLInputElement, (void**)&input);
2856     ok_(__FILE__,line) (hres == S_OK, "Could not get IHTMLInputElement: %08x\n", hres);
2857     if(FAILED(hres))
2858         return;
2859
2860     bstr = a2bstr(val);
2861     hres = IHTMLInputElement_put_value(input, bstr);
2862     ok_(__FILE__,line) (hres == S_OK, "get_value failed: %08x\n", hres);
2863     SysFreeString(bstr);
2864     IHTMLInputElement_Release(input);
2865
2866     _test_input_value(line, unk, val);
2867 }
2868
2869 #define test_input_src(i,s) _test_input_src(__LINE__,i,s)
2870 static void _test_input_src(unsigned line, IHTMLInputElement *input, const char *exsrc)
2871 {
2872     BSTR src;
2873     HRESULT hres;
2874
2875     hres = IHTMLInputElement_get_src(input, &src);
2876     ok_(__FILE__,line) (hres == S_OK, "get_src failed: %08x\n", hres);
2877     if(exsrc)
2878         ok_(__FILE__,line) (!strcmp_wa(src, exsrc), "get_src returned %s expected %s\n", wine_dbgstr_w(src), exsrc);
2879     else
2880         ok_(__FILE__,line) (!src, "get_src returned %s expected NULL\n", wine_dbgstr_w(src));
2881     SysFreeString(src);
2882 }
2883
2884 #define test_input_set_src(u,s) _test_input_set_src(__LINE__,u,s)
2885 static void _test_input_set_src(unsigned line, IHTMLInputElement *input, const char *src)
2886 {
2887     BSTR tmp;
2888     HRESULT hres;
2889
2890     tmp = a2bstr(src);
2891     hres = IHTMLInputElement_put_src(input, tmp);
2892     SysFreeString(tmp);
2893     ok_(__FILE__,line) (hres == S_OK, "put_src failed: %08x\n", hres);
2894
2895     _test_input_src(line, input, src);
2896 }
2897
2898 #define test_elem_class(u,c) _test_elem_class(__LINE__,u,c)
2899 static void _test_elem_class(unsigned line, IUnknown *unk, const char *exclass)
2900 {
2901     IHTMLElement *elem = _get_elem_iface(line, unk);
2902     BSTR class = (void*)0xdeadbeef;
2903     HRESULT hres;
2904
2905     hres = IHTMLElement_get_className(elem, &class);
2906     IHTMLElement_Release(elem);
2907     ok_(__FILE__,line) (hres == S_OK, "get_className failed: %08x\n", hres);
2908     if(exclass)
2909         ok_(__FILE__,line) (!strcmp_wa(class, exclass), "unexpected className %s\n", wine_dbgstr_w(class));
2910     else
2911         ok_(__FILE__,line) (!class, "class != NULL\n");
2912     SysFreeString(class);
2913 }
2914
2915 #define test_elem_tabindex(u,i) _test_elem_tabindex(__LINE__,u,i)
2916 static void _test_elem_tabindex(unsigned line, IUnknown *unk, short exindex)
2917 {
2918     IHTMLElement2 *elem2 = _get_elem2_iface(line, unk);
2919     short index = -3;
2920     HRESULT hres;
2921
2922     hres = IHTMLElement2_get_tabIndex(elem2, &index);
2923     IHTMLElement2_Release(elem2);
2924     ok_(__FILE__,line) (hres == S_OK, "get_tabIndex failed: %08x\n", hres);
2925     ok_(__FILE__,line) (index == exindex, "unexpected index %d\n", index);
2926 }
2927
2928 #define test_elem_set_tabindex(u,i) _test_elem_set_tabindex(__LINE__,u,i)
2929 static void _test_elem_set_tabindex(unsigned line, IUnknown *unk, short index)
2930 {
2931     IHTMLElement2 *elem2 = _get_elem2_iface(line, unk);
2932     HRESULT hres;
2933
2934     hres = IHTMLElement2_put_tabIndex(elem2, index);
2935     IHTMLElement2_Release(elem2);
2936     ok_(__FILE__,line) (hres == S_OK, "get_tabIndex failed: %08x\n", hres);
2937
2938     _test_elem_tabindex(line, unk, index);
2939 }
2940
2941 #define test_style_media(s,m) _test_style_media(__LINE__,s,m)
2942 static void _test_style_media(unsigned line, IUnknown *unk, const char *exmedia)
2943 {
2944     IHTMLStyleElement *style = _get_style_iface(line, unk);
2945     BSTR media;
2946     HRESULT hres;
2947
2948     hres = IHTMLStyleElement_get_media(style, &media);
2949     ok_(__FILE__,line)(hres == S_OK, "get_media failed: %08x\n", hres);
2950     if(exmedia)
2951         ok_(__FILE__,line)(!strcmp_wa(media, exmedia), "media = %s, expected %s\n", wine_dbgstr_w(media), exmedia);
2952     else
2953         ok_(__FILE__,line)(!media, "media = %s, expected NULL\n", wine_dbgstr_w(media));
2954
2955     IHTMLStyleElement_Release(style);
2956     SysFreeString(media);
2957 }
2958
2959 #define test_style_put_media(s,m) _test_style_put_media(__LINE__,s,m)
2960 static void _test_style_put_media(unsigned line, IUnknown *unk, const char *media)
2961 {
2962     IHTMLStyleElement *style = _get_style_iface(line, unk);
2963     BSTR str;
2964     HRESULT hres;
2965
2966     str = a2bstr(media);
2967     hres = IHTMLStyleElement_put_media(style, str);
2968     ok_(__FILE__,line)(hres == S_OK, "put_media failed: %08x\n", hres);
2969     IHTMLStyleElement_Release(style);
2970     SysFreeString(str);
2971
2972     _test_style_media(line, unk, media);
2973 }
2974
2975 #define test_style_type(s,m) _test_style_type(__LINE__,s,m)
2976 static void _test_style_type(unsigned line, IUnknown *unk, const char *extype)
2977 {
2978     IHTMLStyleElement *style = _get_style_iface(line, unk);
2979     BSTR type;
2980     HRESULT hres;
2981
2982     hres = IHTMLStyleElement_get_type(style, &type);
2983     ok_(__FILE__,line)(hres == S_OK, "get_type failed: %08x\n", hres);
2984     if(extype)
2985         ok_(__FILE__,line)(!strcmp_wa(type, extype), "type = %s, expected %s\n", wine_dbgstr_w(type), extype);
2986     else
2987         ok_(__FILE__,line)(!type, "type = %s, expected NULL\n", wine_dbgstr_w(type));
2988
2989     IHTMLStyleElement_Release(style);
2990     SysFreeString(type);
2991 }
2992
2993 #define test_style_put_type(s,m) _test_style_put_type(__LINE__,s,m)
2994 static void _test_style_put_type(unsigned line, IUnknown *unk, const char *type)
2995 {
2996     IHTMLStyleElement *style = _get_style_iface(line, unk);
2997     BSTR str;
2998     HRESULT hres;
2999
3000     str = a2bstr(type);
3001     hres = IHTMLStyleElement_put_type(style, str);
3002     ok_(__FILE__,line)(hres == S_OK, "put_type failed: %08x\n", hres);
3003     IHTMLStyleElement_Release(style);
3004     SysFreeString(str);
3005
3006     _test_style_type(line, unk, type);
3007 }
3008
3009 #define test_elem_filters(u) _test_elem_filters(__LINE__,u)
3010 static void _test_elem_filters(unsigned line, IUnknown *unk)
3011 {
3012     IHTMLElement *elem = _get_elem_iface(line, unk);
3013     HRESULT hres;
3014     IHTMLFiltersCollection *filters;
3015
3016     hres = IHTMLElement_get_filters(elem, &filters);
3017     ok_(__FILE__,line) (hres == S_OK || broken(hres == REGDB_E_CLASSNOTREG) /* NT4 */,
3018                         "get_filters failed: %08x\n", hres);
3019     if(hres == S_OK)
3020     {
3021         LONG len;
3022         IDispatchEx *dispex;
3023
3024         hres = IHTMLFiltersCollection_get_length(filters, &len);
3025         ok_(__FILE__,line) (hres == S_OK, "get_length failed: %08x\n", hres);
3026         ok_(__FILE__,line) (len == 0, "expect 0 got %d\n", len);
3027
3028         hres = IHTMLFiltersCollection_QueryInterface(filters, &IID_IDispatchEx, (void**)&dispex);
3029         ok_(__FILE__,line) (hres == S_OK || broken(hres == E_NOINTERFACE),
3030                             "Could not get IDispatchEx interface: %08x\n", hres);
3031         if(SUCCEEDED(hres)) {
3032             test_disp((IUnknown*)filters, &IID_IHTMLFiltersCollection, "[object]");
3033             IDispatchEx_Release(dispex);
3034         }
3035
3036         IHTMLFiltersCollection_Release(filters);
3037     }
3038
3039     IHTMLElement_Release(elem);
3040 }
3041
3042 #define test_elem_set_class(u,c) _test_elem_set_class(__LINE__,u,c)
3043 static void _test_elem_set_class(unsigned line, IUnknown *unk, const char *class)
3044 {
3045     IHTMLElement *elem = _get_elem_iface(line, unk);
3046     BSTR tmp;
3047     HRESULT hres;
3048
3049     tmp = class ? a2bstr(class) : NULL;
3050     hres = IHTMLElement_put_className(elem, tmp);
3051     IHTMLElement_Release(elem);
3052     ok_(__FILE__,line) (hres == S_OK, "put_className failed: %08x\n", hres);
3053     SysFreeString(tmp);
3054
3055     _test_elem_class(line, unk, class);
3056 }
3057
3058 #define test_elem_id(e,i) _test_elem_id(__LINE__,e,i)
3059 static void _test_elem_id(unsigned line, IUnknown *unk, const char *exid)
3060 {
3061     IHTMLElement *elem = _get_elem_iface(line, unk);
3062     BSTR id = (void*)0xdeadbeef;
3063     HRESULT hres;
3064
3065     hres = IHTMLElement_get_id(elem, &id);
3066     IHTMLElement_Release(elem);
3067     ok_(__FILE__,line) (hres == S_OK, "get_id failed: %08x\n", hres);
3068
3069     if(exid)
3070         ok_(__FILE__,line) (!strcmp_wa(id, exid), "unexpected id %s\n", wine_dbgstr_w(id));
3071     else
3072         ok_(__FILE__,line) (!id, "id=%s\n", wine_dbgstr_w(id));
3073
3074     SysFreeString(id);
3075 }
3076
3077 #define test_elem_put_id(u,i) _test_elem_put_id(__LINE__,u,i)
3078 static void _test_elem_put_id(unsigned line, IUnknown *unk, const char *new_id)
3079 {
3080     IHTMLElement *elem = _get_elem_iface(line, unk);
3081     BSTR tmp = a2bstr(new_id);
3082     HRESULT hres;
3083
3084     hres = IHTMLElement_put_id(elem, tmp);
3085     IHTMLElement_Release(elem);
3086     SysFreeString(tmp);
3087     ok_(__FILE__,line) (hres == S_OK, "put_id failed: %08x\n", hres);
3088
3089     _test_elem_id(line, unk, new_id);
3090 }
3091
3092 #define test_elem_title(u,t) _test_elem_title(__LINE__,u,t)
3093 static void _test_elem_title(unsigned line, IUnknown *unk, const char *extitle)
3094 {
3095     IHTMLElement *elem = _get_elem_iface(line, unk);
3096     BSTR title;
3097     HRESULT hres;
3098
3099     hres = IHTMLElement_get_title(elem, &title);
3100     IHTMLElement_Release(elem);
3101     ok_(__FILE__,line) (hres == S_OK, "get_title failed: %08x\n", hres);
3102     if(extitle)
3103         ok_(__FILE__,line) (!strcmp_wa(title, extitle), "unexpected title %s\n", wine_dbgstr_w(title));
3104     else
3105         ok_(__FILE__,line) (!title, "title=%s, expected NULL\n", wine_dbgstr_w(title));
3106
3107     SysFreeString(title);
3108 }
3109
3110 #define test_elem_set_title(u,t) _test_elem_set_title(__LINE__,u,t)
3111 static void _test_elem_set_title(unsigned line, IUnknown *unk, const char *title)
3112 {
3113     IHTMLElement *elem = _get_elem_iface(line, unk);
3114     BSTR tmp;
3115     HRESULT hres;
3116
3117     tmp = a2bstr(title);
3118     hres = IHTMLElement_put_title(elem, tmp);
3119     ok_(__FILE__,line) (hres == S_OK, "get_title failed: %08x\n", hres);
3120
3121     IHTMLElement_Release(elem);
3122     SysFreeString(tmp);
3123 }
3124
3125 #define test_node_get_value_str(u,e) _test_node_get_value_str(__LINE__,u,e)
3126 static void _test_node_get_value_str(unsigned line, IUnknown *unk, const char *exval)
3127 {
3128     IHTMLDOMNode *node = _get_node_iface(line, unk);
3129     VARIANT var;
3130     HRESULT hres;
3131
3132     hres = IHTMLDOMNode_get_nodeValue(node, &var);
3133     IHTMLDOMNode_Release(node);
3134     ok_(__FILE__,line) (hres == S_OK, "get_nodeValue failed: %08x, expected VT_BSTR\n", hres);
3135
3136     if(exval) {
3137         ok_(__FILE__,line) (V_VT(&var) == VT_BSTR, "vt=%d\n", V_VT(&var));
3138         ok_(__FILE__,line) (!strcmp_wa(V_BSTR(&var), exval), "unexpected value %s\n", wine_dbgstr_w(V_BSTR(&var)));
3139     }else {
3140         ok_(__FILE__,line) (V_VT(&var) == VT_NULL, "vt=%d, expected VT_NULL\n", V_VT(&var));
3141     }
3142
3143     VariantClear(&var);
3144 }
3145
3146 #define test_node_put_value_str(u,v) _test_node_put_value_str(__LINE__,u,v)
3147 static void _test_node_put_value_str(unsigned line, IUnknown *unk, const char *val)
3148 {
3149     IHTMLDOMNode *node = _get_node_iface(line, unk);
3150     VARIANT var;
3151     HRESULT hres;
3152
3153     V_VT(&var) = VT_BSTR;
3154     V_BSTR(&var) = a2bstr(val);
3155
3156     hres = IHTMLDOMNode_put_nodeValue(node, var);
3157     ok_(__FILE__,line) (hres == S_OK, "get_nodeValue failed: %08x, expected VT_BSTR\n", hres);
3158     IHTMLDOMNode_Release(node);
3159     VariantClear(&var);
3160 }
3161
3162 #define test_elem_client_size(u) _test_elem_client_size(__LINE__,u)
3163 static void _test_elem_client_size(unsigned line, IUnknown *unk)
3164 {
3165     IHTMLElement2 *elem = _get_elem2_iface(line, unk);
3166     LONG l;
3167     HRESULT hres;
3168
3169     hres = IHTMLElement2_get_clientWidth(elem, &l);
3170     ok_(__FILE__,line) (hres == S_OK, "get_clientWidth failed: %08x\n", hres);
3171     hres = IHTMLElement2_get_clientHeight(elem, &l);
3172     ok_(__FILE__,line) (hres == S_OK, "get_clientHeight failed: %08x\n", hres);
3173
3174     IHTMLElement2_Release(elem);
3175 }
3176
3177 #define test_elem_client_rect(u) _test_elem_client_rect(__LINE__,u)
3178 static void _test_elem_client_rect(unsigned line, IUnknown *unk)
3179 {
3180     IHTMLElement2 *elem = _get_elem2_iface(line, unk);
3181     LONG l;
3182     HRESULT hres;
3183
3184     hres = IHTMLElement2_get_clientLeft(elem, &l);
3185     ok_(__FILE__,line) (hres == S_OK, "get_clientLeft failed: %08x\n", hres);
3186     ok_(__FILE__,line) (!l, "clientLeft = %d\n", l);
3187
3188     hres = IHTMLElement2_get_clientTop(elem, &l);
3189     ok_(__FILE__,line) (hres == S_OK, "get_clientTop failed: %08x\n", hres);
3190     ok_(__FILE__,line) (!l, "clientTop = %d\n", l);
3191
3192     IHTMLElement2_Release(elem);
3193 }
3194
3195 #define test_form_length(e,l) _test_form_length(__LINE__,e,l)
3196 static void _test_form_length(unsigned line, IUnknown *unk, LONG exlen)
3197 {
3198     IHTMLFormElement *form = _get_form_iface(line, unk);
3199     LONG len = 0xdeadbeef;
3200     HRESULT hres;
3201
3202     hres = IHTMLFormElement_get_length(form, &len);
3203     ok_(__FILE__,line)(hres == S_OK, "get_length failed: %08x\n", hres);
3204     ok_(__FILE__,line)(len == exlen, "length=%d, expected %d\n", len, exlen);
3205
3206     IHTMLFormElement_Release(form);
3207 }
3208
3209 #define test_form_action(f,a) _test_form_action(__LINE__,f,a)
3210 static void _test_form_action(unsigned line, IUnknown *unk, const char *ex)
3211 {
3212     IHTMLFormElement *form = _get_form_iface(line, unk);
3213     BSTR action = (void*)0xdeadbeef;
3214     HRESULT hres;
3215
3216     hres = IHTMLFormElement_get_action(form, &action);
3217     ok_(__FILE__,line)(hres == S_OK, "get_action failed: %08x\n", hres);
3218     if(ex)
3219         ok_(__FILE__,line)(!strcmp_wa(action, ex), "action=%s, expected %s\n", wine_dbgstr_w(action), ex);
3220     else
3221         ok_(__FILE__,line)(!action, "action=%p\n", action);
3222
3223     SysFreeString(action);
3224     IHTMLFormElement_Release(form);
3225 }
3226
3227 #define test_form_put_action(f,a) _test_form_put_action(__LINE__,f,a)
3228 static void _test_form_put_action(unsigned line, IUnknown *unk, const char *action)
3229 {
3230     IHTMLFormElement *form = _get_form_iface(line, unk);
3231     BSTR tmp = a2bstr(action);
3232     HRESULT hres;
3233
3234     hres = IHTMLFormElement_put_action(form, tmp);
3235     ok_(__FILE__,line)(hres == S_OK, "put_action failed: %08x\n", hres);
3236     SysFreeString(tmp);
3237     IHTMLFormElement_Release(form);
3238
3239     _test_form_action(line, unk, action);
3240 }
3241
3242 #define test_form_method(f,a) _test_form_method(__LINE__,f,a)
3243 static void _test_form_method(unsigned line, IUnknown *unk, const char *ex)
3244 {
3245     IHTMLFormElement *form = _get_form_iface(line, unk);
3246     BSTR method = (void*)0xdeadbeef;
3247     HRESULT hres;
3248
3249     hres = IHTMLFormElement_get_method(form, &method);
3250     ok_(__FILE__,line)(hres == S_OK, "get_method failed: %08x\n", hres);
3251     if(ex)
3252         ok_(__FILE__,line)(!strcmp_wa(method, ex), "method=%s, expected %s\n", wine_dbgstr_w(method), ex);
3253     else
3254         ok_(__FILE__,line)(!method, "method=%p\n", method);
3255
3256     SysFreeString(method);
3257     IHTMLFormElement_Release(form);
3258 }
3259
3260 #define test_form_put_method(f,r,a) _test_form_put_method(__LINE__,f,r,a)
3261 static void _test_form_put_method(unsigned line, IUnknown *unk, HRESULT exp_hres, const char *method)
3262 {
3263     IHTMLFormElement *form = _get_form_iface(line, unk);
3264     BSTR tmp = a2bstr(method);
3265     HRESULT hres;
3266
3267     hres = IHTMLFormElement_put_method(form, tmp);
3268     ok_(__FILE__,line)(hres == exp_hres, "put_method returned: %08x, expected %08x\n", hres, exp_hres);
3269     SysFreeString(tmp);
3270     IHTMLFormElement_Release(form);
3271
3272     if(exp_hres == S_OK)
3273         _test_form_method(line, unk, method);
3274 }
3275
3276 #define test_form_name(f,a) _test_form_name(__LINE__,f,a)
3277 static void _test_form_name(unsigned line, IUnknown *unk, const char *ex)
3278 {
3279     IHTMLFormElement *form = _get_form_iface(line, unk);
3280     BSTR name = (void*)0xdeadbeef;
3281     HRESULT hres;
3282
3283     hres = IHTMLFormElement_get_name(form, &name);
3284     ok_(__FILE__,line)(hres == S_OK, "get_name failed: %08x\n", hres);
3285     if(ex)
3286         ok_(__FILE__,line)(!strcmp_wa(name, ex), "name=%s, expected %s\n", wine_dbgstr_w(name), ex);
3287     else
3288         ok_(__FILE__,line)(!name, "name=%p\n", name);
3289
3290     SysFreeString(name);
3291     IHTMLFormElement_Release(form);
3292 }
3293
3294 #define test_form_put_name(f,a) _test_form_put_name(__LINE__,f,a)
3295 static void _test_form_put_name(unsigned line, IUnknown *unk, const char *name)
3296 {
3297     IHTMLFormElement *form = _get_form_iface(line, unk);
3298     BSTR tmp = a2bstr(name);
3299     HRESULT hres;
3300
3301     hres = IHTMLFormElement_put_name(form, tmp);
3302     ok_(__FILE__,line)(hres == S_OK, "put_name failed: %08x\n", hres);
3303     SysFreeString(tmp);
3304     IHTMLFormElement_Release(form);
3305
3306     _test_form_name(line, unk, name);
3307 }
3308
3309 #define test_form_encoding(f,a) _test_form_encoding(__LINE__,f,a)
3310 static void _test_form_encoding(unsigned line, IUnknown *unk, const char *ex)
3311 {
3312     IHTMLFormElement *form = _get_form_iface(line, unk);
3313     BSTR encoding = (void*)0xdeadbeef;
3314     HRESULT hres;
3315
3316     hres = IHTMLFormElement_get_encoding(form, &encoding);
3317     ok_(__FILE__,line)(hres == S_OK, "get_encoding failed: %08x\n", hres);
3318     if(ex)
3319         ok_(__FILE__,line)(!strcmp_wa(encoding, ex), "encoding=%s, expected %s\n", wine_dbgstr_w(encoding), ex);
3320     else
3321         ok_(__FILE__,line)(!encoding, "encoding=%p\n", encoding);
3322
3323     SysFreeString(encoding);
3324     IHTMLFormElement_Release(form);
3325 }
3326
3327 #define test_form_put_encoding(f,r,a) _test_form_put_encoding(__LINE__,f,r,a)
3328 static void _test_form_put_encoding(unsigned line, IUnknown *unk, HRESULT exp_hres, const char *encoding)
3329 {
3330     IHTMLFormElement *form = _get_form_iface(line, unk);
3331     BSTR tmp = a2bstr(encoding);
3332     HRESULT hres;
3333
3334     hres = IHTMLFormElement_put_encoding(form, tmp);
3335     ok_(__FILE__,line)(hres == exp_hres, "put_encoding returned: %08x, expected %08x\n", hres, exp_hres);
3336     SysFreeString(tmp);
3337     IHTMLFormElement_Release(form);
3338
3339     if(exp_hres == S_OK)
3340         _test_form_encoding(line, unk, encoding);
3341 }
3342
3343 #define test_form_elements(a) _test_form_elements(__LINE__,a)
3344 static void _test_form_elements(unsigned line, IUnknown *unk)
3345 {
3346     IHTMLFormElement *form = _get_form_iface(line, unk);
3347     IDispatch *disp;
3348     HRESULT hres;
3349
3350     disp = NULL;
3351     hres = IHTMLFormElement_get_elements(form, &disp);
3352     ok_(__FILE__,line)(hres == S_OK, "get_elements failed: %08x\n", hres);
3353     ok_(__FILE__,line)(disp != NULL, "disp = NULL\n");
3354     ok_(__FILE__,line)(iface_cmp((IUnknown*)form, (IUnknown*)disp), "disp != form\n");
3355
3356     IDispatch_Release(disp);
3357     IHTMLFormElement_Release(form);
3358 }
3359
3360 #define test_meta_name(a,b) _test_meta_name(__LINE__,a,b)
3361 static void _test_meta_name(unsigned line, IUnknown *unk, const char *exname)
3362 {
3363     IHTMLMetaElement *meta;
3364     BSTR name = NULL;
3365     HRESULT hres;
3366
3367
3368     meta = _get_metaelem_iface(line, unk);
3369     hres = IHTMLMetaElement_get_name(meta, &name);
3370     ok_(__FILE__,line)(hres == S_OK, "get_name failed: %08x\n", hres);
3371     ok_(__FILE__,line)(!strcmp_wa(name, exname), "name = %s, expected %s\n", wine_dbgstr_w(name), exname);
3372     SysFreeString(name);
3373     IHTMLMetaElement_Release(meta);
3374 }
3375
3376 #define test_meta_content(a,b) _test_meta_content(__LINE__,a,b)
3377 static void _test_meta_content(unsigned line, IUnknown *unk, const char *excontent)
3378 {
3379     IHTMLMetaElement *meta;
3380     BSTR content = NULL;
3381     HRESULT hres;
3382
3383
3384     meta = _get_metaelem_iface(line, unk);
3385     hres = IHTMLMetaElement_get_content(meta, &content);
3386     ok_(__FILE__,line)(hres == S_OK, "get_content failed: %08x\n", hres);
3387     ok_(__FILE__,line)(!strcmp_wa(content, excontent), "content = %s, expected %s\n", wine_dbgstr_w(content), excontent);
3388     SysFreeString(content);
3389     IHTMLMetaElement_Release(meta);
3390 }
3391
3392 #define test_meta_httpequiv(a,b) _test_meta_httpequiv(__LINE__,a,b)
3393 static void _test_meta_httpequiv(unsigned line, IUnknown *unk, const char *exval)
3394 {
3395     IHTMLMetaElement *meta;
3396     BSTR val = NULL;
3397     HRESULT hres;
3398
3399
3400     meta = _get_metaelem_iface(line, unk);
3401     hres = IHTMLMetaElement_get_httpEquiv(meta, &val);
3402     ok_(__FILE__,line)(hres == S_OK, "get_httpEquiv failed: %08x\n", hres);
3403     ok_(__FILE__,line)(!strcmp_wa(val, exval), "httpEquiv = %s, expected %s\n", wine_dbgstr_w(val), exval);
3404     SysFreeString(val);
3405     IHTMLMetaElement_Release(meta);
3406 }
3407
3408 #define get_elem_doc(e) _get_elem_doc(__LINE__,e)
3409 static IHTMLDocument2 *_get_elem_doc(unsigned line, IUnknown *unk)
3410 {
3411     IHTMLElement *elem = _get_elem_iface(line, unk);
3412     IHTMLDocument2 *doc;
3413     IDispatch *disp;
3414     HRESULT hres;
3415
3416     disp = NULL;
3417     hres = IHTMLElement_get_document(elem, &disp);
3418     ok(hres == S_OK, "get_document failed: %08x\n", hres);
3419     ok(disp != NULL, "disp == NULL\n");
3420
3421     hres = IDispatch_QueryInterface(disp, &IID_IHTMLDocument2, (void**)&doc);
3422     IDispatch_Release(disp);
3423     ok(hres == S_OK, "Could not get IHTMLDocument2 iface: %08x\n", hres);
3424
3425     return doc;
3426 }
3427
3428 #define get_elem_attr_node(a,b,c) _get_elem_attr_node(__LINE__,a,b,c)
3429 static IHTMLDOMAttribute *_get_elem_attr_node(unsigned line, IUnknown *unk, const char *attr_name, BOOL expect_success)
3430 {
3431     IHTMLElement4 *elem = _get_elem4_iface(line, unk);
3432     BSTR str = a2bstr(attr_name);
3433     IHTMLDOMAttribute *attr;
3434     HRESULT hres;
3435
3436     attr = (void*)0xdeadbeef;
3437     hres = IHTMLElement4_getAttributeNode(elem, str, &attr);
3438     ok_(__FILE__,line)(hres == S_OK, "getAttributeNode failed: %08x\n", hres);
3439     if(expect_success)
3440         ok_(__FILE__,line)(attr != NULL, "attr = NULL\n");
3441     else
3442         ok_(__FILE__,line)(!attr, "attr = %p\n", attr);
3443
3444     IHTMLElement4_Release(elem);
3445     SysFreeString(str);
3446     return attr;
3447 }
3448
3449 #define get_attr_node_value(a,b,c) _get_attr_node_value(__LINE__,a,b,c)
3450 static void _get_attr_node_value(unsigned line, IHTMLDOMAttribute *attr, VARIANT *v, VARTYPE vt)
3451 {
3452     HRESULT hres;
3453
3454     hres = IHTMLDOMAttribute_get_nodeValue(attr, v);
3455     ok_(__FILE__,line) (hres == S_OK, "get_nodeValue failed: %08x, expected VT_BSTR\n", hres);
3456     ok_(__FILE__,line) (V_VT(v) == vt, "vt=%d, expected %d\n", V_VT(v), vt);
3457 }
3458
3459 #define get_window_doc(e) _get_window_doc(__LINE__,e)
3460 static IHTMLDocument2 *_get_window_doc(unsigned line, IHTMLWindow2 *window)
3461 {
3462     IHTMLDocument2 *doc;
3463     HRESULT hres;
3464
3465     doc = NULL;
3466     hres = IHTMLWindow2_get_document(window, &doc);
3467     ok(hres == S_OK, "get_document failed: %08x\n", hres);
3468     ok(doc != NULL, "disp == NULL\n");
3469
3470     return doc;
3471 }
3472
3473 #define doc_get_body(d) _doc_get_body(__LINE__,d)
3474 static IHTMLElement *_doc_get_body(unsigned line, IHTMLDocument2 *doc)
3475 {
3476     IHTMLElement *elem;
3477     HRESULT hres;
3478
3479     hres = IHTMLDocument2_get_body(doc, &elem);
3480     ok_(__FILE__,line)(hres == S_OK, "get_body failed: %08x\n", hres);
3481     ok_(__FILE__,line)(elem != NULL, "body == NULL\n");
3482
3483     return elem;
3484 }
3485
3486 #define test_create_elem(d,t) _test_create_elem(__LINE__,d,t)
3487 static IHTMLElement *_test_create_elem(unsigned line, IHTMLDocument2 *doc, const char *tag)
3488 {
3489     IHTMLElement *elem = NULL;
3490     BSTR tmp;
3491     HRESULT hres;
3492
3493     tmp = a2bstr(tag);
3494     hres = IHTMLDocument2_createElement(doc, tmp, &elem);
3495     ok_(__FILE__,line) (hres == S_OK, "createElement failed: %08x\n", hres);
3496     ok_(__FILE__,line) (elem != NULL, "elem == NULL\n");
3497     SysFreeString(tmp);
3498
3499     return elem;
3500 }
3501
3502 #define test_create_text(d,t) _test_create_text(__LINE__,d,t)
3503 static IHTMLDOMNode *_test_create_text(unsigned line, IHTMLDocument2 *doc, const char *text)
3504 {
3505     IHTMLDocument3 *doc3;
3506     IHTMLDOMNode *node = NULL;
3507     BSTR tmp;
3508     HRESULT hres;
3509
3510     hres = IHTMLDocument2_QueryInterface(doc, &IID_IHTMLDocument3, (void**)&doc3);
3511     ok_(__FILE__,line) (hres == S_OK, "Could not get IHTMLDocument3: %08x\n", hres);
3512
3513     tmp = a2bstr(text);
3514     hres = IHTMLDocument3_createTextNode(doc3, tmp, &node);
3515     IHTMLDocument3_Release(doc3);
3516     SysFreeString(tmp);
3517     ok_(__FILE__,line) (hres == S_OK, "createElement failed: %08x\n", hres);
3518     ok_(__FILE__,line) (node != NULL, "node == NULL\n");
3519
3520     return node;
3521 }
3522
3523 #define test_node_append_child(n,c) _test_node_append_child(__LINE__,n,c)
3524 static IHTMLDOMNode *_test_node_append_child(unsigned line, IUnknown *node_unk, IUnknown *child_unk)
3525 {
3526     IHTMLDOMNode *node = _get_node_iface(line, node_unk);
3527     IHTMLDOMNode *child = _get_node_iface(line, child_unk);
3528     IHTMLDOMNode *new_child = NULL;
3529     HRESULT hres;
3530
3531     hres = IHTMLDOMNode_appendChild(node, child, &new_child);
3532     ok_(__FILE__,line) (hres == S_OK, "appendChild failed: %08x\n", hres);
3533     ok_(__FILE__,line) (new_child != NULL, "new_child == NULL\n");
3534     /* TODO  ok_(__FILE__,line) (new_child != child, "new_child == child\n"); */
3535
3536     IHTMLDOMNode_Release(node);
3537     IHTMLDOMNode_Release(child);
3538
3539     return new_child;
3540 }
3541
3542 #define test_node_insertbefore(n,c,v) _test_node_insertbefore(__LINE__,n,c,v)
3543 static IHTMLDOMNode *_test_node_insertbefore(unsigned line, IUnknown *node_unk, IHTMLDOMNode *child, VARIANT *var)
3544 {
3545     IHTMLDOMNode *node = _get_node_iface(line, node_unk);
3546     IHTMLDOMNode *new_child = NULL;
3547     HRESULT hres;
3548
3549     hres = IHTMLDOMNode_insertBefore(node, child, *var, &new_child);
3550     ok_(__FILE__,line) (hres == S_OK, "insertBefore failed: %08x\n", hres);
3551     ok_(__FILE__,line) (new_child != NULL, "new_child == NULL\n");
3552     /* TODO  ok_(__FILE__,line) (new_child != child, "new_child == child\n"); */
3553
3554     IHTMLDOMNode_Release(node);
3555
3556     return new_child;
3557 }
3558
3559 #define test_node_remove_child(n,c) _test_node_remove_child(__LINE__,n,c)
3560 static void _test_node_remove_child(unsigned line, IUnknown *unk, IHTMLDOMNode *child)
3561 {
3562     IHTMLDOMNode *node = _get_node_iface(line, unk);
3563     IHTMLDOMNode *new_node = NULL;
3564     HRESULT hres;
3565
3566     hres = IHTMLDOMNode_removeChild(node, child, &new_node);
3567     ok_(__FILE__,line) (hres == S_OK, "removeChild failed: %08x\n", hres);
3568     ok_(__FILE__,line) (new_node != NULL, "new_node == NULL\n");
3569     /* TODO ok_(__FILE__,line) (new_node != child, "new_node == child\n"); */
3570
3571     IHTMLDOMNode_Release(node);
3572     IHTMLDOMNode_Release(new_node);
3573 }
3574
3575 #define test_doc_title(d,t) _test_doc_title(__LINE__,d,t)
3576 static void _test_doc_title(unsigned line, IHTMLDocument2 *doc, const char *extitle)
3577 {
3578     BSTR title = NULL;
3579     HRESULT hres;
3580
3581     hres = IHTMLDocument2_get_title(doc, &title);
3582     ok_(__FILE__,line) (hres == S_OK, "get_title failed: %08x\n", hres);
3583     ok_(__FILE__,line) (!strcmp_wa(title, extitle), "unexpected title %s\n", wine_dbgstr_w(title));
3584     SysFreeString(title);
3585 }
3586
3587 #define test_doc_set_title(d,t) _test_doc_set_title(__LINE__,d,t)
3588 static void _test_doc_set_title(unsigned line, IHTMLDocument2 *doc, const char *title)
3589 {
3590     BSTR tmp;
3591     HRESULT hres;
3592
3593     tmp = a2bstr(title);
3594     hres = IHTMLDocument2_put_title(doc, tmp);
3595     ok_(__FILE__,line) (hres == S_OK, "get_title failed: %08x\n", hres);
3596     SysFreeString(tmp);
3597 }
3598
3599 static void test_elem_bounding_client_rect(IUnknown *unk)
3600 {
3601     IHTMLRect *rect, *rect2;
3602     IHTMLElement2 *elem2;
3603     LONG l;
3604     HRESULT hres;
3605
3606     elem2 = get_elem2_iface(unk);
3607     hres = IHTMLElement2_getBoundingClientRect(elem2, &rect);
3608     ok(hres == S_OK, "getBoundingClientRect failed: %08x\n", hres);
3609     hres = IHTMLElement2_getBoundingClientRect(elem2, &rect2);
3610     IHTMLElement2_Release(elem2);
3611     ok(hres == S_OK, "getBoundingClientRect failed: %08x\n", hres);
3612     ok(rect != NULL, "rect == NULL\n");
3613     ok(rect != rect2, "rect == rect2\n");
3614     IHTMLRect_Release(rect2);
3615
3616     test_disp((IUnknown*)rect, &IID_IHTMLRect, "[object]");
3617
3618     l = 0xdeadbeef;
3619     hres = IHTMLRect_get_top(rect, &l);
3620     ok(hres == S_OK, "get_top failed: %08x\n", hres);
3621     ok(l != 0xdeadbeef, "l = 0xdeadbeef\n");
3622
3623     l = 0xdeadbeef;
3624     hres = IHTMLRect_get_left(rect, &l);
3625     ok(hres == S_OK, "get_left failed: %08x\n", hres);
3626     ok(l != 0xdeadbeef, "l = 0xdeadbeef\n");
3627
3628     l = 0xdeadbeef;
3629     hres = IHTMLRect_get_bottom(rect, &l);
3630     ok(hres == S_OK, "get_bottom failed: %08x\n", hres);
3631     ok(l != 0xdeadbeef, "l = 0xdeadbeef\n");
3632
3633     l = 0xdeadbeef;
3634     hres = IHTMLRect_get_right(rect, &l);
3635     ok(hres == S_OK, "get_right failed: %08x\n", hres);
3636     ok(l != 0xdeadbeef, "l = 0xdeadbeef\n");
3637
3638     IHTMLRect_Release(rect);
3639 }
3640
3641 static void test_elem_col_item(IHTMLElementCollection *col, const char *n,
3642         const elem_type_t *elem_types, LONG len)
3643 {
3644     IDispatch *disp;
3645     VARIANT name, index;
3646     DWORD i;
3647     HRESULT hres;
3648
3649     V_VT(&index) = VT_EMPTY;
3650     V_VT(&name) = VT_BSTR;
3651     V_BSTR(&name) = a2bstr(n);
3652
3653     hres = IHTMLElementCollection_item(col, name, index, &disp);
3654     ok(hres == S_OK, "item failed: %08x\n", hres);
3655
3656     test_elem_collection((IUnknown*)disp, elem_types, len);
3657     IDispatch_Release(disp);
3658
3659     V_VT(&index) = VT_I4;
3660
3661     for(i=0; i<len; i++) {
3662         V_I4(&index) = i;
3663         disp = (void*)0xdeadbeef;
3664         hres = IHTMLElementCollection_item(col, name, index, &disp);
3665         ok(hres == S_OK, "item failed: %08x\n", hres);
3666         ok(disp != NULL, "disp == NULL\n");
3667         if(FAILED(hres) || !disp)
3668             continue;
3669
3670         test_elem_type((IUnknown*)disp, elem_types[i]);
3671
3672         IDispatch_Release(disp);
3673     }
3674
3675     V_I4(&index) = len;
3676     disp = (void*)0xdeadbeef;
3677     hres = IHTMLElementCollection_item(col, name, index, &disp);
3678     ok(hres == S_OK, "item failed: %08x\n", hres);
3679     ok(disp == NULL, "disp != NULL\n");
3680
3681     V_I4(&index) = -1;
3682     disp = (void*)0xdeadbeef;
3683     hres = IHTMLElementCollection_item(col, name, index, &disp);
3684     ok(hres == E_INVALIDARG, "item failed: %08x, expected E_INVALIDARG\n", hres);
3685     ok(disp == NULL, "disp != NULL\n");
3686
3687     SysFreeString(V_BSTR(&name));
3688 }
3689
3690 static IHTMLElement *get_elem_by_id(IHTMLDocument2 *doc, const char *id, BOOL expect_success)
3691 {
3692     IHTMLElementCollection *col;
3693     IHTMLElement *elem;
3694     IDispatch *disp = (void*)0xdeadbeef;
3695     VARIANT name, index;
3696     HRESULT hres;
3697
3698     hres = IHTMLDocument2_get_all(doc, &col);
3699     ok(hres == S_OK, "get_all failed: %08x\n", hres);
3700     ok(col != NULL, "col == NULL\n");
3701     if(FAILED(hres) || !col)
3702         return NULL;
3703
3704     V_VT(&index) = VT_EMPTY;
3705     V_VT(&name) = VT_BSTR;
3706     V_BSTR(&name) = a2bstr(id);
3707
3708     hres = IHTMLElementCollection_item(col, name, index, &disp);
3709     IHTMLElementCollection_Release(col);
3710     SysFreeString(V_BSTR(&name));
3711     ok(hres == S_OK, "item failed: %08x\n", hres);
3712     if(!expect_success) {
3713         ok(disp == NULL, "disp != NULL\n");
3714         return NULL;
3715     }
3716
3717     ok(disp != NULL, "disp == NULL\n");
3718     if(!disp)
3719         return NULL;
3720
3721     elem = get_elem_iface((IUnknown*)disp);
3722     IDispatch_Release(disp);
3723
3724     return elem;
3725 }
3726
3727 static IHTMLElement *get_doc_elem_by_id(IHTMLDocument2 *doc, const char *id)
3728 {
3729     IHTMLDocument3 *doc3;
3730     IHTMLElement *elem;
3731     BSTR tmp;
3732     HRESULT hres;
3733
3734     hres = IHTMLDocument2_QueryInterface(doc, &IID_IHTMLDocument3, (void**)&doc3);
3735     ok(hres == S_OK, "Could not get IHTMLDocument3 iface: %08x\n", hres);
3736
3737     tmp = a2bstr(id);
3738     hres = IHTMLDocument3_getElementById(doc3, tmp, &elem);
3739     SysFreeString(tmp);
3740     ok(hres == S_OK, "getElementById(%s) failed: %08x\n", id, hres);
3741
3742     IHTMLDocument3_Release(doc3);
3743
3744     return elem;
3745 }
3746
3747 static void test_select_elem(IHTMLSelectElement *select)
3748 {
3749     IDispatch *disp, *disp2;
3750     VARIANT name, index;
3751     HRESULT hres;
3752
3753     test_select_type(select, "select-one");
3754     test_select_length(select, 2);
3755     test_select_selidx(select, 0);
3756     test_select_put_selidx(select, 1);
3757
3758     test_select_set_value(select, "val1");
3759     test_select_value(select, "val1");
3760
3761     test_select_get_disabled(select, VARIANT_FALSE);
3762     test_select_set_disabled(select, VARIANT_TRUE);
3763     test_select_set_disabled(select, VARIANT_FALSE);
3764
3765     disp = NULL;
3766     hres = IHTMLSelectElement_get_options(select, &disp);
3767     ok(hres == S_OK, "get_options failed: %08x\n", hres);
3768     ok(disp != NULL, "options == NULL\n");
3769     ok(iface_cmp((IUnknown*)disp, (IUnknown*)select), "disp != select\n");
3770     IDispatch_Release(disp);
3771
3772     V_VT(&index) = VT_EMPTY;
3773     V_VT(&name) = VT_I4;
3774     V_I4(&name) = -1;
3775     disp = (void*)0xdeadbeef;
3776     hres = IHTMLSelectElement_item(select, name, index, &disp);
3777     ok(hres == E_INVALIDARG, "item failed: %08x, expected E_INVALIDARG\n", hres);
3778     ok(!disp, "disp = %p\n", disp);
3779
3780     V_I4(&name) = 2;
3781     disp = (void*)0xdeadbeef;
3782     hres = IHTMLSelectElement_item(select, name, index, &disp);
3783     ok(hres == S_OK, "item failed: %08x\n", hres);
3784     ok(!disp, "disp = %p\n", disp);
3785
3786     V_I4(&name) = 1;
3787     hres = IHTMLSelectElement_item(select, name, index, NULL);
3788     ok(hres == E_POINTER || broken(hres == E_INVALIDARG), "item failed: %08x, expected E_POINTER\n", hres);
3789
3790     disp = NULL;
3791     hres = IHTMLSelectElement_item(select, name, index, &disp);
3792     ok(hres == S_OK, "item failed: %08x\n", hres);
3793     ok(disp != NULL, "disp = NULL\n");
3794     test_disp((IUnknown*)disp, &DIID_DispHTMLOptionElement, NULL);
3795
3796     V_VT(&index) = VT_I4;
3797     V_I4(&index) = 1;
3798     disp2 = NULL;
3799     hres = IHTMLSelectElement_item(select, name, index, &disp2);
3800     ok(hres == S_OK, "item failed: %08x\n", hres);
3801     ok(disp2 != NULL, "disp = NULL\n");
3802     ok(iface_cmp((IUnknown*)disp, (IUnknown*)disp2), "disp != disp2\n");
3803     IDispatch_Release(disp2);
3804     IDispatch_Release(disp);
3805 }
3806
3807 static void test_form_item(IHTMLElement *elem)
3808 {
3809     IHTMLFormElement *form = get_form_iface((IUnknown*)elem);
3810     IDispatch *disp, *disp2;
3811     VARIANT name, index;
3812     HRESULT hres;
3813
3814     V_VT(&index) = VT_EMPTY;
3815     V_VT(&name) = VT_I4;
3816     V_I4(&name) = -1;
3817     disp = (void*)0xdeadbeef;
3818     hres = IHTMLFormElement_item(form, name, index, &disp);
3819     ok(hres == E_INVALIDARG, "item failed: %08x, expected E_INVALIDARG\n", hres);
3820     ok(!disp, "disp = %p\n", disp);
3821
3822     V_I4(&name) = 2;
3823     disp = (void*)0xdeadbeef;
3824     hres = IHTMLFormElement_item(form, name, index, &disp);
3825     ok(hres == S_OK, "item failed: %08x\n", hres);
3826     ok(!disp, "disp = %p\n", disp);
3827
3828     V_I4(&name) = 1;
3829     hres = IHTMLFormElement_item(form, name, index, NULL);
3830     ok(hres == E_INVALIDARG, "item failed: %08x, expected E_INVALIDARG\n", hres);
3831
3832     disp = NULL;
3833     hres = IHTMLFormElement_item(form, name, index, &disp);
3834     ok(hres == S_OK, "item failed: %08x\n", hres);
3835     ok(disp != NULL, "disp = NULL\n");
3836     test_disp((IUnknown*)disp, &DIID_DispHTMLInputElement, NULL);
3837
3838     V_VT(&index) = VT_I4;
3839     V_I4(&index) = 1;
3840     disp2 = NULL;
3841     hres = IHTMLFormElement_item(form, name, index, &disp2);
3842     ok(hres == S_OK, "item failed: %08x\n", hres);
3843     ok(disp2 != NULL, "disp = NULL\n");
3844     ok(iface_cmp((IUnknown*)disp, (IUnknown*)disp2), "disp != disp2\n");
3845     IDispatch_Release(disp2);
3846     IDispatch_Release(disp);
3847 }
3848
3849 static void test_create_option_elem(IHTMLDocument2 *doc)
3850 {
3851     IHTMLOptionElement *option;
3852
3853     option = create_option_elem(doc, "test text", "test value");
3854
3855     test_option_put_text(option, "new text");
3856     test_option_put_value(option, "new value");
3857     test_option_put_selected(option, VARIANT_TRUE);
3858     test_option_put_selected(option, VARIANT_FALSE);
3859
3860     IHTMLOptionElement_Release(option);
3861 }
3862
3863 static void test_create_img_elem(IHTMLDocument2 *doc)
3864 {
3865     IHTMLImgElement *img;
3866
3867     img = create_img_elem(doc, 10, 15);
3868
3869     if(img){
3870         test_img_put_width(img, 5);
3871         test_img_put_height(img, 20);
3872
3873         IHTMLImgElement_Release(img);
3874         img = NULL;
3875     }
3876
3877     img = create_img_elem(doc, -1, -1);
3878
3879     if(img){
3880         test_img_put_width(img, 5);
3881         test_img_put_height(img, 20);
3882
3883         IHTMLImgElement_Release(img);
3884     }
3885 }
3886
3887 static IHTMLTxtRange *test_create_body_range(IHTMLDocument2 *doc)
3888 {
3889     IHTMLBodyElement *body;
3890     IHTMLTxtRange *range;
3891     IHTMLElement *elem;
3892     HRESULT hres;
3893
3894     elem = doc_get_body(doc);
3895     hres = IHTMLElement_QueryInterface(elem, &IID_IHTMLBodyElement, (void**)&body);
3896     ok(hres == S_OK, "QueryInterface failed: %08x\n", hres);
3897     IHTMLElement_Release(elem);
3898
3899     hres = IHTMLBodyElement_createTextRange(body, &range);
3900     IHTMLBodyElement_Release(body);
3901     ok(hres == S_OK, "createTextRange failed: %08x\n", hres);
3902
3903     return range;
3904 }
3905
3906 static void test_txtrange(IHTMLDocument2 *doc)
3907 {
3908     IHTMLTxtRange *body_range, *range, *range2;
3909     IHTMLSelectionObject *selection;
3910     IDispatch *disp_range;
3911     HRESULT hres;
3912
3913     body_range = test_create_body_range(doc);
3914
3915     test_range_text(body_range, "test abc 123\r\nit's text");
3916
3917     hres = IHTMLTxtRange_duplicate(body_range, &range);
3918     ok(hres == S_OK, "duplicate failed: %08x\n", hres);
3919
3920     hres = IHTMLTxtRange_duplicate(body_range, &range2);
3921     ok(hres == S_OK, "duplicate failed: %08x\n", hres);
3922     test_range_isequal(range, range2, VARIANT_TRUE);
3923
3924     test_range_text(range, "test abc 123\r\nit's text");
3925     test_range_text(body_range, "test abc 123\r\nit's text");
3926
3927     test_range_collapse(range, TRUE);
3928     test_range_isequal(range, range2, VARIANT_FALSE);
3929     test_range_inrange(range, range2, VARIANT_FALSE);
3930     test_range_inrange(range2, range, VARIANT_TRUE);
3931     IHTMLTxtRange_Release(range2);
3932
3933     test_range_expand(range, wordW, VARIANT_TRUE, "test ");
3934     test_range_expand(range, wordW, VARIANT_FALSE, "test ");
3935     test_range_move(range, characterW, 2, 2);
3936     test_range_expand(range, wordW, VARIANT_TRUE, "test ");
3937
3938     test_range_collapse(range, FALSE);
3939     test_range_expand(range, wordW, VARIANT_TRUE, "abc ");
3940
3941     test_range_collapse(range, FALSE);
3942     test_range_expand(range, wordW, VARIANT_TRUE, "123");
3943     test_range_expand(range, wordW, VARIANT_FALSE, "123");
3944     test_range_move(range, characterW, 2, 2);
3945     test_range_expand(range, wordW, VARIANT_TRUE, "123");
3946     test_range_moveend(range, characterW, -5, -5);
3947     test_range_text(range, NULL);
3948     test_range_moveend(range, characterW, 3, 3);
3949     test_range_text(range, "c 1");
3950     test_range_expand(range, texteditW, VARIANT_TRUE, "test abc 123\r\nit's text");
3951     test_range_collapse(range, TRUE);
3952     test_range_move(range, characterW, 4, 4);
3953     test_range_moveend(range, characterW, 1, 1);
3954     test_range_text(range, " ");
3955     test_range_move(range, wordW, 1, 1);
3956     test_range_moveend(range, characterW, 2, 2);
3957     test_range_text(range, "ab");
3958
3959     IHTMLTxtRange_Release(range);
3960
3961     hres = IHTMLTxtRange_duplicate(body_range, &range);
3962     ok(hres == S_OK, "duplicate failed: %08x\n", hres);
3963
3964     test_range_text(range, "test abc 123\r\nit's text");
3965     test_range_move(range, characterW, 3, 3);
3966     test_range_moveend(range, characterW, 1, 1);
3967     test_range_text(range, "t");
3968     test_range_moveend(range, characterW, 3, 3);
3969     test_range_text(range, "t ab");
3970     test_range_moveend(range, characterW, -2, -2);
3971     test_range_text(range, "t ");
3972     test_range_move(range, characterW, 6, 6);
3973     test_range_moveend(range, characterW, 3, 3);
3974     test_range_text(range, "123");
3975     test_range_moveend(range, characterW, 2, 2);
3976     test_range_text(range, "123\r\ni");
3977
3978     IHTMLTxtRange_Release(range);
3979
3980     hres = IHTMLTxtRange_duplicate(body_range, &range);
3981     ok(hres == S_OK, "duplicate failed: %08x\n", hres);
3982
3983     test_range_move(range, wordW, 1, 1);
3984     test_range_moveend(range, characterW, 2, 2);
3985     test_range_text(range, "ab");
3986
3987     test_range_move(range, characterW, -2, -2);
3988     test_range_moveend(range, characterW, 2, 2);
3989     test_range_text(range, "t ");
3990
3991     test_range_move(range, wordW, 3, 3);
3992     test_range_move(range, wordW, -2, -2);
3993     test_range_moveend(range, characterW, 2, 2);
3994     test_range_text(range, "ab");
3995
3996     test_range_move(range, characterW, -6, -5);
3997     test_range_moveend(range, characterW, -1, 0);
3998     test_range_moveend(range, characterW, -6, 0);
3999     test_range_move(range, characterW, 2, 2);
4000     test_range_moveend(range, characterW, 2, 2);
4001     test_range_text(range, "st");
4002     test_range_moveend(range, characterW, -6, -4);
4003     test_range_moveend(range, characterW, 2, 2);
4004
4005     IHTMLTxtRange_Release(range);
4006
4007     hres = IHTMLTxtRange_duplicate(body_range, &range);
4008     ok(hres == S_OK, "duplicate failed: %08x\n", hres);
4009
4010     test_range_move(range, wordW, 2, 2);
4011     test_range_moveend(range, characterW, 2, 2);
4012     test_range_text(range, "12");
4013
4014     test_range_move(range, characterW, 15, 14);
4015     test_range_move(range, characterW, -2, -2);
4016     test_range_moveend(range, characterW, 3, 2);
4017     test_range_text(range, "t");
4018     test_range_moveend(range, characterW, -1, -1);
4019     test_range_text(range, "t");
4020     test_range_expand(range, wordW, VARIANT_TRUE, "text");
4021     test_range_move(range, characterW, -2, -2);
4022     test_range_moveend(range, characterW, 2, 2);
4023     test_range_text(range, "s ");
4024     test_range_move(range, characterW, 100, 7);
4025     test_range_move(range, wordW, 1, 0);
4026     test_range_move(range, characterW, -2, -2);
4027     test_range_moveend(range, characterW, 3, 2);
4028     test_range_text(range, "t");
4029
4030     IHTMLTxtRange_Release(range);
4031
4032     hres = IHTMLTxtRange_duplicate(body_range, &range);
4033     ok(hres == S_OK, "duplicate failed: %08x\n", hres);
4034
4035     test_range_collapse(range, TRUE);
4036     test_range_expand(range, wordW, VARIANT_TRUE, "test ");
4037     test_range_put_text(range, "word");
4038     test_range_text(body_range, "wordabc 123\r\nit's text");
4039     test_range_text(range, NULL);
4040     test_range_moveend(range, characterW, 3, 3);
4041     test_range_text(range, "abc");
4042     test_range_movestart(range, characterW, -2, -2);
4043     test_range_text(range, "rdabc");
4044     test_range_movestart(range, characterW, 3, 3);
4045     test_range_text(range, "bc");
4046     test_range_movestart(range, characterW, 4, 4);
4047     test_range_text(range, NULL);
4048     test_range_movestart(range, characterW, -3, -3);
4049     test_range_text(range, "c 1");
4050     test_range_movestart(range, characterW, -7, -6);
4051     test_range_text(range, "wordabc 1");
4052     test_range_movestart(range, characterW, 100, 22);
4053     test_range_text(range, NULL);
4054
4055     IHTMLTxtRange_Release(range);
4056     IHTMLTxtRange_Release(body_range);
4057
4058     hres = IHTMLDocument2_get_selection(doc, &selection);
4059     ok(hres == S_OK, "IHTMLDocument2_get_selection failed: %08x\n", hres);
4060
4061     hres = IHTMLSelectionObject_createRange(selection, &disp_range);
4062     ok(hres == S_OK, "IHTMLSelectionObject_createRange failed: %08x\n", hres);
4063     IHTMLSelectionObject_Release(selection);
4064
4065     hres = IDispatch_QueryInterface(disp_range, &IID_IHTMLTxtRange, (void **)&range);
4066     ok(hres == S_OK, "Could not get IID_IHTMLTxtRange interface: 0x%08x\n", hres);
4067     IDispatch_Release(disp_range);
4068
4069     test_range_text(range, NULL);
4070     test_range_moveend(range, characterW, 3, 3);
4071     test_range_text(range, "wor");
4072     test_range_parent(range, ET_BODY);
4073     test_range_expand(range, texteditW, VARIANT_TRUE, "wordabc 123\r\nit's text");
4074     test_range_expand(range, texteditW, VARIANT_TRUE, "wordabc 123\r\nit's text");
4075     test_range_move(range, characterW, 3, 3);
4076     test_range_expand(range, wordW, VARIANT_TRUE, "wordabc ");
4077     test_range_moveend(range, characterW, -4, -4);
4078     test_range_put_text(range, "abc def ");
4079     test_range_expand(range, texteditW, VARIANT_TRUE, "abc def abc 123\r\nit's text");
4080     test_range_move(range, wordW, 1, 1);
4081     test_range_movestart(range, characterW, -1, -1);
4082     test_range_text(range, " ");
4083     test_range_move(range, wordW, 1, 1);
4084     test_range_moveend(range, characterW, 3, 3);
4085     test_range_text(range, "def");
4086     test_range_put_text(range, "xyz");
4087     test_range_moveend(range, characterW, 1, 1);
4088     test_range_move(range, wordW, 1, 1);
4089     test_range_moveend(range, characterW, 2, 2);
4090     test_range_text(range, "ab");
4091
4092     IHTMLTxtRange_Release(range);
4093 }
4094
4095 static void test_txtrange2(IHTMLDocument2 *doc)
4096 {
4097     IHTMLTxtRange *range;
4098
4099     range = test_create_body_range(doc);
4100
4101     test_range_text(range, "abc\r\n\r\n123\r\n\r\n\r\ndef");
4102     test_range_move(range, characterW, 5, 5);
4103     test_range_moveend(range, characterW, 1, 1);
4104     test_range_text(range, "2");
4105     test_range_move(range, characterW, -3, -3);
4106     test_range_moveend(range, characterW, 3, 3);
4107     test_range_text(range, "c\r\n\r\n1");
4108     test_range_collapse(range, VARIANT_FALSE);
4109     test_range_moveend(range, characterW, 4, 4);
4110     test_range_text(range, "23");
4111     test_range_moveend(range, characterW, 1, 1);
4112     test_range_text(range, "23\r\n\r\n\r\nd");
4113     test_range_moveend(range, characterW, -1, -1);
4114     test_range_text(range, "23");
4115     test_range_moveend(range, characterW, -1, -1);
4116     test_range_text(range, "23");
4117     test_range_moveend(range, characterW, -2, -2);
4118     test_range_text(range, "2");
4119
4120     IHTMLTxtRange_Release(range);
4121 }
4122
4123 static void test_compatmode(IHTMLDocument2 *doc)
4124 {
4125     IHTMLDocument5 *doc5;
4126     BSTR mode;
4127     HRESULT hres;
4128
4129     hres = IHTMLDocument2_QueryInterface(doc, &IID_IHTMLDocument5, (void**)&doc5);
4130     ok(hres == S_OK, "Could not get IHTMLDocument5 interface: %08x\n", hres);
4131     if(FAILED(hres))
4132         return;
4133
4134     hres = IHTMLDocument5_get_compatMode(doc5, &mode);
4135     IHTMLDocument5_Release(doc5);
4136     ok(hres == S_OK, "get_compatMode failed: %08x\n", hres);
4137     ok(!strcmp_wa(mode, "BackCompat"), "compatMode=%s\n", wine_dbgstr_w(mode));
4138     SysFreeString(mode);
4139 }
4140
4141 static void test_location(IHTMLDocument2 *doc)
4142 {
4143     IHTMLLocation *location, *location2;
4144     IHTMLWindow2 *window;
4145     BSTR str;
4146     ULONG ref;
4147     HRESULT hres;
4148
4149     hres = IHTMLDocument2_get_location(doc, &location);
4150     ok(hres == S_OK, "get_location failed: %08x\n", hres);
4151
4152     hres = IHTMLDocument2_get_location(doc, &location2);
4153     ok(hres == S_OK, "get_location failed: %08x\n", hres);
4154
4155     ok(location == location2, "location != location2\n");
4156     IHTMLLocation_Release(location2);
4157
4158     hres = IHTMLDocument2_get_parentWindow(doc, &window);
4159     ok(hres == S_OK, "get_parentWindow failed: %08x\n", hres);
4160
4161     hres = IHTMLWindow2_get_location(window, &location2);
4162     ok(hres == S_OK, "get_location failed: %08x\n", hres);
4163     ok(location == location2, "location != location2\n");
4164     IHTMLLocation_Release(location2);
4165
4166     test_ifaces((IUnknown*)location, location_iids);
4167     test_disp2((IUnknown*)location, &DIID_DispHTMLLocation, &IID_IHTMLLocation, "about:blank");
4168
4169     hres = IHTMLLocation_get_pathname(location, &str);
4170     ok(hres == S_OK, "get_pathname failed: %08x\n", hres);
4171     ok(!strcmp_wa(str, "blank"), "unexpected pathname %s\n", wine_dbgstr_w(str));
4172     SysFreeString(str);
4173
4174     hres = IHTMLLocation_get_href(location, NULL);
4175     ok(hres == E_POINTER, "get_href passed: %08x\n", hres);
4176
4177     hres = IHTMLLocation_get_href(location, &str);
4178     ok(hres == S_OK, "get_href failed: %08x\n", hres);
4179     ok(!strcmp_wa(str, "about:blank"), "unexpected href %s\n", wine_dbgstr_w(str));
4180     SysFreeString(str);
4181
4182     ref = IHTMLLocation_Release(location);
4183     ok(!ref, "location chould be destroyed here\n");
4184 }
4185
4186 static void test_plugins_col(IHTMLDocument2 *doc)
4187 {
4188     IHTMLPluginsCollection *col, *col2;
4189     IHTMLWindow2 *window;
4190     IOmNavigator *nav;
4191     ULONG ref;
4192     LONG len;
4193     HRESULT hres;
4194
4195     window = get_doc_window(doc);
4196     hres = IHTMLWindow2_get_navigator(window, &nav);
4197     IHTMLWindow2_Release(window);
4198
4199     hres = IOmNavigator_get_plugins(nav, &col);
4200     ok(hres == S_OK, "get_plugins failed: %08x\n", hres);
4201
4202     hres = IOmNavigator_get_plugins(nav, &col2);
4203     ok(hres == S_OK, "get_plugins failed: %08x\n", hres);
4204     ok(iface_cmp((IUnknown*)col, (IUnknown*)col2), "col != col2\n");
4205     IHTMLPluginsCollection_Release(col2);
4206
4207     test_disp2((IUnknown*)col, &DIID_DispCPlugins, &IID_IHTMLPluginsCollection, "[object]");
4208
4209     len = 0xdeadbeef;
4210     hres = IHTMLPluginsCollection_get_length(col, &len);
4211     ok(hres == S_OK, "get_length failed: %08x\n", hres);
4212     ok(!len, "length = %d\n", len);
4213
4214     ref = IHTMLPluginsCollection_Release(col);
4215     ok(!ref, "ref=%d\n", ref);
4216
4217     IOmNavigator_Release(nav);
4218 }
4219
4220 static void test_mime_types_col(IOmNavigator *nav)
4221 {
4222     IHTMLMimeTypesCollection *col, *col2;
4223     LONG length;
4224     ULONG ref;
4225     HRESULT hres;
4226
4227     hres = IOmNavigator_get_mimeTypes(nav, &col);
4228     ok(hres == S_OK, "get_mimeTypes failed: %08x\n", hres);
4229
4230     hres = IOmNavigator_get_mimeTypes(nav, &col2);
4231     ok(hres == S_OK, "get_mimeTypes failed: %08x\n", hres);
4232     ok(iface_cmp((IUnknown*)col, (IUnknown*)col2), "col != col2\n");
4233     IHTMLMimeTypesCollection_Release(col2);
4234
4235     test_disp((IUnknown*)col, &IID_IHTMLMimeTypesCollection, "[object]");
4236
4237     length = 0xdeadbeef;
4238     hres = IHTMLMimeTypesCollection_get_length(col, &length);
4239     ok(hres == S_OK, "get_length failed: %08x\n", hres);
4240     ok(!length, "length = %d\n", length);
4241
4242     ref = IHTMLMimeTypesCollection_Release(col);
4243     ok(!ref, "ref=%d\n", ref);
4244 }
4245
4246 #define test_framebase_name(a,b) _test_framebase_name(__LINE__,a,b)
4247 static void _test_framebase_name(unsigned line, IHTMLElement *elem, const char *name)
4248 {
4249     BSTR str = (void*)0xdeadbeef;
4250     IHTMLFrameBase *fbase;
4251     HRESULT hres;
4252
4253     hres = IHTMLElement_QueryInterface(elem, &IID_IHTMLFrameBase, (void**)&fbase);
4254     ok(hres == S_OK, "Could not get IHTMLFrameBase interface: 0x%08x\n", hres);
4255
4256     hres = IHTMLFrameBase_get_name(fbase, &str);
4257     ok_(__FILE__,line)(hres == S_OK, "IHTMLFrameBase_get_name failed: 0x%08x\n", hres);
4258     if(name)
4259         ok_(__FILE__,line)(!strcmp_wa(str, name), "name = %s, expected %s\n", wine_dbgstr_w(str), name);
4260     else
4261         ok_(__FILE__,line)(!str, "name = %s, expected NULL\n", wine_dbgstr_w(str));
4262     SysFreeString(str);
4263
4264     IHTMLFrameBase_Release(fbase);
4265 }
4266
4267 #define test_framebase_put_name(a,b) _test_framebase_put_name(__LINE__,a,b)
4268 static void _test_framebase_put_name(unsigned line, IHTMLElement *elem, const char *name)
4269 {
4270     IHTMLFrameBase *fbase;
4271     HRESULT hres;
4272     BSTR str;
4273
4274     hres = IHTMLElement_QueryInterface(elem, &IID_IHTMLFrameBase, (void**)&fbase);
4275     ok(hres == S_OK, "Could not get IHTMLFrameBase interface: 0x%08x\n", hres);
4276
4277     str = name ? a2bstr(name) : NULL;
4278     hres = IHTMLFrameBase_put_name(fbase, str);
4279     ok_(__FILE__,line)(hres == S_OK, "put_name failed: %08x\n", hres);
4280     SysFreeString(str);
4281
4282     _test_framebase_name(line, elem, name);
4283     IHTMLFrameBase_Release(fbase);
4284 }
4285
4286 static void test_framebase(IUnknown *unk)
4287 {
4288     IHTMLFrameBase *fbase;
4289     BSTR str;
4290     HRESULT hres;
4291
4292     /* get/put scrolling */
4293     hres = IUnknown_QueryInterface(unk, &IID_IHTMLFrameBase, (void**)&fbase);
4294     ok(hres == S_OK, "Could not get IHTMLFrameBase interface: 0x%08x\n", hres);
4295
4296     hres = IHTMLFrameBase_get_scrolling(fbase, &str);
4297     ok(hres == S_OK, "IHTMLFrameBase_get_scrolling failed: 0x%08x\n", hres);
4298     ok(!strcmp_wa(str, "auto"), "get_scrolling should have given 'auto', gave: %s\n", wine_dbgstr_w(str));
4299     SysFreeString(str);
4300
4301     str = a2bstr("no");
4302     hres = IHTMLFrameBase_put_scrolling(fbase, str);
4303     ok(hres == S_OK, "IHTMLFrameBase_put_scrolling failed: 0x%08x\n", hres);
4304     SysFreeString(str);
4305
4306     hres = IHTMLFrameBase_get_scrolling(fbase, &str);
4307     ok(hres == S_OK, "IHTMLFrameBase_get_scrolling failed: 0x%08x\n", hres);
4308     ok(!strcmp_wa(str, "no"), "get_scrolling should have given 'no', gave: %s\n", wine_dbgstr_w(str));
4309     SysFreeString(str);
4310
4311     str = a2bstr("junk");
4312     hres = IHTMLFrameBase_put_scrolling(fbase, str);
4313     ok(hres == E_INVALIDARG, "IHTMLFrameBase_put_scrolling should have failed "
4314             "with E_INVALIDARG, instead: 0x%08x\n", hres);
4315     SysFreeString(str);
4316
4317     hres = IHTMLFrameBase_get_scrolling(fbase, &str);
4318     ok(hres == S_OK, "IHTMLFrameBase_get_scrolling failed: 0x%08x\n", hres);
4319     ok(!strcmp_wa(str, "no"), "get_scrolling should have given 'no', gave: %s\n", wine_dbgstr_w(str));
4320     SysFreeString(str);
4321
4322     hres = IHTMLFrameBase_get_frameBorder(fbase, &str);
4323     ok(hres == S_OK, "get_frameBorder failed: %08x\n", hres);
4324     ok(!str, "frameBorder = %s\n", wine_dbgstr_w(str));
4325
4326     str = a2bstr("1");
4327     hres = IHTMLFrameBase_put_frameBorder(fbase, str);
4328     ok(hres == S_OK, "put_frameBorder failed: %08x\n", hres);
4329     SysFreeString(str);
4330
4331     hres = IHTMLFrameBase_get_frameBorder(fbase, &str);
4332     ok(hres == S_OK, "get_frameBorder failed: %08x\n", hres);
4333     ok(!strcmp_wa(str, "1"), "frameBorder = %s, expected \"1\"\n", wine_dbgstr_w(str));
4334
4335     IHTMLFrameBase_Release(fbase);
4336 }
4337
4338 #define test_language_string(a,b) _test_language_string(__LINE__,a,b)
4339 static void _test_language_string(unsigned line, const WCHAR *lang, LCID lcid)
4340 {
4341     WCHAR buf[64];
4342     int res;
4343
4344     if(pLCIDToLocaleName) {
4345         res = pLCIDToLocaleName(lcid, buf, sizeof(buf)/sizeof(WCHAR), 0);
4346         ok_(__FILE__,line)(res, "LCIDToLocaleName failed: %u\n", GetLastError());
4347         ok_(__FILE__,line)(!lstrcmpW(lang, buf), "lang = %s, expected %s\n", wine_dbgstr_w(lang), wine_dbgstr_w(buf));
4348     }else {
4349         win_skip("LCIDToLocaleName not available, unable to test language string\n");
4350         ok_(__FILE__,line)(lang != NULL, "lang == NULL\n");
4351     }
4352 }
4353
4354 static void test_navigator(IHTMLDocument2 *doc)
4355 {
4356     IHTMLWindow2 *window;
4357     IOmNavigator *navigator, *navigator2;
4358     char buf[512];
4359     DWORD size;
4360     ULONG ref;
4361     BSTR bstr;
4362     HRESULT hres;
4363
4364     static const WCHAR v40[] = {'4','.','0'};
4365
4366     hres = IHTMLDocument2_get_parentWindow(doc, &window);
4367     ok(hres == S_OK, "parentWidnow failed: %08x\n", hres);
4368
4369     hres = IHTMLWindow2_get_navigator(window, &navigator);
4370     ok(hres == S_OK, "get_navigator failed: %08x\n", hres);
4371     ok(navigator != NULL, "navigator == NULL\n");
4372     test_disp2((IUnknown*)navigator, &DIID_DispHTMLNavigator, &IID_IOmNavigator, "[object]");
4373
4374     hres = IHTMLWindow2_get_navigator(window, &navigator2);
4375     ok(hres == S_OK, "get_navigator failed: %08x\n", hres);
4376     ok(navigator != navigator2, "navigator2 != navihgator\n");
4377
4378     IHTMLWindow2_Release(window);
4379     IOmNavigator_Release(navigator2);
4380
4381     hres = IOmNavigator_get_appCodeName(navigator, &bstr);
4382     ok(hres == S_OK, "get_appCodeName failed: %08x\n", hres);
4383     ok(!strcmp_wa(bstr, "Mozilla"), "Unexpected appCodeName %s\n", wine_dbgstr_w(bstr));
4384     SysFreeString(bstr);
4385
4386     bstr = NULL;
4387     hres = IOmNavigator_get_appName(navigator, &bstr);
4388     ok(hres == S_OK, "get_appName failed: %08x\n", hres);
4389     ok(!strcmp_wa(bstr, "Microsoft Internet Explorer"), "Unexpected appCodeName %s\n", wine_dbgstr_w(bstr));
4390     SysFreeString(bstr);
4391
4392     bstr = NULL;
4393     hres = IOmNavigator_get_platform(navigator, &bstr);
4394     ok(hres == S_OK, "get_platform failed: %08x\n", hres);
4395     ok(!strcmp_wa(bstr, sizeof(void*) == 8 ? "Win64" : "Win32")
4396        || (sizeof(void*) == 8 && broken(!strcmp_wa(bstr, "Win32") /* IE6 */)), "unexpected platform %s\n", wine_dbgstr_w(bstr));
4397     SysFreeString(bstr);
4398
4399     bstr = NULL;
4400     hres = IOmNavigator_get_cpuClass(navigator, &bstr);
4401     ok(hres == S_OK, "get_cpuClass failed: %08x\n", hres);
4402     ok(!strcmp_wa(bstr, sizeof(void*) == 8 ? "x64" : "x86"), "unexpected cpuClass %s\n", wine_dbgstr_w(bstr));
4403     SysFreeString(bstr);
4404
4405     bstr = NULL;
4406     hres = IOmNavigator_get_appVersion(navigator, &bstr);
4407     ok(hres == S_OK, "get_appVersion failed: %08x\n", hres);
4408     ok(!memcmp(bstr, v40, sizeof(v40)), "appVersion is %s\n", wine_dbgstr_w(bstr));
4409     SysFreeString(bstr);
4410
4411     bstr = NULL;
4412     hres = IOmNavigator_get_systemLanguage(navigator, &bstr);
4413     ok(hres == S_OK, "get_systemLanguage failed: %08x\n", hres);
4414     test_language_string(bstr, LOCALE_SYSTEM_DEFAULT);
4415     SysFreeString(bstr);
4416
4417     if (pGetUserDefaultUILanguage)
4418     {
4419         bstr = NULL;
4420         hres = IOmNavigator_get_browserLanguage(navigator, &bstr);
4421         ok(hres == S_OK, "get_browserLanguage failed: %08x\n", hres);
4422         test_language_string(bstr, pGetUserDefaultUILanguage());
4423         SysFreeString(bstr);
4424     }
4425     else
4426         win_skip("GetUserDefaultUILanguage not available\n");
4427
4428     bstr = NULL;
4429     hres = IOmNavigator_get_userLanguage(navigator, &bstr);
4430     ok(hres == S_OK, "get_userLanguage failed: %08x\n", hres);
4431     test_language_string(bstr, LOCALE_USER_DEFAULT);
4432     SysFreeString(bstr);
4433
4434     hres = IOmNavigator_toString(navigator, NULL);
4435     ok(hres == E_INVALIDARG, "toString failed: %08x\n", hres);
4436
4437     bstr = NULL;
4438     hres = IOmNavigator_toString(navigator, &bstr);
4439     ok(hres == S_OK, "toString failed: %08x\n", hres);
4440     ok(!strcmp_wa(bstr, "[object]"), "toString returned %s\n", wine_dbgstr_w(bstr));
4441     SysFreeString(bstr);
4442
4443     size = sizeof(buf);
4444     hres = ObtainUserAgentString(0, buf, &size);
4445     ok(hres == S_OK, "ObtainUserAgentString failed: %08x\n", hres);
4446
4447     bstr = NULL;
4448     hres = IOmNavigator_get_userAgent(navigator, &bstr);
4449     ok(hres == S_OK, "get_userAgent failed: %08x\n", hres);
4450     ok(!strcmp_wa(bstr, buf), "userAgent returned %s, expected \"%s\"\n", wine_dbgstr_w(bstr), buf);
4451     SysFreeString(bstr);
4452
4453     if(!strncmp(buf, "Mozilla/", 8)) {
4454         bstr = NULL;
4455         hres = IOmNavigator_get_appVersion(navigator, &bstr);
4456         ok(hres == S_OK, "get_appVersion failed: %08x\n", hres);
4457         ok(!strcmp_wa(bstr, buf+8), "appVersion returned %s, expected \"%s\"\n", wine_dbgstr_w(bstr), buf+8);
4458         SysFreeString(bstr);
4459     }else {
4460         skip("nonstandard user agent\n");
4461     }
4462
4463     bstr = NULL;
4464     hres = IOmNavigator_get_appMinorVersion(navigator, &bstr);
4465     ok(hres == S_OK, "get_appMonorVersion failed: %08x\n", hres);
4466     ok(bstr != NULL, "appMinorVersion returned NULL\n");
4467     SysFreeString(bstr);
4468
4469     test_mime_types_col(navigator);
4470
4471     ref = IOmNavigator_Release(navigator);
4472     ok(!ref, "navigator should be destroyed here\n");
4473 }
4474
4475 static void test_screen(IHTMLWindow2 *window)
4476 {
4477     IHTMLScreen *screen, *screen2;
4478     IDispatchEx *dispex;
4479     RECT work_area;
4480     LONG l, exl;
4481     HDC hdc;
4482     HRESULT hres;
4483
4484     screen = NULL;
4485     hres = IHTMLWindow2_get_screen(window, &screen);
4486     ok(hres == S_OK, "get_screen failed: %08x\n", hres);
4487     ok(screen != NULL, "screen == NULL\n");
4488
4489     screen2 = NULL;
4490     hres = IHTMLWindow2_get_screen(window, &screen2);
4491     ok(hres == S_OK, "get_screen failed: %08x\n", hres);
4492     ok(screen2 != NULL, "screen == NULL\n");
4493     ok(iface_cmp((IUnknown*)screen2, (IUnknown*)screen), "screen2 != screen\n");
4494     IHTMLScreen_Release(screen2);
4495
4496     hres = IHTMLScreen_QueryInterface(screen, &IID_IDispatchEx, (void**)&dispex);
4497     ok(hres == S_OK || broken(hres == E_NOINTERFACE), "Could not get IDispatchEx interface: %08x\n", hres);
4498     if(SUCCEEDED(hres)) {
4499         test_disp((IUnknown*)screen, &DIID_DispHTMLScreen, "[object]");
4500         IDispatchEx_Release(dispex);
4501     }
4502
4503     hdc = CreateICA("DISPLAY", NULL, NULL, NULL);
4504
4505     exl = GetDeviceCaps(hdc, HORZRES);
4506     l = 0xdeadbeef;
4507     hres = IHTMLScreen_get_width(screen, &l);
4508     ok(hres == S_OK, "get_width failed: %08x\n", hres);
4509     ok(l == exl, "width = %d, expected %d\n", l, exl);
4510
4511     exl = GetDeviceCaps(hdc, VERTRES);
4512     l = 0xdeadbeef;
4513     hres = IHTMLScreen_get_height(screen, &l);
4514     ok(hres == S_OK, "get_height failed: %08x\n", hres);
4515     ok(l == exl, "height = %d, expected %d\n", l, exl);
4516
4517     exl = GetDeviceCaps(hdc, BITSPIXEL);
4518     l = 0xdeadbeef;
4519     hres = IHTMLScreen_get_colorDepth(screen, &l);
4520     ok(hres == S_OK, "get_height failed: %08x\n", hres);
4521     ok(l == exl, "height = %d, expected %d\n", l, exl);
4522
4523     DeleteObject(hdc);
4524
4525     SystemParametersInfoW(SPI_GETWORKAREA, 0, &work_area, 0);
4526
4527     l = 0xdeadbeef;
4528     hres = IHTMLScreen_get_availHeight(screen, &l);
4529     ok(hres == S_OK, "get_availHeight failed: %08x\n", hres);
4530     ok(l == work_area.bottom-work_area.top, "availHeight = %d, expected %d\n", l, work_area.bottom-work_area.top);
4531
4532     l = 0xdeadbeef;
4533     hres = IHTMLScreen_get_availWidth(screen, &l);
4534     ok(hres == S_OK, "get_availWidth failed: %08x\n", hres);
4535     ok(l == work_area.right-work_area.left, "availWidth = %d, expected %d\n", l, work_area.right-work_area.left);
4536
4537     IHTMLScreen_Release(screen);
4538 }
4539
4540 static void test_default_selection(IHTMLDocument2 *doc)
4541 {
4542     IHTMLSelectionObject *selection;
4543     IHTMLTxtRange *range;
4544     IDispatch *disp;
4545     BSTR str;
4546     HRESULT hres;
4547
4548     hres = IHTMLDocument2_get_selection(doc, &selection);
4549     ok(hres == S_OK, "get_selection failed: %08x\n", hres);
4550
4551     hres = IHTMLSelectionObject_get_type(selection, &str);
4552     ok(hres == S_OK, "get_type failed: %08x\n", hres);
4553     ok(!strcmp_wa(str, "None"), "type = %s\n", wine_dbgstr_w(str));
4554     SysFreeString(str);
4555
4556     hres = IHTMLSelectionObject_createRange(selection, &disp);
4557     IHTMLSelectionObject_Release(selection);
4558     ok(hres == S_OK, "createRange failed: %08x\n", hres);
4559
4560     hres = IDispatch_QueryInterface(disp, &IID_IHTMLTxtRange, (void**)&range);
4561     IDispatch_Release(disp);
4562     ok(hres == S_OK, "Could not get IHTMLTxtRange interface: %08x\n", hres);
4563
4564     test_range_text(range, NULL);
4565     IHTMLTxtRange_Release(range);
4566 }
4567
4568 static void test_doc_elem(IHTMLDocument2 *doc)
4569 {
4570     IHTMLDocument2 *doc_node, *owner_doc;
4571     IHTMLElement *elem;
4572     IHTMLDocument3 *doc3;
4573     HRESULT hres;
4574
4575     hres = IHTMLDocument2_QueryInterface(doc, &IID_IHTMLDocument3, (void**)&doc3);
4576     ok(hres == S_OK, "QueryInterface(IID_IHTMLDocument3) failed: %08x\n", hres);
4577
4578     hres = IHTMLDocument3_get_documentElement(doc3, &elem);
4579     IHTMLDocument3_Release(doc3);
4580     ok(hres == S_OK, "get_documentElement failed: %08x\n", hres);
4581
4582     test_node_name((IUnknown*)elem, "HTML");
4583     test_elem_tag((IUnknown*)elem, "HTML");
4584
4585     doc_node = get_doc_node(doc);
4586     owner_doc = get_owner_doc((IUnknown*)elem);
4587     ok(iface_cmp((IUnknown *)doc_node, (IUnknown *)owner_doc), "doc_node != owner_doc\n");
4588     IHTMLDocument2_Release(owner_doc);
4589
4590     owner_doc = get_owner_doc((IUnknown*)doc_node);
4591     ok(!owner_doc, "owner_doc = %p\n", owner_doc);
4592     IHTMLDocument2_Release(doc_node);
4593
4594     test_elem_client_rect((IUnknown*)elem);
4595
4596     IHTMLElement_Release(elem);
4597 }
4598
4599 static void test_default_body(IHTMLBodyElement *body)
4600 {
4601     LONG l;
4602     BSTR bstr;
4603     HRESULT hres;
4604     VARIANT v;
4605
4606     bstr = (void*)0xdeadbeef;
4607     hres = IHTMLBodyElement_get_background(body, &bstr);
4608     ok(hres == S_OK, "get_background failed: %08x\n", hres);
4609     ok(bstr == NULL, "bstr != NULL\n");
4610
4611     l = elem_get_scroll_height((IUnknown*)body);
4612     ok(l != -1, "scrollHeight == -1\n");
4613     l = elem_get_scroll_width((IUnknown*)body);
4614     ok(l != -1, "scrollWidth == -1\n");
4615     l = elem_get_scroll_top((IUnknown*)body);
4616     ok(!l, "scrollTop = %d\n", l);
4617     elem_get_scroll_left((IUnknown*)body);
4618
4619     /* get_text tests */
4620     hres = IHTMLBodyElement_get_text(body, &v);
4621     ok(hres == S_OK, "expect S_OK got 0x%08d\n", hres);
4622     ok(V_VT(&v) == VT_BSTR, "Expected VT_BSTR got %d\n", V_VT(&v));
4623     ok(V_BSTR(&v) == NULL, "bstr != NULL\n");
4624
4625     /* get_text - Invalid Text */
4626     V_VT(&v) = VT_BSTR;
4627     V_BSTR(&v) = a2bstr("Invalid");
4628     hres = IHTMLBodyElement_put_text(body, v);
4629     ok(hres == S_OK, "expect S_OK got 0x%08d\n", hres);
4630     VariantClear(&v);
4631
4632     V_VT(&v) = VT_NULL;
4633     hres = IHTMLBodyElement_get_text(body, &v);
4634     ok(hres == S_OK, "expect S_OK got 0x%08d\n", hres);
4635     ok(V_VT(&v) == VT_BSTR, "Expected VT_BSTR got %d\n", V_VT(&v));
4636     ok(!strcmp_wa(V_BSTR(&v), "#00a0d0"), "v = %s, expected '#00a0d0'\n", wine_dbgstr_w(V_BSTR(&v)));
4637     VariantClear(&v);
4638
4639     /* get_text - Valid Text */
4640     V_VT(&v) = VT_BSTR;
4641     V_BSTR(&v) = a2bstr("#FF0000");
4642     hres = IHTMLBodyElement_put_text(body, v);
4643     ok(hres == S_OK, "expect S_OK got 0x%08d\n", hres);
4644     VariantClear(&v);
4645
4646     V_VT(&v) = VT_NULL;
4647     hres = IHTMLBodyElement_get_text(body, &v);
4648     ok(hres == S_OK, "expect S_OK got 0x%08d\n", hres);
4649     ok(V_VT(&v) == VT_BSTR, "Expected VT_BSTR got %d\n", V_VT(&v));
4650     ok(!strcmp_wa(V_BSTR(&v), "#ff0000"), "v = %s, expected '#ff0000'\n", wine_dbgstr_w(V_BSTR(&v)));
4651     VariantClear(&v);
4652 }
4653
4654 static void test_body_funs(IHTMLBodyElement *body)
4655 {
4656     VARIANT vbg, vDefaultbg;
4657     HRESULT hres;
4658
4659     hres = IHTMLBodyElement_get_bgColor(body, &vDefaultbg);
4660     ok(hres == S_OK, "get_bgColor failed: %08x\n", hres);
4661     ok(V_VT(&vDefaultbg) == VT_BSTR, "bstr != NULL\n");
4662     ok(!V_BSTR(&vDefaultbg), "V_BSTR(bgColor) = %s\n", wine_dbgstr_w(V_BSTR(&vDefaultbg)));
4663
4664     V_VT(&vbg) = VT_BSTR;
4665     V_BSTR(&vbg) = a2bstr("red");
4666     hres = IHTMLBodyElement_put_bgColor(body, vbg);
4667     ok(hres == S_OK, "put_bgColor failed: %08x\n", hres);
4668     VariantClear(&vbg);
4669
4670     hres = IHTMLBodyElement_get_bgColor(body, &vbg);
4671     ok(hres == S_OK, "get_bgColor failed: %08x\n", hres);
4672     ok(V_VT(&vDefaultbg) == VT_BSTR, "V_VT(&vDefaultbg) != VT_BSTR\n");
4673     ok(!strcmp_wa(V_BSTR(&vbg), "#ff0000"), "Unexpected bgcolor %s\n", wine_dbgstr_w(V_BSTR(&vbg)));
4674     VariantClear(&vbg);
4675
4676     /* Restore Originial */
4677     hres = IHTMLBodyElement_put_bgColor(body, vDefaultbg);
4678     ok(hres == S_OK, "put_bgColor failed: %08x\n", hres);
4679     VariantClear(&vDefaultbg);
4680 }
4681
4682 static void test_history(IHTMLWindow2 *window)
4683 {
4684     IOmHistory *history, *history2;
4685     HRESULT hres;
4686
4687     history = NULL;
4688     hres = IHTMLWindow2_get_history(window, &history);
4689     ok(hres == S_OK, "get_history failed: %08x\n", hres);
4690     ok(history != NULL, "history = NULL\n");
4691
4692     test_disp2((IUnknown*)history, &DIID_DispHTMLHistory, &IID_IOmHistory, "[object]");
4693
4694     history2 = NULL;
4695     hres = IHTMLWindow2_get_history(window, &history2);
4696     ok(hres == S_OK, "get_history failed: %08x\n", hres);
4697     ok(history2 != NULL, "history2 = NULL\n");
4698     ok(iface_cmp((IUnknown*)history, (IUnknown*)history2), "history != history2\n");
4699
4700     IOmHistory_Release(history2);
4701     IOmHistory_Release(history);
4702 }
4703
4704 static void test_window(IHTMLDocument2 *doc)
4705 {
4706     IHTMLWindow2 *window, *window2, *self, *parent;
4707     IHTMLDocument2 *doc2 = NULL;
4708     IDispatch *disp;
4709     IUnknown *unk;
4710     VARIANT v;
4711     BSTR str;
4712     HRESULT hres;
4713
4714     hres = IHTMLDocument2_get_parentWindow(doc, &window);
4715     ok(hres == S_OK, "get_parentWindow failed: %08x\n", hres);
4716     test_ifaces((IUnknown*)window, window_iids);
4717     hres = IHTMLWindow2_QueryInterface(window, &IID_ITravelLogClient, (void**)&unk);
4718     if(hres == S_OK)
4719         IUnknown_Release(unk);
4720     else
4721         win_skip("IID_ITravelLogClient not supported\n");
4722
4723     test_disp((IUnknown*)window, &DIID_DispHTMLWindow2, "[object]");
4724
4725     hres = IHTMLWindow2_get_document(window, &doc2);
4726     ok(hres == S_OK, "get_document failed: %08x\n", hres);
4727     ok(doc2 != NULL, "doc2 == NULL\n");
4728
4729     test_ifaces((IUnknown*)doc2, doc_node_iids);
4730     test_disp((IUnknown*)doc2, &DIID_DispHTMLDocument, "[object]");
4731     test_class_info((IUnknown*)doc2);
4732
4733     test_ifaces((IUnknown*)doc, doc_obj_iids);
4734     test_disp((IUnknown*)doc, &DIID_DispHTMLDocument, "[object]");
4735     test_class_info((IUnknown*)doc);
4736
4737     unk = (void*)0xdeadbeef;
4738     hres = IHTMLDocument2_QueryInterface(doc2, &IID_ICustomDoc, (void**)&unk);
4739     ok(hres == E_NOINTERFACE, "QueryInterface(IID_ICustomDoc) returned: %08x\n", hres);
4740     ok(!unk, "unk = %p\n", unk);
4741
4742     IHTMLDocument2_Release(doc2);
4743
4744     hres = IHTMLWindow2_get_window(window, &window2);
4745     ok(hres == S_OK, "get_window failed: %08x\n", hres);
4746     ok(window2 != NULL, "window2 == NULL\n");
4747
4748     hres = IHTMLWindow2_get_self(window, &self);
4749     ok(hres == S_OK, "get_self failed: %08x\n", hres);
4750     ok(window2 != NULL, "self == NULL\n");
4751
4752     ok(self == window2, "self != window2\n");
4753
4754     IHTMLWindow2_Release(window2);
4755
4756     disp = NULL;
4757     hres = IHTMLDocument2_get_Script(doc, &disp);
4758     ok(hres == S_OK, "get_Script failed: %08x\n", hres);
4759     ok(disp == (void*)window, "disp != window\n");
4760     IDispatch_Release(disp);
4761
4762     hres = IHTMLWindow2_toString(window, NULL);
4763     ok(hres == E_INVALIDARG, "toString failed: %08x\n", hres);
4764
4765     str = NULL;
4766     hres = IHTMLWindow2_toString(window, &str);
4767     ok(hres == S_OK, "toString failed: %08x\n", hres);
4768     ok(!strcmp_wa(str, "[object]") ||
4769        !strcmp_wa(str, "[object Window]") /* win7 ie9 */, "toString returned %s\n", wine_dbgstr_w(str));
4770     SysFreeString(str);
4771
4772     V_VT(&v) = VT_ERROR;
4773     hres = IHTMLWindow2_get_opener(window, &v);
4774     ok(hres == S_OK, "get_opener failed: %08x\n", hres);
4775     ok(V_VT(&v) == VT_EMPTY, "V_VT(opener) = %d\n", V_VT(&v));
4776
4777     parent = NULL;
4778     hres = IHTMLWindow2_get_parent(window, &parent);
4779     ok(hres == S_OK, "get_parent failed: %08x\n", hres);
4780     ok(parent != NULL, "parent == NULL\n");
4781     ok(parent == self, "parent != window\n");
4782     IHTMLWindow2_Release(parent);
4783     IHTMLWindow2_Release(self);
4784
4785     test_window_name(window, NULL);
4786     set_window_name(window, "test");
4787     test_window_length(window, 0);
4788     test_screen(window);
4789     test_window_status(window);
4790     set_window_status(window, "Test!");
4791     test_history(window);
4792
4793     IHTMLWindow2_Release(window);
4794 }
4795
4796 static void test_defaults(IHTMLDocument2 *doc)
4797 {
4798     IHTMLStyleSheetsCollection *stylesheetcol;
4799     IHTMLCurrentStyle *cstyle;
4800     IHTMLBodyElement *body;
4801     IHTMLElement2 *elem2;
4802     IHTMLElement *elem;
4803     IHTMLStyle *style;
4804     BSTR str;
4805     LONG l;
4806     HRESULT hres;
4807     IHTMLElementCollection *collection;
4808
4809     elem = doc_get_body(doc);
4810
4811     hres = IHTMLDocument2_get_images(doc, NULL);
4812     ok(hres == E_INVALIDARG, "hres %08x\n", hres);
4813
4814     hres = IHTMLDocument2_get_images(doc, &collection);
4815     ok(hres == S_OK, "get_images failed: %08x\n", hres);
4816     if(hres == S_OK)
4817     {
4818         test_elem_collection((IUnknown*)collection, NULL, 0);
4819         IHTMLElementCollection_Release(collection);
4820     }
4821
4822     hres = IHTMLDocument2_get_applets(doc, NULL);
4823     ok(hres == E_INVALIDARG, "hres %08x\n", hres);
4824
4825     hres = IHTMLDocument2_get_applets(doc, &collection);
4826     ok(hres == S_OK, "get_applets failed: %08x\n", hres);
4827     if(hres == S_OK)
4828     {
4829         test_elem_collection((IUnknown*)collection, NULL, 0);
4830         IHTMLElementCollection_Release(collection);
4831     }
4832
4833     hres = IHTMLDocument2_get_links(doc, NULL);
4834     ok(hres == E_INVALIDARG, "hres %08x\n", hres);
4835
4836     hres = IHTMLDocument2_get_links(doc, &collection);
4837     ok(hres == S_OK, "get_links failed: %08x\n", hres);
4838     if(hres == S_OK)
4839     {
4840         test_elem_collection((IUnknown*)collection, NULL, 0);
4841         IHTMLElementCollection_Release(collection);
4842     }
4843
4844     hres = IHTMLDocument2_get_forms(doc, NULL);
4845     ok(hres == E_INVALIDARG, "hres %08x\n", hres);
4846
4847     hres = IHTMLDocument2_get_forms(doc, &collection);
4848     ok(hres == S_OK, "get_forms failed: %08x\n", hres);
4849     if(hres == S_OK)
4850     {
4851         test_elem_collection((IUnknown*)collection, NULL, 0);
4852         IHTMLElementCollection_Release(collection);
4853     }
4854
4855     hres = IHTMLDocument2_get_anchors(doc, NULL);
4856     ok(hres == E_INVALIDARG, "hres %08x\n", hres);
4857
4858     hres = IHTMLDocument2_get_anchors(doc, &collection);
4859     ok(hres == S_OK, "get_anchors failed: %08x\n", hres);
4860     if(hres == S_OK)
4861     {
4862         test_elem_collection((IUnknown*)collection, NULL, 0);
4863         IHTMLElementCollection_Release(collection);
4864     }
4865
4866     hres = IHTMLElement_QueryInterface(elem, &IID_IHTMLBodyElement, (void**)&body);
4867     ok(hres == S_OK, "Could not get IHTMBodyElement: %08x\n", hres);
4868     test_default_body(body);
4869     test_body_funs(body);
4870     IHTMLBodyElement_Release(body);
4871
4872     hres = IHTMLElement_get_style(elem, &style);
4873     ok(hres == S_OK, "get_style failed: %08x\n", hres);
4874
4875     test_disp((IUnknown*)style, &DIID_DispHTMLStyle, "[object]");
4876     test_ifaces((IUnknown*)style, style_iids);
4877     IHTMLStyle_Release(style);
4878
4879     str = NULL;
4880     hres = IHTMLDocument2_get_charset(doc, &str);
4881     ok(hres == S_OK, "get_charset failed: %08x\n", hres);
4882     ok(str && *str, "charset is empty\n"); /* FIXME: better tests */
4883     SysFreeString(str);
4884
4885     test_window(doc);
4886     test_compatmode(doc);
4887     test_location(doc);
4888     test_navigator(doc);
4889     test_plugins_col(doc);
4890
4891     elem2 = get_elem2_iface((IUnknown*)elem);
4892     hres = IHTMLElement2_get_currentStyle(elem2, &cstyle);
4893     ok(hres == S_OK, "get_currentStyle failed: %08x\n", hres);
4894     if(SUCCEEDED(hres)) {
4895         IUnknown *unk;
4896
4897         test_disp((IUnknown*)cstyle, &DIID_DispHTMLCurrentStyle, "[object]");
4898         test_ifaces((IUnknown*)cstyle, cstyle_iids);
4899
4900         hres = IHTMLCurrentStyle_QueryInterface(cstyle, &IID_IHTMLCurrentStyle4, (void**)&unk);
4901         if(SUCCEEDED(hres))
4902             IUnknown_Release(unk);
4903         else
4904         {
4905            /*IE6 doesn't have interface */
4906            win_skip("IID_IHTMLCurrentStyle4 not supported\n");
4907         }
4908
4909         IHTMLCurrentStyle_Release(cstyle);
4910     }
4911     IHTMLElement2_Release(elem2);
4912
4913     IHTMLElement_Release(elem);
4914
4915     hres = IHTMLDocument2_get_styleSheets(doc, &stylesheetcol);
4916     ok(hres == S_OK, "get_styleSheets failed: %08x\n", hres);
4917
4918     l = 0xdeadbeef;
4919     hres = IHTMLStyleSheetsCollection_get_length(stylesheetcol, &l);
4920     ok(hres == S_OK, "get_length failed: %08x\n", hres);
4921     ok(l == 0, "length = %d\n", l);
4922
4923     IHTMLStyleSheetsCollection_Release(stylesheetcol);
4924
4925     hres = IHTMLElement_QueryInterface(elem, &IID_IHTMLFiltersCollection, (void**)&body);
4926     ok(hres == E_NOINTERFACE, "got interface IHTMLFiltersCollection\n");
4927
4928     test_default_selection(doc);
4929     test_doc_title(doc, "");
4930 }
4931
4932 static void test_tr_elem(IHTMLElement *elem)
4933 {
4934     IHTMLElementCollection *col;
4935     IHTMLTableRow *row;
4936     HRESULT hres;
4937
4938     static const elem_type_t cell_types[] = {ET_TD,ET_TD};
4939
4940     hres = IHTMLElement_QueryInterface(elem, &IID_IHTMLTableRow, (void**)&row);
4941     ok(hres == S_OK, "Could not get IHTMLTableRow iface: %08x\n", hres);
4942     if(FAILED(hres))
4943         return;
4944
4945     col = NULL;
4946     hres = IHTMLTableRow_get_cells(row, &col);
4947     ok(hres == S_OK, "get_cells failed: %08x\n", hres);
4948     ok(col != NULL, "get_cells returned NULL\n");
4949
4950     test_elem_collection((IUnknown*)col, cell_types, sizeof(cell_types)/sizeof(*cell_types));
4951     IHTMLElementCollection_Release(col);
4952
4953     IHTMLTableRow_Release(row);
4954 }
4955
4956 static void test_table_elem(IHTMLElement *elem)
4957 {
4958     IHTMLElementCollection *col;
4959     IHTMLTable *table;
4960     IHTMLDOMNode *node;
4961     HRESULT hres;
4962
4963     static const elem_type_t row_types[] = {ET_TR,ET_TR};
4964     static const elem_type_t all_types[] = {ET_TBODY,ET_TR,ET_TR,ET_TD,ET_TD};
4965
4966     hres = IHTMLElement_QueryInterface(elem, &IID_IHTMLTable, (void**)&table);
4967     ok(hres == S_OK, "Could not get IHTMLTable iface: %08x\n", hres);
4968     if(FAILED(hres))
4969         return;
4970
4971     col = NULL;
4972     hres = IHTMLTable_get_rows(table, &col);
4973     ok(hres == S_OK, "get_rows failed: %08x\n", hres);
4974     ok(col != NULL, "get_ros returned NULL\n");
4975
4976     test_elem_collection((IUnknown*)col, row_types, sizeof(row_types)/sizeof(*row_types));
4977     IHTMLElementCollection_Release(col);
4978
4979     test_elem_all((IUnknown*)table, all_types, sizeof(all_types)/sizeof(*all_types));
4980
4981     node = clone_node((IUnknown*)table, VARIANT_TRUE);
4982     test_elem_tag((IUnknown*)node, "TABLE");
4983     test_elem_all((IUnknown*)node, all_types, sizeof(all_types)/sizeof(*all_types));
4984     IHTMLDOMNode_Release(node);
4985
4986     node = clone_node((IUnknown*)table, VARIANT_FALSE);
4987     test_elem_tag((IUnknown*)node, "TABLE");
4988     test_elem_all((IUnknown*)node, NULL, 0);
4989     IHTMLDOMNode_Release(node);
4990
4991     IHTMLTable_Release(table);
4992 }
4993
4994 static void doc_write(IHTMLDocument2 *doc, BOOL ln, const char *text)
4995 {
4996     SAFEARRAYBOUND dim;
4997     SAFEARRAY *sa;
4998     VARIANT *var;
4999     HRESULT hres;
5000
5001     dim.lLbound = 0;
5002     dim.cElements = 1;
5003     sa = SafeArrayCreate(VT_VARIANT, 1, &dim);
5004     SafeArrayAccessData(sa, (void**)&var);
5005     V_VT(var) = VT_BSTR;
5006     V_BSTR(var) = a2bstr(text);
5007     SafeArrayUnaccessData(sa);
5008
5009     if(ln)
5010         hres = IHTMLDocument2_writeln(doc, sa);
5011     else
5012         hres = IHTMLDocument2_write(doc, sa);
5013     ok(hres == S_OK, "write failed: %08x\n", hres);
5014
5015     SafeArrayDestroy(sa);
5016 }
5017
5018 static void doc_complex_write(IHTMLDocument2 *doc)
5019 {
5020     SAFEARRAYBOUND dim = {5, 0};
5021     SAFEARRAY *sa;
5022     VARIANT *args;
5023     HRESULT hres;
5024
5025     sa = SafeArrayCreate(VT_VARIANT, 1, &dim);
5026     SafeArrayAccessData(sa, (void**)&args);
5027
5028     V_VT(args) = VT_BSTR;
5029     V_BSTR(args) = a2bstr("<body i4val=\"");
5030     V_VT(args+1) = VT_I4;
5031     V_I4(args+1) = 4;
5032     V_VT(args+2) = VT_BSTR;
5033     V_BSTR(args+2) = a2bstr("\" r8val=\"");
5034     V_VT(args+3) = VT_R8;
5035     V_R8(args+3) = 3.14;
5036     V_VT(args+4) = VT_BSTR;
5037     V_BSTR(args+4) = a2bstr("\">");
5038     SafeArrayUnaccessData(sa);
5039
5040     hres = IHTMLDocument2_write(doc, sa);
5041     ok(hres == S_OK, "write failed: %08x\n", hres);
5042
5043     SafeArrayDestroy(sa);
5044 }
5045
5046 static void test_frame_doc(IUnknown *frame_elem, BOOL iframe)
5047 {
5048     IHTMLDocument2 *window_doc, *elem_doc;
5049     IHTMLFrameElement3 *frame_elem3;
5050     IHTMLWindow2 *content_window;
5051     HRESULT hres;
5052
5053     content_window = get_frame_content_window(frame_elem);
5054     test_ifaces((IUnknown*)content_window, window_iids);
5055     window_doc = get_window_doc(content_window);
5056     IHTMLWindow2_Release(content_window);
5057
5058     elem_doc = get_elem_doc(frame_elem);
5059     ok(iface_cmp((IUnknown*)window_doc, (IUnknown*)elem_doc), "content_doc != elem_doc\n");
5060
5061     if(!iframe) {
5062         hres = IUnknown_QueryInterface(frame_elem, &IID_IHTMLFrameElement3, (void**)&frame_elem3);
5063         if(SUCCEEDED(hres)) {
5064             IDispatch *disp = NULL;
5065
5066             hres = IHTMLFrameElement3_get_contentDocument(frame_elem3, &disp);
5067             ok(hres == S_OK, "get_contentDocument failed: %08x\n", hres);
5068             ok(disp != NULL, "contentDocument == NULL\n");
5069             ok(iface_cmp((IUnknown*)disp, (IUnknown*)window_doc), "contentDocument != contentWindow.document\n");
5070
5071             IDispatch_Release(disp);
5072             IHTMLFrameElement3_Release(frame_elem3);
5073         }else {
5074             win_skip("IHTMLFrameElement3 not supported\n");
5075         }
5076     }
5077
5078     IHTMLDocument2_Release(elem_doc);
5079     IHTMLDocument2_Release(window_doc);
5080 }
5081
5082 static void test_iframe_elem(IHTMLElement *elem)
5083 {
5084     IHTMLDocument2 *content_doc, *owner_doc;
5085     IHTMLIFrameElement3 *iframe3;
5086     IHTMLElementCollection *col;
5087     IHTMLWindow2 *content_window;
5088     IHTMLElement *body;
5089     IDispatch *disp;
5090     VARIANT errv;
5091     BSTR str;
5092     HRESULT hres;
5093
5094     static const elem_type_t all_types[] = {
5095         ET_HTML,
5096         ET_HEAD,
5097         ET_TITLE,
5098         ET_BODY,
5099         ET_BR
5100     };
5101
5102     test_frame_doc((IUnknown*)elem, TRUE);
5103     test_framebase((IUnknown*)elem);
5104
5105     content_window = get_frame_content_window((IUnknown*)elem);
5106     test_ifaces((IUnknown*)content_window, window_iids);
5107     test_window_length(content_window, 0);
5108
5109     content_doc = get_window_doc(content_window);
5110     IHTMLWindow2_Release(content_window);
5111
5112     hres = IHTMLElement_QueryInterface(elem, &IID_IHTMLIFrameElement3, (void**)&iframe3);
5113     if(SUCCEEDED(hres)) {
5114         hres = IHTMLIFrameElement3_get_contentDocument(iframe3, &disp);
5115         ok(hres == S_OK, "get_contentDocument failed: %08x\n", hres);
5116         ok(iface_cmp((IUnknown*)content_doc, (IUnknown*)disp), "content_doc != disp\n");
5117         IDispatch_Release(disp);
5118
5119         IHTMLIFrameElement3_Release(iframe3);
5120     }else {
5121         win_skip("IHTMLIFrameElement3 not supported\n");
5122     }
5123
5124     str = a2bstr("text/html");
5125     V_VT(&errv) = VT_ERROR;
5126     disp = NULL;
5127     hres = IHTMLDocument2_open(content_doc, str, errv, errv, errv, &disp);
5128     SysFreeString(str);
5129     ok(hres == S_OK, "open failed: %08x\n", hres);
5130     ok(disp != NULL, "disp == NULL\n");
5131     ok(iface_cmp((IUnknown*)disp, (IUnknown*)content_window), "disp != content_window\n");
5132     IDispatch_Release(disp);
5133
5134     doc_write(content_doc, FALSE, "<html><head><title>test</title></head>");
5135     doc_complex_write(content_doc);
5136     doc_write(content_doc, TRUE, "<br />");
5137     doc_write(content_doc, TRUE, "</html>");
5138
5139     hres = IHTMLDocument2_get_all(content_doc, &col);
5140     ok(hres == S_OK, "get_all failed: %08x\n", hres);
5141     test_elem_collection((IUnknown*)col, all_types, sizeof(all_types)/sizeof(all_types[0]));
5142     IHTMLElementCollection_Release(col);
5143
5144     body = doc_get_body(content_doc);
5145     test_elem_attr(body, "i4val", "4");
5146     test_elem_attr(body, "r8val", "3.14");
5147     IHTMLElement_Release(body);
5148
5149     hres = IHTMLDocument2_close(content_doc);
5150     ok(hres == S_OK, "close failed: %08x\n", hres);
5151
5152     owner_doc = get_owner_doc((IUnknown*)content_doc);
5153     ok(!owner_doc, "owner_doc = %p\n", owner_doc);
5154
5155     IHTMLDocument2_Release(content_doc);
5156 }
5157
5158 static void test_stylesheet(IDispatch *disp)
5159 {
5160     IHTMLStyleSheetRulesCollection *col = NULL;
5161     IHTMLStyleSheet *stylesheet;
5162     HRESULT hres;
5163
5164     hres = IDispatch_QueryInterface(disp, &IID_IHTMLStyleSheet, (void**)&stylesheet);
5165     ok(hres == S_OK, "Could not get IHTMLStyleSheet: %08x\n", hres);
5166
5167     hres = IHTMLStyleSheet_get_rules(stylesheet, &col);
5168     ok(hres == S_OK, "get_rules failed: %08x\n", hres);
5169     ok(col != NULL, "col == NULL\n");
5170
5171     IHTMLStyleSheetRulesCollection_Release(col);
5172     IHTMLStyleSheet_Release(stylesheet);
5173 }
5174
5175 static void test_stylesheets(IHTMLDocument2 *doc)
5176 {
5177     IHTMLStyleSheetsCollection *col = NULL;
5178     VARIANT idx, res;
5179     LONG len = 0;
5180     HRESULT hres;
5181
5182     hres = IHTMLDocument2_get_styleSheets(doc, &col);
5183     ok(hres == S_OK, "get_styleSheets failed: %08x\n", hres);
5184     ok(col != NULL, "col == NULL\n");
5185
5186     test_disp2((IUnknown*)col, &DIID_DispHTMLStyleSheetsCollection, &IID_IHTMLStyleSheetsCollection, "[object]");
5187
5188     hres = IHTMLStyleSheetsCollection_get_length(col, &len);
5189     ok(hres == S_OK, "get_length failed: %08x\n", hres);
5190     ok(len == 1, "len=%d\n", len);
5191
5192     VariantInit(&res);
5193     V_VT(&idx) = VT_I4;
5194     V_I4(&idx) = 0;
5195
5196     hres = IHTMLStyleSheetsCollection_item(col, &idx, &res);
5197     ok(hres == S_OK, "item failed: %08x\n", hres);
5198     ok(V_VT(&res) == VT_DISPATCH, "V_VT(res) = %d\n", V_VT(&res));
5199     ok(V_DISPATCH(&res) != NULL, "V_DISPATCH(&res) == NULL\n");
5200     test_stylesheet(V_DISPATCH(&res));
5201     VariantClear(&res);
5202
5203     V_VT(&res) = VT_I4;
5204     V_VT(&idx) = VT_I4;
5205     V_I4(&idx) = 1;
5206
5207     hres = IHTMLStyleSheetsCollection_item(col, &idx, &res);
5208     ok(hres == E_INVALIDARG, "item failed: %08x, expected E_INVALIDARG\n", hres);
5209     ok(V_VT(&res) == VT_EMPTY, "V_VT(res) = %d\n", V_VT(&res));
5210     ok(V_DISPATCH(&res) != NULL, "V_DISPATCH(&res) == NULL\n");
5211     VariantClear(&res);
5212
5213     IHTMLStyleSheetsCollection_Release(col);
5214 }
5215
5216 static void test_child_col_disp(IHTMLDOMChildrenCollection *col)
5217 {
5218     IDispatchEx *dispex;
5219     IHTMLDOMNode *node;
5220     DISPPARAMS dp = {NULL, NULL, 0, 0};
5221     VARIANT var;
5222     EXCEPINFO ei;
5223     LONG type;
5224     DISPID id;
5225     BSTR bstr;
5226     HRESULT hres;
5227
5228     static const WCHAR w0[] = {'0',0};
5229     static const WCHAR w100[] = {'1','0','0',0};
5230
5231     hres = IHTMLDOMChildrenCollection_QueryInterface(col, &IID_IDispatchEx, (void**)&dispex);
5232     ok(hres == S_OK, "Could not get IDispatchEx: %08x\n", hres);
5233
5234     bstr = SysAllocString(w0);
5235     hres = IDispatchEx_GetDispID(dispex, bstr, fdexNameCaseSensitive, &id);
5236     ok(hres == S_OK, "GetDispID failed: %08x\n", hres);
5237     SysFreeString(bstr);
5238
5239     VariantInit(&var);
5240     hres = IDispatchEx_InvokeEx(dispex, id, LOCALE_NEUTRAL, INVOKE_PROPERTYGET, &dp, &var, &ei, NULL);
5241     ok(hres == S_OK, "InvokeEx failed: %08x\n", hres);
5242     ok(V_VT(&var) == VT_DISPATCH, "V_VT(var)=%d\n", V_VT(&var));
5243     ok(V_DISPATCH(&var) != NULL, "V_DISPATCH(var) == NULL\n");
5244     node = get_node_iface((IUnknown*)V_DISPATCH(&var));
5245     type = get_node_type((IUnknown*)node);
5246     ok(type == 3, "type=%d\n", type);
5247     IHTMLDOMNode_Release(node);
5248     VariantClear(&var);
5249
5250     bstr = SysAllocString(w100);
5251     hres = IDispatchEx_GetDispID(dispex, bstr, fdexNameCaseSensitive, &id);
5252     ok(hres == DISP_E_UNKNOWNNAME, "GetDispID failed: %08x, expected DISP_E_UNKNOWNNAME\n", hres);
5253     SysFreeString(bstr);
5254
5255     IDispatchEx_Release(dispex);
5256 }
5257
5258
5259
5260 static void test_elems(IHTMLDocument2 *doc)
5261 {
5262     IHTMLElementCollection *col;
5263     IHTMLDOMChildrenCollection *child_col;
5264     IHTMLElement *elem, *elem2, *elem3;
5265     IHTMLDOMNode *node, *node2;
5266     IHTMLWindow2 *window;
5267     IDispatch *disp;
5268     LONG type;
5269     HRESULT hres;
5270     IHTMLElementCollection *collection;
5271     IHTMLDocument3 *doc3;
5272     BSTR str;
5273
5274     static const elem_type_t all_types[] = {
5275         ET_HTML,
5276         ET_HEAD,
5277         ET_TITLE,
5278         ET_STYLE,
5279         ET_META,
5280         ET_BODY,
5281         ET_COMMENT,
5282         ET_A,
5283         ET_INPUT,
5284         ET_SELECT,
5285         ET_OPTION,
5286         ET_OPTION,
5287         ET_TEXTAREA,
5288         ET_TABLE,
5289         ET_TBODY,
5290         ET_TR,
5291         ET_TR,
5292         ET_TD,
5293         ET_TD,
5294         ET_SCRIPT,
5295         ET_TEST,
5296         ET_OBJECT,
5297         ET_EMBED,
5298         ET_IMG,
5299         ET_IFRAME,
5300         ET_FORM,
5301         ET_DIV
5302     };
5303
5304     static const elem_type_t item_types[] = {
5305         ET_A,
5306         ET_OPTION,
5307         ET_TEXTAREA
5308     };
5309
5310     hres = IHTMLDocument2_get_all(doc, &col);
5311     ok(hres == S_OK, "get_all failed: %08x\n", hres);
5312     test_elem_collection((IUnknown*)col, all_types, sizeof(all_types)/sizeof(all_types[0]));
5313     test_elem_col_item(col, "x", item_types, sizeof(item_types)/sizeof(item_types[0]));
5314     IHTMLElementCollection_Release(col);
5315
5316     hres = IHTMLDocument2_get_images(doc, &collection);
5317     ok(hres == S_OK, "get_images failed: %08x\n", hres);
5318     if(hres == S_OK)
5319     {
5320         static const elem_type_t images_types[] = {ET_IMG};
5321         test_elem_collection((IUnknown*)collection, images_types, 1);
5322
5323         IHTMLElementCollection_Release(collection);
5324     }
5325
5326     hres = IHTMLDocument2_get_links(doc, &collection);
5327     ok(hres == S_OK, "get_links failed: %08x\n", hres);
5328     if(hres == S_OK)
5329     {
5330         static const elem_type_t images_types[] = {ET_A};
5331         test_elem_collection((IUnknown*)collection, images_types, 1);
5332
5333         IHTMLElementCollection_Release(collection);
5334     }
5335
5336     hres = IHTMLDocument2_get_anchors(doc, &collection);
5337     ok(hres == S_OK, "get_anchors failed: %08x\n", hres);
5338     if(hres == S_OK)
5339     {
5340         static const elem_type_t anchor_types[] = {ET_A};
5341         test_elem_collection((IUnknown*)collection, anchor_types, 1);
5342
5343         IHTMLElementCollection_Release(collection);
5344     }
5345
5346     test_plugins_col(doc);
5347
5348     elem = get_doc_elem(doc);
5349     test_elem_all((IUnknown*)elem, all_types+1, sizeof(all_types)/sizeof(all_types[0])-1);
5350     IHTMLElement_Release(elem);
5351
5352     get_elem_by_id(doc, "xxx", FALSE);
5353     elem = get_doc_elem_by_id(doc, "xxx");
5354     ok(!elem, "elem != NULL\n");
5355
5356     elem = get_doc_elem_by_id(doc, "s");
5357     ok(elem != NULL, "elem == NULL\n");
5358     if(elem) {
5359         test_elem_type((IUnknown*)elem, ET_SELECT);
5360         test_elem_attr(elem, "xxx", NULL);
5361         test_elem_attr(elem, "id", "s");
5362         test_elem_class((IUnknown*)elem, NULL);
5363         test_elem_set_class((IUnknown*)elem, "cl");
5364         test_elem_set_class((IUnknown*)elem, NULL);
5365         test_elem_tabindex((IUnknown*)elem, 0);
5366         test_elem_set_tabindex((IUnknown*)elem, 1);
5367         test_elem_filters((IUnknown*)elem);
5368
5369         node = test_node_get_parent((IUnknown*)elem);
5370         ok(node != NULL, "node == NULL\n");
5371         test_node_name((IUnknown*)node, "BODY");
5372         node2 = test_node_get_parent((IUnknown*)node);
5373         IHTMLDOMNode_Release(node);
5374         ok(node2 != NULL, "node == NULL\n");
5375         test_node_name((IUnknown*)node2, "HTML");
5376         node = test_node_get_parent((IUnknown*)node2);
5377         IHTMLDOMNode_Release(node2);
5378         ok(node != NULL, "node == NULL\n");
5379         if (node)
5380         {
5381             test_node_name((IUnknown*)node, "#document");
5382             type = get_node_type((IUnknown*)node);
5383             ok(type == 9, "type=%d, expected 9\n", type);
5384             node2 = test_node_get_parent((IUnknown*)node);
5385             IHTMLDOMNode_Release(node);
5386             ok(node2 == NULL, "node != NULL\n");
5387         }
5388
5389         elem2 = test_elem_get_parent((IUnknown*)elem);
5390         ok(elem2 != NULL, "elem2 == NULL\n");
5391         test_node_name((IUnknown*)elem2, "BODY");
5392
5393         elem3 = test_elem_get_parent((IUnknown*)elem2);
5394         ok(elem3 != NULL, "elem3 == NULL\n");
5395         test_node_name((IUnknown*)elem3, "HTML");
5396
5397         test_elem_contains(elem3, elem2, VARIANT_TRUE);
5398         test_elem_contains(elem3, elem, VARIANT_TRUE);
5399         test_elem_contains(elem2, elem, VARIANT_TRUE);
5400         test_elem_contains(elem2, elem3, VARIANT_FALSE);
5401         test_elem_contains(elem, elem3, VARIANT_FALSE);
5402         test_elem_contains(elem, elem2, VARIANT_FALSE);
5403         test_elem_contains(elem, elem, VARIANT_TRUE);
5404         IHTMLElement_Release(elem2);
5405
5406         elem2 = test_elem_get_parent((IUnknown*)elem3);
5407         ok(elem2 == NULL, "elem2 != NULL\n");
5408         IHTMLElement_Release(elem3);
5409
5410         test_elem_getelembytag((IUnknown*)elem, ET_OPTION, 2, NULL);
5411         test_elem_getelembytag((IUnknown*)elem, ET_SELECT, 0, NULL);
5412         test_elem_getelembytag((IUnknown*)elem, ET_HTML, 0, NULL);
5413
5414         test_elem_innertext(elem, "opt1opt2");
5415
5416         IHTMLElement_Release(elem);
5417     }
5418
5419     elem = get_elem_by_id(doc, "s", TRUE);
5420     if(elem) {
5421         IHTMLSelectElement *select = get_select_iface((IUnknown*)elem);
5422         IHTMLDocument2 *doc_node, *elem_doc;
5423
5424         test_select_elem(select);
5425
5426         test_elem_title((IUnknown*)select, NULL);
5427         test_elem_set_title((IUnknown*)select, "Title");
5428         test_elem_title((IUnknown*)select, "Title");
5429         test_elem_offset((IUnknown*)select, "BODY");
5430         test_elem_bounding_client_rect((IUnknown*)select);
5431
5432         node = get_first_child((IUnknown*)select);
5433         ok(node != NULL, "node == NULL\n");
5434         if(node) {
5435             test_elem_type((IUnknown*)node, ET_OPTION);
5436             IHTMLDOMNode_Release(node);
5437         }
5438
5439         type = get_node_type((IUnknown*)select);
5440         ok(type == 1, "type=%d\n", type);
5441
5442         IHTMLSelectElement_Release(select);
5443
5444         elem_doc = get_elem_doc((IUnknown*)elem);
5445
5446         doc_node = get_doc_node(doc);
5447         ok(iface_cmp((IUnknown*)elem_doc, (IUnknown*)doc_node), "disp != doc\n");
5448         IHTMLDocument2_Release(doc_node);
5449         IHTMLDocument2_Release(elem_doc);
5450
5451         IHTMLElement_Release(elem);
5452     }
5453
5454     elem = get_elem_by_id(doc, "sc", TRUE);
5455     if(elem) {
5456         IHTMLScriptElement *script;
5457         BSTR type;
5458
5459         hres = IHTMLElement_QueryInterface(elem, &IID_IHTMLScriptElement, (void**)&script);
5460         ok(hres == S_OK, "Could not get IHTMLScriptElement interface: %08x\n", hres);
5461
5462         if(hres == S_OK)
5463         {
5464             VARIANT_BOOL vb;
5465
5466             hres = IHTMLScriptElement_put_type (script, NULL);
5467             ok(hres == S_OK, "put_type failed: %08x\n", hres);
5468             hres = IHTMLScriptElement_get_type(script, &type);
5469             ok(hres == S_OK, "get_type failed: %08x\n", hres);
5470             ok(type == NULL, "Unexpected type %s\n", wine_dbgstr_w(type));
5471
5472             hres = IHTMLScriptElement_put_type (script, a2bstr ("text/javascript"));
5473             ok(hres == S_OK, "put_type failed: %08x\n", hres);
5474             hres = IHTMLScriptElement_get_type(script, &type);
5475             ok(hres == S_OK, "get_type failed: %08x\n", hres);
5476             ok(!strcmp_wa(type, "text/javascript"), "Unexpected type %s\n", wine_dbgstr_w(type));
5477
5478             SysFreeString(type);
5479
5480             /* test defer */
5481             hres = IHTMLScriptElement_put_defer(script, VARIANT_TRUE);
5482             ok(hres == S_OK, "put_defer failed: %08x\n", hres);
5483
5484             hres = IHTMLScriptElement_get_defer(script, &vb);
5485             ok(hres == S_OK, "get_defer failed: %08x\n", hres);
5486             ok(vb == VARIANT_TRUE, "get_defer result is %08x\n", hres);
5487
5488             hres = IHTMLScriptElement_put_defer(script, VARIANT_FALSE);
5489             ok(hres == S_OK, "put_defer failed: %08x\n", hres);
5490
5491             str = (BSTR)0xdeadbeef;
5492             hres = IHTMLScriptElement_get_src(script, &str);
5493             ok(hres == S_OK, "get_src failed: %08x\n", hres);
5494             ok(!str, "src = %s\n", wine_dbgstr_w(str));
5495         }
5496
5497         IHTMLScriptElement_Release(script);
5498     }
5499
5500     elem = get_elem_by_id(doc, "in", TRUE);
5501     if(elem) {
5502         IHTMLInputElement *input;
5503
5504         hres = IHTMLElement_QueryInterface(elem, &IID_IHTMLInputElement, (void**)&input);
5505         ok(hres == S_OK, "Could not get IHTMLInputElement: %08x\n", hres);
5506
5507         test_elem_id((IUnknown*)elem, "in");
5508         test_elem_put_id((IUnknown*)elem, "newin");
5509         test_input_get_disabled(input, VARIANT_FALSE);
5510         test_input_set_disabled(input, VARIANT_TRUE);
5511         test_input_set_disabled(input, VARIANT_FALSE);
5512         test_elem3_set_disabled((IUnknown*)input, VARIANT_TRUE);
5513         test_input_get_disabled(input, VARIANT_TRUE);
5514         test_elem3_set_disabled((IUnknown*)input, VARIANT_FALSE);
5515         test_input_get_disabled(input, VARIANT_FALSE);
5516         test_elem_client_size((IUnknown*)elem);
5517         test_input_type(input, "text");
5518
5519         test_node_get_value_str((IUnknown*)elem, NULL);
5520         test_node_put_value_str((IUnknown*)elem, "test");
5521         test_node_get_value_str((IUnknown*)elem, NULL);
5522         test_input_value((IUnknown*)elem, NULL);
5523         test_input_put_value((IUnknown*)elem, "test");
5524         test_elem_class((IUnknown*)elem, "testclass");
5525         test_elem_tabindex((IUnknown*)elem, 2);
5526         test_elem_set_tabindex((IUnknown*)elem, 3);
5527         test_elem_title((IUnknown*)elem, "test title");
5528
5529         test_input_get_defaultchecked(input, VARIANT_FALSE);
5530         test_input_set_defaultchecked(input, VARIANT_TRUE);
5531         test_input_set_defaultchecked(input, VARIANT_FALSE);
5532
5533         test_input_get_checked(input, VARIANT_FALSE);
5534         test_input_set_checked(input, VARIANT_TRUE);
5535         test_input_set_checked(input, VARIANT_FALSE);
5536
5537         test_input_maxlength(input, 0x7fffffff);
5538         test_input_set_maxlength(input, 30);
5539
5540         test_input_name(input, NULL);
5541         test_input_set_name(input, "test");
5542
5543         test_input_src(input, NULL);
5544         test_input_set_src(input, "about:blank");
5545
5546         IHTMLInputElement_Release(input);
5547         IHTMLElement_Release(elem);
5548     }
5549
5550     elem = get_elem_by_id(doc, "imgid", TRUE);
5551     if(elem) {
5552         test_img_src((IUnknown*)elem, "", NULL);
5553         test_img_set_src((IUnknown*)elem, "about:blank");
5554         test_img_src((IUnknown*)elem, "about:blank", NULL);
5555         test_img_alt((IUnknown*)elem, NULL);
5556         test_img_set_alt((IUnknown*)elem, "alt test");
5557         test_img_name((IUnknown*)elem, "WineImg");
5558         IHTMLElement_Release(elem);
5559     }
5560
5561     elem = get_elem_by_id(doc, "attr", TRUE);
5562     if(elem) {
5563         test_dynamic_properties(elem);
5564         test_attr_collection(elem);
5565         IHTMLElement_Release(elem);
5566     }
5567
5568     elem = get_elem_by_id(doc, "styleid", TRUE);
5569     if(elem) {
5570         test_style_media((IUnknown*)elem, NULL);
5571         test_style_put_media((IUnknown*)elem, "screen");
5572         test_style_type((IUnknown*)elem, NULL);
5573         test_style_put_type((IUnknown*)elem, "text/css");
5574         IHTMLElement_Release(elem);
5575     }
5576
5577     elem = get_doc_elem_by_id(doc, "tbl");
5578     ok(elem != NULL, "elem == NULL\n");
5579     if(elem) {
5580         test_table_elem(elem);
5581         IHTMLElement_Release(elem);
5582     }
5583
5584     elem = get_doc_elem_by_id(doc, "row2");
5585     ok(elem != NULL, "elem == NULL\n");
5586     if(elem) {
5587         test_tr_elem(elem);
5588         IHTMLElement_Release(elem);
5589     }
5590
5591     elem = get_doc_elem_by_id(doc, "ifr");
5592     ok(elem != NULL, "elem == NULL\n");
5593     if(elem) {
5594         test_iframe_elem(elem);
5595         IHTMLElement_Release(elem);
5596     }
5597
5598     elem = get_doc_elem_by_id(doc, "objid");
5599     ok(elem != NULL, "elem == NULL\n");
5600     if(elem) {
5601         test_object_vspace((IUnknown*)elem, 100);
5602         IHTMLElement_Release(elem);
5603     }
5604
5605     elem = get_elem_by_id(doc, "a", TRUE);
5606     if(elem) {
5607         test_anchor_href((IUnknown*)elem, "http://test/");
5608
5609         /* Change the href */
5610         test_anchor_put_href((IUnknown*)elem, "http://test1/");
5611         test_anchor_href((IUnknown*)elem, "http://test1/");
5612
5613         /* Restore the href */
5614         test_anchor_put_href((IUnknown*)elem, "http://test/");
5615         test_anchor_href((IUnknown*)elem, "http://test/");
5616
5617         /* target */
5618         test_anchor_get_target((IUnknown*)elem, NULL);
5619
5620         /* Change the target */
5621         test_anchor_put_target((IUnknown*)elem, "wine");
5622         test_anchor_get_target((IUnknown*)elem, "wine");
5623
5624         /* Restore the target */
5625         test_anchor_put_target((IUnknown*)elem, NULL);
5626         test_anchor_get_target((IUnknown*)elem, NULL);
5627
5628         test_anchor_name((IUnknown*)elem, "x");
5629         test_anchor_put_name((IUnknown*)elem, "anchor name");
5630         test_anchor_put_name((IUnknown*)elem, NULL);
5631         test_anchor_put_name((IUnknown*)elem, "x");
5632
5633         IHTMLElement_Release(elem);
5634     }
5635
5636     elem = get_doc_elem_by_id(doc, "metaid");
5637     if(elem) {
5638         test_meta_name((IUnknown*)elem, "meta name");
5639         test_meta_content((IUnknown*)elem, "text/html; charset=utf-8");
5640         test_meta_httpequiv((IUnknown*)elem, "Content-Type");
5641         IHTMLElement_Release(elem);
5642     }
5643
5644     elem = doc_get_body(doc);
5645
5646     node = get_first_child((IUnknown*)elem);
5647     ok(node != NULL, "node == NULL\n");
5648     if(node) {
5649         test_ifaces((IUnknown*)node, text_iids);
5650         test_disp((IUnknown*)node, &DIID_DispHTMLDOMTextNode, "[object]");
5651
5652         node2 = get_first_child((IUnknown*)node);
5653         ok(!node2, "node2 != NULL\n");
5654
5655         type = get_node_type((IUnknown*)node);
5656         ok(type == 3, "type=%d\n", type);
5657
5658         test_node_get_value_str((IUnknown*)node, "text test");
5659         test_node_put_value_str((IUnknown*)elem, "test text");
5660         test_node_get_value_str((IUnknown*)node, "text test");
5661
5662         hres = IHTMLDOMNode_get_attributes(node, &disp);
5663         ok(hres == S_OK, "get_attributes failed: %08x\n", hres);
5664         ok(!disp, "disp != NULL\n");
5665
5666         IHTMLDOMNode_Release(node);
5667     }
5668
5669     child_col = get_child_nodes((IUnknown*)elem);
5670     ok(child_col != NULL, "child_coll == NULL\n");
5671     if(child_col) {
5672         LONG length = 0;
5673
5674         test_disp((IUnknown*)child_col, &DIID_DispDOMChildrenCollection, "[object]");
5675
5676         hres = IHTMLDOMChildrenCollection_get_length(child_col, &length);
5677         ok(hres == S_OK, "get_length failed: %08x\n", hres);
5678         ok(length, "length=0\n");
5679
5680         node2 = NULL;
5681         node = get_child_item(child_col, 0);
5682         ok(node != NULL, "node == NULL\n");
5683         if(node) {
5684             IHTMLDOMNode *prev;
5685
5686             type = get_node_type((IUnknown*)node);
5687             ok(type == 3, "type=%d\n", type);
5688             node2 = node_get_next((IUnknown*)node);
5689
5690             prev = node_get_prev((IUnknown*)node2);
5691             ok(iface_cmp((IUnknown*)node, (IUnknown*)prev), "node != prev\n");
5692             IHTMLDOMNode_Release(prev);
5693
5694             IHTMLDOMNode_Release(node);
5695         }
5696
5697         node = get_child_item(child_col, 1);
5698         ok(node != NULL, "node == NULL\n");
5699         if(node) {
5700             type = get_node_type((IUnknown*)node);
5701             ok(type == 8, "type=%d\n", type);
5702
5703             test_elem_id((IUnknown*)node, NULL);
5704             ok(iface_cmp((IUnknown*)node2, (IUnknown*)node), "node2 != node\n");
5705             IHTMLDOMNode_Release(node2);
5706             IHTMLDOMNode_Release(node);
5707         }
5708
5709         hres = IHTMLDOMChildrenCollection_item(child_col, length - 1, NULL);
5710         ok(hres == E_POINTER, "item failed: %08x, expected E_POINTER\n", hres);
5711
5712         hres = IHTMLDOMChildrenCollection_item(child_col, length, NULL);
5713         ok(hres == E_POINTER, "item failed: %08x, expected E_POINTER\n", hres);
5714
5715         hres = IHTMLDOMChildrenCollection_item(child_col, 6000, &disp);
5716         ok(hres == E_INVALIDARG, "item failed: %08x, expected E_INVALIDARG\n", hres);
5717
5718         hres = IHTMLDOMChildrenCollection_item(child_col, length, &disp);
5719         ok(hres == E_INVALIDARG, "item failed: %08x, expected E_INVALIDARG\n", hres);
5720
5721         test_child_col_disp(child_col);
5722
5723         IHTMLDOMChildrenCollection_Release(child_col);
5724     }
5725
5726     test_elem3_get_disabled((IUnknown*)elem, VARIANT_FALSE);
5727     test_elem3_set_disabled((IUnknown*)elem, VARIANT_TRUE);
5728     test_elem3_set_disabled((IUnknown*)elem, VARIANT_FALSE);
5729
5730     IHTMLElement_Release(elem);
5731
5732     elem = get_doc_elem_by_id(doc, "frm");
5733     ok(elem != NULL, "elem == NULL\n");
5734     if(elem) {
5735         test_form_length((IUnknown*)elem, 0);
5736         test_form_elements((IUnknown*)elem);
5737         IHTMLElement_Release(elem);
5738     }
5739
5740     test_stylesheets(doc);
5741     test_create_option_elem(doc);
5742     test_create_img_elem(doc);
5743
5744     elem = get_doc_elem_by_id(doc, "tbl");
5745     ok(elem != NULL, "elem = NULL\n");
5746     test_elem_set_innertext(elem, "inner text");
5747     IHTMLElement_Release(elem);
5748
5749     test_doc_title(doc, "test");
5750     test_doc_set_title(doc, "test title");
5751     test_doc_title(doc, "test title");
5752
5753     disp = NULL;
5754     hres = IHTMLDocument2_get_Script(doc, &disp);
5755     ok(hres == S_OK, "get_Script failed: %08x\n", hres);
5756     if(hres == S_OK)
5757     {
5758         IDispatchEx *dispex;
5759         hres = IDispatch_QueryInterface(disp, &IID_IDispatchEx, (void**)&dispex);
5760         ok(hres == S_OK, "IDispatch_QueryInterface failed: %08x\n", hres);
5761         if(hres == S_OK)
5762         {
5763             DISPID pid = -1;
5764             BSTR str = a2bstr("Testing");
5765             hres = IDispatchEx_GetDispID(dispex, str, 1, &pid);
5766             ok(hres == S_OK, "GetDispID failed: %08x\n", hres);
5767             ok(pid != -1, "pid == -1\n");
5768             SysFreeString(str);
5769             IDispatchEx_Release(dispex);
5770         }
5771     }
5772     IDispatch_Release(disp);
5773
5774     hres = IHTMLDocument2_QueryInterface(doc, &IID_IHTMLDocument3, (void**)&doc3);
5775     ok(hres == S_OK, "Could not get IHTMLDocument3 iface: %08x\n", hres);
5776
5777     str = a2bstr("img");
5778     hres = IHTMLDocument3_getElementsByTagName(doc3, str, &col);
5779     ok(hres == S_OK, "getElementsByTagName(%s) failed: %08x\n", wine_dbgstr_w(str), hres);
5780     SysFreeString(str);
5781     if(hres == S_OK)
5782     {
5783         static const elem_type_t img_types[] = { ET_IMG };
5784
5785         test_elem_collection((IUnknown*)col, img_types, sizeof(img_types)/sizeof(img_types[0]));
5786         IHTMLElementCollection_Release(col);
5787     }
5788
5789     elem = get_doc_elem_by_id(doc, "y");
5790     test_elem_set_innerhtml((IUnknown*)elem, "inner html");
5791     test_elem_innerhtml((IUnknown*)elem, "inner html");
5792     test_elem_set_innerhtml((IUnknown*)elem, "");
5793     test_elem_innerhtml((IUnknown*)elem, NULL);
5794     node = node_get_next((IUnknown*)elem);
5795     ok(!node, "node = %p\n", node);
5796
5797     elem2 = get_doc_elem_by_id(doc, "x");
5798     test_elem_tag((IUnknown*)elem2, "A");
5799     node = node_get_next((IUnknown*)elem2);
5800     IHTMLDOMNode_Release(node);
5801     IHTMLElement_Release(elem2);
5802     IHTMLElement_Release(elem);
5803
5804     IHTMLDocument3_Release(doc3);
5805
5806     elem = get_elem_by_id(doc, "s", TRUE);
5807     if(elem) {
5808         static const elem_type_t select_types[] = { ET_OPTION, ET_OPTION, ET_OPTION };
5809
5810         test_select_put_length((IUnknown*)elem, 3);
5811         test_elem_all((IUnknown*)elem, select_types, sizeof(select_types)/sizeof(*select_types));
5812         test_select_put_length((IUnknown*)elem, 1);
5813         test_elem_all((IUnknown*)elem, select_types, 1);
5814         IHTMLElement_Release(elem);
5815     }
5816
5817     window = get_doc_window(doc);
5818     test_window_name(window, NULL);
5819     set_window_name(window, "test name");
5820     test_window_length(window, 1);
5821     IHTMLWindow2_Release(window);
5822 }
5823
5824 static void test_attr(IHTMLElement *elem)
5825 {
5826     IHTMLDOMAttribute *attr, *attr2;
5827     VARIANT v;
5828
5829     get_elem_attr_node((IUnknown*)elem, "noattr", FALSE);
5830
5831     attr = get_elem_attr_node((IUnknown*)elem, "id", TRUE);
5832
5833     test_disp((IUnknown*)attr, &DIID_DispHTMLDOMAttribute, "[object]");
5834     test_ifaces((IUnknown*)attr, attr_iids);
5835     test_no_iface((IUnknown*)attr, &IID_IHTMLDOMNode);
5836     test_attr_specified(attr, VARIANT_TRUE);
5837
5838     attr2 = get_elem_attr_node((IUnknown*)elem, "id", TRUE);
5839     ok(iface_cmp((IUnknown*)attr, (IUnknown*)attr2), "attr != attr2\n");
5840     IHTMLDOMAttribute_Release(attr2);
5841
5842     get_attr_node_value(attr, &v, VT_BSTR);
5843     ok(!strcmp_wa(V_BSTR(&v), "divid"), "V_BSTR(v) = %s\n", wine_dbgstr_w(V_BSTR(&v)));
5844     VariantClear(&v);
5845
5846     IHTMLDOMAttribute_Release(attr);
5847
5848     attr = get_elem_attr_node((IUnknown*)elem, "emptyattr", TRUE);
5849     get_attr_node_value(attr, &v, VT_BSTR);
5850     ok(!V_BSTR(&v), "V_BSTR(v) = %s\n", wine_dbgstr_w(V_BSTR(&v)));
5851     VariantClear(&v);
5852     test_attr_specified(attr, VARIANT_TRUE);
5853     IHTMLDOMAttribute_Release(attr);
5854
5855     V_VT(&v) = VT_I4;
5856     V_I4(&v) = 100;
5857     set_dispex_value((IUnknown*)elem, "dispprop", &v);
5858     attr = get_elem_attr_node((IUnknown*)elem, "dispprop", TRUE);
5859     get_attr_node_value(attr, &v, VT_I4);
5860     ok(V_I4(&v) == 100, "V_I4(v) = %d\n", V_I4(&v));
5861     test_attr_specified(attr, VARIANT_TRUE);
5862     IHTMLDOMAttribute_Release(attr);
5863
5864     attr = get_elem_attr_node((IUnknown*)elem, "tabIndex", TRUE);
5865     test_attr_specified(attr, VARIANT_FALSE);
5866     IHTMLDOMAttribute_Release(attr);
5867 }
5868
5869 static void test_blocked(IHTMLDocument2 *doc, IHTMLElement *outer_elem)
5870 {
5871     IHTMLElement *elem;
5872
5873     test_elem_set_innerhtml((IUnknown*)outer_elem,
5874             "<img id=\"imgid\" src=\"BLOCKED::http://www.winehq.org/img.png\" />");
5875     elem = get_elem_by_id(doc, "imgid", TRUE);
5876     if(elem) {
5877         test_img_src((IUnknown*)elem, "BLOCKED::", "blocked::http://www.winehq.org/img.png");
5878         IHTMLElement_Release(elem);
5879     }
5880
5881     test_elem_set_innerhtml((IUnknown*)outer_elem,
5882             "<img id=\"imgid\" src=\"BLOCKE::http://www.winehq.org/img.png\" />");
5883     elem = get_elem_by_id(doc, "imgid", TRUE);
5884     if(elem) {
5885         test_img_src((IUnknown*)elem, "blocke::http://www.winehq.org/img.png", NULL);
5886         test_img_set_src((IUnknown*)elem, "BLOCKED:http://www.winehq.org/img.png");
5887         test_img_src((IUnknown*)elem, "blocked:http://www.winehq.org/img.png", NULL);
5888         test_img_set_src((IUnknown*)elem, "blocked::http://www.winehq.org/img.png");
5889         test_img_src((IUnknown*)elem, "BLOCKED::", "blocked::http://www.winehq.org/img.png");
5890         IHTMLElement_Release(elem);
5891     }
5892 }
5893
5894 static void test_elems2(IHTMLDocument2 *doc)
5895 {
5896     IHTMLElement *elem, *elem2, *div;
5897
5898     static const elem_type_t outer_types[] = {
5899         ET_BR,
5900         ET_A
5901     };
5902
5903     div = get_doc_elem_by_id(doc, "divid");
5904
5905     test_elem_set_innerhtml((IUnknown*)div, "<div id=\"innerid\"></div>");
5906     elem2 = get_doc_elem_by_id(doc, "innerid");
5907     ok(elem2 != NULL, "elem2 == NULL\n");
5908     test_elem_set_outerhtml((IUnknown*)elem2, "<br><a href=\"about:blank\" id=\"aid\">a</a>");
5909     test_elem_all((IUnknown*)div, outer_types, sizeof(outer_types)/sizeof(*outer_types));
5910     IHTMLElement_Release(elem2);
5911
5912     elem2 = get_doc_elem_by_id(doc, "aid");
5913     ok(elem2 != NULL, "elem2 == NULL\n");
5914     test_elem_set_outerhtml((IUnknown*)elem2, "");
5915     test_elem_all((IUnknown*)div, outer_types, 1);
5916     IHTMLElement_Release(elem2);
5917
5918     test_elem_set_innerhtml((IUnknown*)div, "<textarea id=\"ta\"></textarea>");
5919     elem = get_elem_by_id(doc, "ta", TRUE);
5920     if(elem) {
5921         test_textarea_value((IUnknown*)elem, NULL);
5922         test_textarea_put_value((IUnknown*)elem, "test");
5923         test_textarea_readonly((IUnknown*)elem, VARIANT_FALSE);
5924         test_textarea_put_readonly((IUnknown*)elem, VARIANT_TRUE);
5925         test_textarea_put_readonly((IUnknown*)elem, VARIANT_FALSE);
5926         test_textarea_type((IUnknown*)elem);
5927         IHTMLElement_Release(elem);
5928     }
5929
5930     test_elem_set_innerhtml((IUnknown*)div,
5931             "<form id=\"form\"><input type=\"button\" /><div><input type=\"text\" /></div></textarea>");
5932     elem = get_elem_by_id(doc, "form", TRUE);
5933     if(elem) {
5934         test_form_length((IUnknown*)elem, 2);
5935         test_form_item(elem);
5936         test_form_action((IUnknown*)elem, NULL);
5937         test_form_put_action((IUnknown*)elem, "about:blank");
5938         test_form_method((IUnknown*)elem, "get");
5939         test_form_put_method((IUnknown*)elem, S_OK, "post");
5940         test_form_put_method((IUnknown*)elem, E_INVALIDARG, "put");
5941         test_form_method((IUnknown*)elem, "post");
5942         test_form_name((IUnknown*)elem, NULL);
5943         test_form_put_name((IUnknown*)elem, "Name");
5944         test_form_encoding((IUnknown*)elem, "application/x-www-form-urlencoded");
5945         test_form_put_encoding((IUnknown*)elem, S_OK, "text/plain");
5946         test_form_put_encoding((IUnknown*)elem, S_OK, "multipart/form-data");
5947         test_form_put_encoding((IUnknown*)elem, E_INVALIDARG, "image/png");
5948         test_form_encoding((IUnknown*)elem, "multipart/form-data");
5949         test_form_elements((IUnknown*)elem);
5950         IHTMLElement_Release(elem);
5951     }
5952
5953     test_attr(div);
5954     test_blocked(doc, div);
5955
5956     IHTMLElement_Release(div);
5957 }
5958
5959 static void test_create_elems(IHTMLDocument2 *doc)
5960 {
5961     IHTMLElement *elem, *body, *elem2;
5962     IHTMLDOMNode *node, *node2, *node3, *comment;
5963     IHTMLDocument5 *doc5;
5964     IDispatch *disp;
5965     VARIANT var;
5966     LONG type;
5967     HRESULT hres;
5968     BSTR str;
5969
5970     static const elem_type_t types1[] = { ET_TESTG };
5971
5972     elem = test_create_elem(doc, "TEST");
5973     test_elem_tag((IUnknown*)elem, "TEST");
5974     type = get_node_type((IUnknown*)elem);
5975     ok(type == 1, "type=%d\n", type);
5976     test_ifaces((IUnknown*)elem, elem_iids);
5977     test_disp((IUnknown*)elem, &DIID_DispHTMLGenericElement, "[object]");
5978
5979     body = doc_get_body(doc);
5980     test_node_has_child((IUnknown*)body, VARIANT_FALSE);
5981
5982     node = test_node_append_child((IUnknown*)body, (IUnknown*)elem);
5983     test_node_has_child((IUnknown*)body, VARIANT_TRUE);
5984     elem2 = get_elem_iface((IUnknown*)node);
5985     IHTMLElement_Release(elem2);
5986
5987     hres = IHTMLElement_get_all(body, &disp);
5988     ok(hres == S_OK, "get_all failed: %08x\n", hres);
5989     test_elem_collection((IUnknown*)disp, types1, sizeof(types1)/sizeof(types1[0]));
5990     IDispatch_Release(disp);
5991
5992     test_node_remove_child((IUnknown*)body, node);
5993
5994     hres = IHTMLElement_get_all(body, &disp);
5995     ok(hres == S_OK, "get_all failed: %08x\n", hres);
5996     test_elem_collection((IUnknown*)disp, NULL, 0);
5997     IDispatch_Release(disp);
5998     test_node_has_child((IUnknown*)body, VARIANT_FALSE);
5999
6000     IHTMLElement_Release(elem);
6001     IHTMLDOMNode_Release(node);
6002
6003     node = test_create_text(doc, "test");
6004     test_ifaces((IUnknown*)node, text_iids);
6005     test_disp((IUnknown*)node, &DIID_DispHTMLDOMTextNode, "[object]");
6006     test_text_length((IUnknown*)node, 4);
6007
6008     V_VT(&var) = VT_NULL;
6009     node2 = test_node_insertbefore((IUnknown*)body, node, &var);
6010     IHTMLDOMNode_Release(node);
6011
6012     node = test_create_text(doc, "insert ");
6013
6014     V_VT(&var) = VT_DISPATCH;
6015     V_DISPATCH(&var) = (IDispatch*)node2;
6016     node3 = test_node_insertbefore((IUnknown*)body, node, &var);
6017     IHTMLDOMNode_Release(node);
6018     IHTMLDOMNode_Release(node2);
6019     IHTMLDOMNode_Release(node3);
6020
6021     test_elem_innertext(body, "insert test");
6022     test_elem_innerhtml((IUnknown*)body, "insert test");
6023
6024     node = test_create_text(doc, " Test");
6025     V_VT(&var) = VT_DISPATCH;
6026     V_DISPATCH(&var) = NULL;
6027     test_node_insertbefore((IUnknown*)body, node, &var);
6028     test_elem_innertext(body, "insert test Test");
6029     IHTMLDOMNode_Release(node);
6030
6031     hres = IHTMLDocument2_QueryInterface(doc, &IID_IHTMLDocument5, (void**)&doc5);
6032     if(hres == S_OK)
6033     {
6034         str = a2bstr("testing");
6035         hres = IHTMLDocument5_createComment(doc5, str, &comment);
6036         SysFreeString(str);
6037         ok(hres == S_OK, "createComment failed: %08x\n", hres);
6038         if(hres == S_OK)
6039         {
6040             type = get_node_type((IUnknown*)comment);
6041             ok(type == 8, "type=%d, expected 8\n", type);
6042
6043             test_node_get_value_str((IUnknown*)comment, "testing");
6044             test_elem_title((IUnknown*)comment, NULL);
6045             test_elem_set_title((IUnknown*)comment, "comment title");
6046             test_elem_title((IUnknown*)comment, "comment title");
6047             test_comment_text((IUnknown*)comment, "<!--testing-->");
6048             test_elem_outerhtml((IUnknown*)comment, "<!--testing-->");
6049             test_comment_attrs((IUnknown*)comment);
6050
6051             IHTMLDOMNode_Release(comment);
6052         }
6053
6054         IHTMLDocument5_Release(doc5);
6055     }
6056
6057     IHTMLElement_Release(body);
6058 }
6059
6060 static void test_replacechild_elems(IHTMLDocument2 *doc)
6061 {
6062     IHTMLElement *body;
6063     IHTMLDOMNode *node, *node2, *node3;
6064     IHTMLDOMNode *nodeBody, *nodeNew;
6065     HRESULT hres;
6066     VARIANT var;
6067
6068     body = doc_get_body(doc);
6069
6070     node = test_create_text(doc, "insert");
6071
6072     V_VT(&var) = VT_NULL;
6073     V_DISPATCH(&var) = NULL;
6074     node2 = test_node_insertbefore((IUnknown*)body, node, &var);
6075     IHTMLDOMNode_Release(node);
6076
6077     test_elem_innertext(body, "insert");
6078
6079     node3 = test_create_text(doc, "replaced");
6080
6081     nodeBody = _get_node_iface(__LINE__, (IUnknown *)body);
6082
6083     hres = IHTMLDOMNode_replaceChild(nodeBody, node3, node2, &nodeNew);
6084     ok(hres == S_OK, "Expected S_OK, got 0x%08x\n", hres);
6085
6086     test_elem_innertext(body, "replaced");
6087
6088     IHTMLDOMNode_Release(node2);
6089     IHTMLDOMNode_Release(node3);
6090     IHTMLDOMNode_Release(nodeBody);
6091
6092     IHTMLElement_Release(body);
6093 }
6094
6095 static void test_noscript(IHTMLDocument2 *doc)
6096 {
6097     IHTMLElementCollection *col;
6098     IHTMLElement *body;
6099     HRESULT hres;
6100
6101     static const elem_type_t all_types[] = {
6102         ET_HTML,
6103         ET_HEAD,
6104         ET_TITLE,
6105         ET_NOSCRIPT,
6106         ET_BODY,
6107         ET_NOSCRIPT
6108     };
6109
6110     static const elem_type_t body_all_types[] = {
6111         ET_DIV,
6112         ET_NOSCRIPT
6113     };
6114
6115     hres = IHTMLDocument2_get_all(doc, &col);
6116     ok(hres == S_OK, "get_all failed: %08x\n", hres);
6117     test_elem_collection((IUnknown*)col, all_types, sizeof(all_types)/sizeof(all_types[0]));
6118     IHTMLElementCollection_Release(col);
6119
6120     body = doc_get_body(doc);
6121     test_elem_set_innerhtml((IUnknown*)body, "<div>test</div><noscript><a href=\"about:blank\">A</a></noscript>");
6122     test_elem_all((IUnknown*)body, body_all_types, sizeof(body_all_types)/sizeof(*body_all_types));
6123     IHTMLElement_Release(body);
6124 }
6125
6126 static void test_null_write(IHTMLDocument2 *doc)
6127 {
6128     HRESULT hres;
6129
6130     doc_write(doc, FALSE, NULL);
6131     doc_write(doc, TRUE, NULL);
6132
6133     hres = IHTMLDocument2_write(doc, NULL);
6134     ok(hres == S_OK,
6135        "Expected IHTMLDocument2::write to return S_OK, got 0x%08x\n", hres);
6136
6137     hres = IHTMLDocument2_writeln(doc, NULL);
6138     ok(hres == S_OK,
6139        "Expected IHTMLDocument2::writeln to return S_OK, got 0x%08x\n", hres);
6140 }
6141
6142 static void test_create_stylesheet(IHTMLDocument2 *doc)
6143 {
6144     IHTMLStyleSheet *stylesheet, *stylesheet2;
6145     IHTMLStyleElement *style_elem;
6146     IHTMLElement *doc_elem, *elem;
6147     HRESULT hres;
6148
6149     static const elem_type_t all_types[] = {
6150         ET_HTML,
6151         ET_HEAD,
6152         ET_TITLE,
6153         ET_BODY,
6154         ET_DIV
6155     };
6156
6157     static const elem_type_t all_types2[] = {
6158         ET_HTML,
6159         ET_HEAD,
6160         ET_TITLE,
6161         ET_STYLE,
6162         ET_BODY,
6163         ET_DIV
6164     };
6165
6166     test_doc_all(doc, all_types, sizeof(all_types)/sizeof(*all_types));
6167
6168     hres = IHTMLDocument2_createStyleSheet(doc, NULL, -1, &stylesheet);
6169     ok(hres == S_OK, "createStyleSheet failed: %08x\n", hres);
6170
6171     test_doc_all(doc, all_types2, sizeof(all_types2)/sizeof(*all_types2));
6172
6173     doc_elem = get_doc_elem(doc);
6174
6175     test_elem_getelembytag((IUnknown*)doc_elem, ET_STYLE, 1, &elem);
6176     IHTMLElement_Release(doc_elem);
6177
6178     hres = IHTMLElement_QueryInterface(elem, &IID_IHTMLStyleElement, (void**)&style_elem);
6179     IHTMLElement_Release(elem);
6180     ok(hres == S_OK, "Could not get IHTMLStyleElement iface: %08x\n", hres);
6181
6182     stylesheet2 = NULL;
6183     hres = IHTMLStyleElement_get_styleSheet(style_elem, &stylesheet2);
6184     ok(hres == S_OK, "get_styleSheet failed: %08x\n", hres);
6185     ok(stylesheet2 != NULL, "stylesheet2 == NULL\n");
6186     ok(iface_cmp((IUnknown*)stylesheet, (IUnknown*)stylesheet2), "stylesheet != stylesheet2\n");
6187
6188     IHTMLStyleSheet_Release(stylesheet2);
6189     IHTMLStyleSheet_Release(stylesheet);
6190
6191     IHTMLStyleElement_Release(style_elem);
6192 }
6193
6194 static void test_exec(IUnknown *unk, const GUID *grpid, DWORD cmdid, VARIANT *in, VARIANT *out)
6195 {
6196     IOleCommandTarget *cmdtrg;
6197     HRESULT hres;
6198
6199     hres = IUnknown_QueryInterface(unk, &IID_IOleCommandTarget, (void**)&cmdtrg);
6200     ok(hres == S_OK, "Could not get IOleCommandTarget interface: %08x\n", hres);
6201
6202     hres = IOleCommandTarget_Exec(cmdtrg, grpid, cmdid, 0, in, out);
6203     ok(hres == S_OK, "Exec failed: %08x\n", hres);
6204
6205     IOleCommandTarget_Release(cmdtrg);
6206 }
6207
6208 static void test_indent(IHTMLDocument2 *doc)
6209 {
6210     IHTMLElementCollection *col;
6211     IHTMLTxtRange *range;
6212     HRESULT hres;
6213
6214     static const elem_type_t all_types[] = {
6215         ET_HTML,
6216         ET_HEAD,
6217         ET_TITLE,
6218         ET_BODY,
6219         ET_BR,
6220         ET_A,
6221     };
6222
6223     static const elem_type_t indent_types[] = {
6224         ET_HTML,
6225         ET_HEAD,
6226         ET_TITLE,
6227         ET_BODY,
6228         ET_BLOCKQUOTE,
6229         ET_P,
6230         ET_BR,
6231         ET_A,
6232     };
6233
6234     hres = IHTMLDocument2_get_all(doc, &col);
6235     ok(hres == S_OK, "get_all failed: %08x\n", hres);
6236     test_elem_collection((IUnknown*)col, all_types, sizeof(all_types)/sizeof(all_types[0]));
6237     IHTMLElementCollection_Release(col);
6238
6239     range = test_create_body_range(doc);
6240     test_exec((IUnknown*)range, &CGID_MSHTML, IDM_INDENT, NULL, NULL);
6241     IHTMLTxtRange_Release(range);
6242
6243     hres = IHTMLDocument2_get_all(doc, &col);
6244     ok(hres == S_OK, "get_all failed: %08x\n", hres);
6245     test_elem_collection((IUnknown*)col, indent_types, sizeof(indent_types)/sizeof(indent_types[0]));
6246     IHTMLElementCollection_Release(col);
6247 }
6248
6249 static void test_cond_comment(IHTMLDocument2 *doc)
6250 {
6251     IHTMLElementCollection *col;
6252     HRESULT hres;
6253
6254     static const elem_type_t all_types[] = {
6255         ET_HTML,
6256         ET_HEAD,
6257         ET_TITLE,
6258         ET_BODY,
6259         ET_BR
6260     };
6261
6262     hres = IHTMLDocument2_get_all(doc, &col);
6263     ok(hres == S_OK, "get_all failed: %08x\n", hres);
6264     test_elem_collection((IUnknown*)col, all_types, sizeof(all_types)/sizeof(all_types[0]));
6265     IHTMLElementCollection_Release(col);
6266 }
6267
6268 static void test_frame(IDispatch *disp, const char *exp_id)
6269 {
6270     IHTMLWindow2 *frame2, *parent, *top;
6271     IHTMLDocument2 *parent_doc, *top_doc;
6272     IHTMLWindow4 *frame;
6273     IHTMLFrameBase *frame_elem;
6274     HRESULT hres;
6275
6276     hres = IDispatch_QueryInterface(disp, &IID_IHTMLWindow4, (void**)&frame);
6277     ok(hres == S_OK, "Could not get IHTMLWindow4 interface: 0x%08x\n", hres);
6278     if(FAILED(hres))
6279         return;
6280
6281     hres = IHTMLWindow4_get_frameElement(frame, &frame_elem);
6282     ok(hres == S_OK, "IHTMLWindow4_get_frameElement failed: 0x%08x\n", hres);
6283     IHTMLWindow4_Release(frame);
6284     if(FAILED(hres))
6285         return;
6286
6287     test_elem_type((IUnknown*)frame_elem, ET_FRAME);
6288     test_frame_doc((IUnknown*)frame_elem, FALSE);
6289     test_elem_id((IUnknown*)frame_elem, exp_id);
6290     IHTMLFrameBase_Release(frame_elem);
6291
6292     hres = IDispatch_QueryInterface(disp, &IID_IHTMLWindow2, (void**)&frame2);
6293     ok(hres == S_OK, "Could not get IHTMLWindow2 interface: 0x%08x\n", hres);
6294     if(FAILED(hres))
6295         return;
6296
6297     hres = IHTMLWindow2_get_parent(frame2, &parent);
6298     ok(hres == S_OK, "IHTMLWindow2_get_parent failed: 0x%08x\n", hres);
6299     if(FAILED(hres)){
6300         IHTMLWindow2_Release(frame2);
6301         return;
6302     }
6303
6304     hres = IHTMLWindow2_get_document(parent, &parent_doc);
6305     ok(hres == S_OK, "IHTMLWindow2_get_document failed: 0x%08x\n", hres);
6306     IHTMLWindow2_Release(parent);
6307     if(FAILED(hres)){
6308         IHTMLWindow2_Release(frame2);
6309         return;
6310     }
6311
6312     test_doc_title(parent_doc, "frameset test");
6313     IHTMLDocument2_Release(parent_doc);
6314
6315     /* test get_top */
6316     hres = IHTMLWindow2_get_top(frame2, &top);
6317     ok(hres == S_OK, "IHTMLWindow2_get_top failed: 0x%08x\n", hres);
6318     IHTMLWindow2_Release(frame2);
6319     if(FAILED(hres))
6320         return;
6321
6322     hres = IHTMLWindow2_get_document(top, &top_doc);
6323     ok(hres == S_OK, "IHTMLWindow2_get_document failed: 0x%08x\n", hres);
6324     IHTMLWindow2_Release(top);
6325     if(FAILED(hres))
6326         return;
6327
6328     test_doc_title(top_doc, "frameset test");
6329     IHTMLDocument2_Release(top_doc);
6330 }
6331
6332 static void test_frames_collection(IHTMLFramesCollection2 *frames, const char *frid)
6333 {
6334     VARIANT index_var, result_var;
6335     LONG length;
6336     HRESULT hres;
6337
6338     /* test result length */
6339     hres = IHTMLFramesCollection2_get_length(frames, &length);
6340     ok(hres == S_OK, "IHTMLFramesCollection2_get_length failed: 0x%08x\n", hres);
6341     ok(length == 3, "IHTMLFramesCollection2_get_length should have been 3, was: %d\n", length);
6342
6343     /* test first frame */
6344     V_VT(&index_var) = VT_I4;
6345     V_I4(&index_var) = 0;
6346     hres = IHTMLFramesCollection2_item(frames, &index_var, &result_var);
6347     ok(hres == S_OK, "IHTMLFramesCollection2_item failed: 0x%08x\n", hres);
6348     if(SUCCEEDED(hres)) {
6349         ok(V_VT(&result_var) == VT_DISPATCH, "result type should have been VT_DISPATCH, was: 0x%x\n", V_VT(&result_var));
6350         test_frame((IDispatch*)V_DISPATCH(&result_var), "fr1");
6351     }
6352     VariantClear(&result_var);
6353
6354     /* test second frame */
6355     V_I4(&index_var) = 1;
6356     hres = IHTMLFramesCollection2_item(frames, &index_var, &result_var);
6357     ok(hres == S_OK, "IHTMLFramesCollection2_item failed: 0x%08x\n", hres);
6358     if(SUCCEEDED(hres)) {
6359         ok(V_VT(&result_var) == VT_DISPATCH, "result type should have been VT_DISPATCH, was: 0x%x\n", V_VT(&result_var));
6360         test_frame((IDispatch*)V_DISPATCH(&result_var), "fr2");
6361     }
6362     VariantClear(&result_var);
6363
6364     /* fail on next frame */
6365     V_I4(&index_var) = 3;
6366     hres = IHTMLFramesCollection2_item(frames, &index_var, &result_var);
6367     ok(hres == DISP_E_MEMBERNOTFOUND, "IHTMLFramesCollection2_item should have"
6368            "failed with DISP_E_MEMBERNOTFOUND, instead: 0x%08x\n", hres);
6369     VariantClear(&result_var);
6370
6371     /* string argument (element id lookup) */
6372     V_VT(&index_var) = VT_BSTR;
6373     V_BSTR(&index_var) = a2bstr(frid);
6374     hres = IHTMLFramesCollection2_item(frames, &index_var, &result_var);
6375     ok(hres == S_OK, "IHTMLFramesCollection2_item failed: 0x%08x\n", hres);
6376     if(SUCCEEDED(hres)) {
6377         ok(V_VT(&result_var) == VT_DISPATCH, "result type should have been VT_DISPATCH, was: 0x%x\n", V_VT(&result_var));
6378         test_frame(V_DISPATCH(&result_var), frid);
6379     }
6380     VariantClear(&result_var);
6381     VariantClear(&index_var);
6382
6383     /* invalid argument */
6384     V_VT(&index_var) = VT_BOOL;
6385     V_BOOL(&index_var) = VARIANT_TRUE;
6386     hres = IHTMLFramesCollection2_item(frames, &index_var, &result_var);
6387     ok(hres == E_INVALIDARG, "IHTMLFramesCollection2_item should have"
6388            "failed with E_INVALIDARG, instead: 0x%08x\n", hres);
6389     VariantClear(&result_var);
6390 }
6391
6392 static void test_frameset(IHTMLDocument2 *doc)
6393 {
6394     IHTMLWindow2 *window;
6395     IHTMLFramesCollection2 *frames;
6396     IHTMLElement *elem;
6397     HRESULT hres;
6398
6399     window = get_doc_window(doc);
6400
6401     /* test using IHTMLFramesCollection object */
6402
6403     hres = IHTMLWindow2_get_frames(window, &frames);
6404     ok(hres == S_OK, "IHTMLWindow2_get_frames failed: 0x%08x\n", hres);
6405     if(FAILED(hres))
6406         return;
6407
6408     test_frames_collection(frames, "fr1");
6409     IHTMLFramesCollection2_Release(frames);
6410
6411     hres = IHTMLDocument2_get_frames(doc, &frames);
6412     ok(hres == S_OK, "IHTMLDocument2_get_frames failed: 0x%08x\n", hres);
6413     if(FAILED(hres))
6414         return;
6415
6416     test_frames_collection(frames, "fr1");
6417     IHTMLFramesCollection2_Release(frames);
6418
6419     /* test using IHTMLWindow2 inheritance */
6420     test_frames_collection((IHTMLFramesCollection2*)window, "fr2");
6421
6422     /* getElementById with node name attributes */
6423     elem = get_doc_elem_by_id(doc, "nm1");
6424     test_elem_id((IUnknown*)elem, "fr1");
6425
6426     test_framebase((IUnknown*)elem);
6427     test_framebase_name(elem, "nm1");
6428     test_framebase_put_name(elem, "frame name");
6429     test_framebase_put_name(elem, NULL);
6430     test_framebase_put_name(elem, "nm1");
6431     IHTMLElement_Release(elem);
6432
6433     /* get_name with no name attr */
6434     elem = get_doc_elem_by_id(doc, "fr3");
6435     test_framebase_name(elem, NULL);
6436     test_framebase_put_name(elem, "frame name");
6437     test_framebase_put_name(elem, NULL);
6438     IHTMLElement_Release(elem);
6439
6440     IHTMLWindow2_Release(window);
6441 }
6442
6443 static IHTMLDocument2 *create_docfrag(IHTMLDocument2 *doc)
6444 {
6445     IHTMLDocument2 *frag;
6446     IHTMLDocument3 *doc3;
6447     HRESULT hres;
6448
6449     hres = IHTMLDocument2_QueryInterface(doc, &IID_IHTMLDocument3, (void**)&doc3);
6450     ok(hres == S_OK, "Could not get IHTMLDocument3 iface: %08x\n", hres);
6451
6452     hres = IHTMLDocument3_createDocumentFragment(doc3, &frag);
6453     IHTMLDocument3_Release(doc3);
6454     ok(hres == S_OK, "createDocumentFragment failed: %08x\n", hres);
6455     ok(frag != NULL, "frag == NULL\n");
6456
6457     return frag;
6458 }
6459
6460 static void test_docfrag(IHTMLDocument2 *doc)
6461 {
6462     IHTMLDocument2 *frag, *owner_doc, *doc_node;
6463     IHTMLElement *div, *body, *br;
6464     IHTMLElementCollection *col;
6465     IHTMLLocation *location;
6466     HRESULT hres;
6467
6468     static const elem_type_t all_types[] = {
6469         ET_HTML,
6470         ET_HEAD,
6471         ET_TITLE,
6472         ET_BODY,
6473         ET_DIV,
6474         ET_BR
6475     };
6476
6477     frag = create_docfrag(doc);
6478
6479     test_disp((IUnknown*)frag, &DIID_DispHTMLDocument, "[object]");
6480
6481     body = (void*)0xdeadbeef;
6482     hres = IHTMLDocument2_get_body(frag, &body);
6483     ok(hres == S_OK, "get_body failed: %08x\n", hres);
6484     ok(!body, "body != NULL\n");
6485
6486     location = (void*)0xdeadbeef;
6487     hres = IHTMLDocument2_get_location(frag, &location);
6488     ok(hres == E_UNEXPECTED, "get_location failed: %08x\n", hres);
6489     ok(location == (void*)0xdeadbeef, "location changed\n");
6490
6491     br = test_create_elem(doc, "BR");
6492     test_node_append_child((IUnknown*)frag, (IUnknown*)br);
6493     IHTMLElement_Release(br);
6494
6495     div = get_elem_by_id(doc, "divid", TRUE);
6496     test_node_append_child((IUnknown*)div, (IUnknown*)frag);
6497     IHTMLElement_Release(div);
6498
6499     hres = IHTMLDocument2_get_all(doc, &col);
6500     ok(hres == S_OK, "get_all failed: %08x\n", hres);
6501     test_elem_collection((IUnknown*)col, all_types, sizeof(all_types)/sizeof(all_types[0]));
6502     IHTMLElementCollection_Release(col);
6503
6504     div = test_create_elem(frag, "div");
6505     owner_doc = get_owner_doc((IUnknown*)div);
6506     doc_node = get_doc_node(doc);
6507     ok(iface_cmp((IUnknown*)owner_doc, (IUnknown*)doc_node), "owner_doc != doc_node\n");
6508     IHTMLDocument2_Release(doc_node);
6509     IHTMLDocument2_Release(owner_doc);
6510     IHTMLElement_Release(div);
6511
6512     IHTMLDocument2_Release(frag);
6513 }
6514
6515 static IHTMLDocument2 *notif_doc;
6516 static BOOL doc_complete;
6517
6518 static HRESULT WINAPI PropertyNotifySink_QueryInterface(IPropertyNotifySink *iface,
6519         REFIID riid, void**ppv)
6520 {
6521     if(IsEqualGUID(&IID_IPropertyNotifySink, riid)) {
6522         *ppv = iface;
6523         return S_OK;
6524     }
6525
6526     ok(0, "unexpected call\n");
6527     return E_NOINTERFACE;
6528 }
6529
6530 static ULONG WINAPI PropertyNotifySink_AddRef(IPropertyNotifySink *iface)
6531 {
6532     return 2;
6533 }
6534
6535 static ULONG WINAPI PropertyNotifySink_Release(IPropertyNotifySink *iface)
6536 {
6537     return 1;
6538 }
6539
6540 static HRESULT WINAPI PropertyNotifySink_OnChanged(IPropertyNotifySink *iface, DISPID dispID)
6541 {
6542     if(dispID == DISPID_READYSTATE){
6543         BSTR state;
6544         HRESULT hres;
6545
6546         hres = IHTMLDocument2_get_readyState(notif_doc, &state);
6547         ok(hres == S_OK, "get_readyState failed: %08x\n", hres);
6548
6549         if(!strcmp_wa(state, "complete"))
6550             doc_complete = TRUE;
6551
6552         SysFreeString(state);
6553     }
6554
6555     return S_OK;
6556 }
6557
6558 static HRESULT WINAPI PropertyNotifySink_OnRequestEdit(IPropertyNotifySink *iface, DISPID dispID)
6559 {
6560     ok(0, "unexpected call\n");
6561     return E_NOTIMPL;
6562 }
6563
6564 static IPropertyNotifySinkVtbl PropertyNotifySinkVtbl = {
6565     PropertyNotifySink_QueryInterface,
6566     PropertyNotifySink_AddRef,
6567     PropertyNotifySink_Release,
6568     PropertyNotifySink_OnChanged,
6569     PropertyNotifySink_OnRequestEdit
6570 };
6571
6572 static IPropertyNotifySink PropertyNotifySink = { &PropertyNotifySinkVtbl };
6573
6574 static IHTMLDocument2 *create_doc_with_string(const char *str)
6575 {
6576     IPersistStreamInit *init;
6577     IStream *stream;
6578     IHTMLDocument2 *doc;
6579     HGLOBAL mem;
6580     SIZE_T len;
6581
6582     notif_doc = doc = create_document();
6583     if(!doc)
6584         return NULL;
6585
6586     doc_complete = FALSE;
6587     len = strlen(str);
6588     mem = GlobalAlloc(0, len);
6589     memcpy(mem, str, len);
6590     CreateStreamOnHGlobal(mem, TRUE, &stream);
6591
6592     IHTMLDocument2_QueryInterface(doc, &IID_IPersistStreamInit, (void**)&init);
6593
6594     IPersistStreamInit_Load(init, stream);
6595     IPersistStreamInit_Release(init);
6596     IStream_Release(stream);
6597
6598     return doc;
6599 }
6600
6601 static void do_advise(IUnknown *unk, REFIID riid, IUnknown *unk_advise)
6602 {
6603     IConnectionPointContainer *container;
6604     IConnectionPoint *cp;
6605     DWORD cookie;
6606     HRESULT hres;
6607
6608     hres = IUnknown_QueryInterface(unk, &IID_IConnectionPointContainer, (void**)&container);
6609     ok(hres == S_OK, "QueryInterface(IID_IConnectionPointContainer) failed: %08x\n", hres);
6610
6611     hres = IConnectionPointContainer_FindConnectionPoint(container, riid, &cp);
6612     IConnectionPointContainer_Release(container);
6613     ok(hres == S_OK, "FindConnectionPoint failed: %08x\n", hres);
6614
6615     hres = IConnectionPoint_Advise(cp, unk_advise, &cookie);
6616     IConnectionPoint_Release(cp);
6617     ok(hres == S_OK, "Advise failed: %08x\n", hres);
6618 }
6619
6620 typedef void (*domtest_t)(IHTMLDocument2*);
6621
6622 static void run_domtest(const char *str, domtest_t test)
6623 {
6624     IHTMLDocument2 *doc;
6625     ULONG ref;
6626     MSG msg;
6627
6628     doc = create_doc_with_string(str);
6629     if(!doc)
6630         return;
6631
6632     do_advise((IUnknown*)doc, &IID_IPropertyNotifySink, (IUnknown*)&PropertyNotifySink);
6633
6634     while(!doc_complete && GetMessage(&msg, NULL, 0, 0)) {
6635         TranslateMessage(&msg);
6636         DispatchMessage(&msg);
6637     }
6638
6639     test(doc);
6640
6641     ref = IHTMLDocument2_Release(doc);
6642     ok(!ref || broken(ref == 1), /* Vista */
6643        "ref = %d\n", ref);
6644 }
6645
6646 START_TEST(dom)
6647 {
6648     HMODULE hkernel32 = GetModuleHandleA("kernel32.dll");
6649     pLCIDToLocaleName = (void*)GetProcAddress(hkernel32, "LCIDToLocaleName");
6650     pGetUserDefaultUILanguage = (void*)GetProcAddress(hkernel32, "GetUserDefaultUILanguage");
6651
6652     CoInitialize(NULL);
6653
6654     run_domtest(doc_str1, test_doc_elem);
6655     run_domtest(doc_str1, test_get_set_attr);
6656     run_domtest(range_test_str, test_txtrange);
6657     run_domtest(range_test2_str, test_txtrange2);
6658     if (winetest_interactive || ! is_ie_hardened()) {
6659         run_domtest(elem_test_str, test_elems);
6660         run_domtest(elem_test2_str, test_elems2);
6661         run_domtest(noscript_str, test_noscript);
6662     }else {
6663         skip("IE running in Enhanced Security Configuration\n");
6664     }
6665     run_domtest(doc_blank, test_create_elems);
6666     run_domtest(doc_blank, test_defaults);
6667     run_domtest(doc_blank, test_null_write);
6668     run_domtest(emptydiv_str, test_create_stylesheet);
6669     run_domtest(indent_test_str, test_indent);
6670     run_domtest(cond_comment_str, test_cond_comment);
6671     run_domtest(frameset_str, test_frameset);
6672     run_domtest(emptydiv_str, test_docfrag);
6673     run_domtest(doc_blank, test_replacechild_elems);
6674
6675     CoUninitialize();
6676 }