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