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