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