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