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