2 * Copyright 2007 Jacek Caban for CodeWeavers
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.
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.
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
22 #include <wine/test.h>
32 static const char doc_blank[] = "<html></html>";
33 static const char doc_str1[] = "<html><body>test</body></html>";
34 static const char doc_str2[] =
35 "<html><body>test \na<font size=\"2\">bc\t123<br /> it's\r\n \t</font>text<br /></body></html>";
36 static const char doc_str3[] =
37 "<html><head><title>test</title><body>"
38 "<a href=\"http://test\" name=\"x\">link</a><input />"
39 "<select id=\"s\"><option id=\"x\">opt1</option><option id=\"y\">opt2</option></select>"
40 "<textarea id=\"X\">text text</textarea>"
43 static const WCHAR noneW[] = {'N','o','n','e',0};
45 static WCHAR characterW[] = {'c','h','a','r','a','c','t','e','r',0};
46 static WCHAR texteditW[] = {'t','e','x','t','e','d','i','t',0};
47 static WCHAR wordW[] = {'w','o','r','d',0};
62 static REFIID const none_iids[] = {
67 static REFIID const html_iids[] = {
74 static REFIID const head_iids[] = {
81 static REFIID const title_iids[] = {
88 static REFIID const body_iids[] = {
92 &IID_IHTMLTextContainer,
93 &IID_IHTMLBodyElement,
97 static REFIID const anchor_iids[] = {
101 &IID_IHTMLAnchorElement,
105 static REFIID const input_iids[] = {
109 &IID_IHTMLInputElement,
110 &IID_IHTMLInputTextElement,
114 static REFIID const select_iids[] = {
118 &IID_IHTMLSelectElement,
122 static REFIID const textarea_iids[] = {
126 &IID_IHTMLTextAreaElement,
130 static REFIID const option_iids[] = {
134 &IID_IHTMLOptionElement,
143 static const elem_type_info_t elem_type_infos[] = {
147 {"TITLE", title_iids},
150 {"INPUT", input_iids},
151 {"SELECT", select_iids},
152 {"TEXTAREA", textarea_iids},
153 {"OPTION", option_iids}
156 static const char *dbgstr_w(LPCWSTR str)
158 static char buf[512];
161 WideCharToMultiByte(CP_ACP, 0, str, -1, buf, sizeof(buf), NULL, NULL);
165 static const char *dbgstr_guid(REFIID riid)
169 sprintf(buf, "{%08X-%04X-%04X-%02X%02X-%02X%02X%02X%02X%02X%02X}",
170 riid->Data1, riid->Data2, riid->Data3, riid->Data4[0],
171 riid->Data4[1], riid->Data4[2], riid->Data4[3], riid->Data4[4],
172 riid->Data4[5], riid->Data4[6], riid->Data4[7]);
177 static int strcmp_wa(LPCWSTR strw, const char *stra)
180 MultiByteToWideChar(CP_ACP, 0, stra, -1, buf, sizeof(buf)/sizeof(WCHAR));
181 return lstrcmpW(strw, buf);
184 static BSTR a2bstr(const char *str)
189 len = MultiByteToWideChar(CP_ACP, 0, str, -1, NULL, 0);
190 ret = SysAllocStringLen(NULL, len);
191 MultiByteToWideChar(CP_ACP, 0, str, -1, ret, len);
196 static IHTMLDocument2 *create_document(void)
201 hres = CoCreateInstance(&CLSID_HTMLDocument, NULL, CLSCTX_INPROC_SERVER|CLSCTX_INPROC_HANDLER,
202 &IID_IHTMLDocument2, (void**)&doc);
203 ok(hres == S_OK, "CoCreateInstance failed: %08x\n", hres);
208 #define test_ifaces(i,ids) _test_ifaces(__LINE__,i,ids)
209 static void _test_ifaces(unsigned line, IUnknown *iface, REFIID *iids)
211 const IID * const *piid;
215 for(piid = iids; *piid; piid++) {
216 hres = IDispatch_QueryInterface(iface, *piid, (void**)&unk);
217 ok_(__FILE__,line) (hres == S_OK, "Could not get %s interface: %08x\n", dbgstr_guid(*piid), hres);
219 IUnknown_Release(unk);
223 #define test_node_name(u,n) _test_node_name(__LINE__,u,n)
224 static void _test_node_name(unsigned line, IUnknown *unk, const char *exname)
230 hres = IUnknown_QueryInterface(unk, &IID_IHTMLDOMNode, (void**)&node);
231 ok_(__FILE__, line) (hres == S_OK, "QueryInterface(IID_IHTMLNode) failed: %08x\n", hres);
233 hres = IHTMLDOMNode_get_nodeName(node, &name);
234 IHTMLDOMNode_Release(node);
235 ok_(__FILE__, line) (hres == S_OK, "get_nodeName failed: %08x\n", hres);
236 ok_(__FILE__, line) (!strcmp_wa(name, exname), "got name: %s, expected %s\n", dbgstr_w(name), exname);
241 #define test_elem_tag(u,n) _test_elem_tag(__LINE__,u,n)
242 static void _test_elem_tag(unsigned line, IUnknown *unk, const char *extag)
248 hres = IUnknown_QueryInterface(unk, &IID_IHTMLElement, (void**)&elem);
249 ok_(__FILE__, line) (hres == S_OK, "QueryInterface(IID_IHTMLElement) failed: %08x\n", hres);
251 hres = IHTMLElement_get_tagName(elem, &tag);
252 IHTMLElement_Release(elem);
253 ok_(__FILE__, line) (hres == S_OK, "get_tagName failed: %08x\n", hres);
254 ok_(__FILE__, line) (!strcmp_wa(tag, extag), "got tag: %s, expected %s\n", dbgstr_w(tag), extag);
259 #define test_elem_type(ifc,t) _test_elem_type(__LINE__,ifc,t)
260 static void _test_elem_type(unsigned line, IUnknown *unk, elem_type_t type)
262 _test_elem_tag(line, unk, elem_type_infos[type].tag);
263 _test_ifaces(line, unk, elem_type_infos[type].iids);
266 static void test_doc_elem(IHTMLDocument2 *doc)
269 IHTMLDocument3 *doc3;
272 hres = IHTMLDocument2_QueryInterface(doc, &IID_IHTMLDocument3, (void**)&doc3);
273 ok(hres == S_OK, "QueryInterface(IID_IHTMLDocument3) failed: %08x\n", hres);
275 hres = IHTMLDocument3_get_documentElement(doc3, &elem);
276 IHTMLDocument3_Release(doc3);
277 ok(hres == S_OK, "get_documentElement failed: %08x\n", hres);
279 test_node_name((IUnknown*)elem, "HTML");
280 test_elem_tag((IUnknown*)elem, "HTML");
282 IHTMLElement_Release(elem);
285 #define get_doc_elem(d) _get_doc_elem(__LINE__,d)
286 static IHTMLElement *_get_doc_elem(unsigned line, IHTMLDocument2 *doc)
289 IHTMLDocument3 *doc3;
292 hres = IHTMLDocument2_QueryInterface(doc, &IID_IHTMLDocument3, (void**)&doc3);
293 ok_(__FILE__,line) (hres == S_OK, "Could not get IHTMLDocument3 interface: %08x\n", hres);
294 hres = IHTMLDocument3_get_documentElement(doc3, &elem);
295 ok_(__FILE__,line) (hres == S_OK, "get_documentElement failed: %08x\n", hres);
296 IHTMLDocument3_Release(doc3);
301 #define test_option_text(o,t) _test_option_text(__LINE__,o,t)
302 static void _test_option_text(unsigned line, IHTMLOptionElement *option, const char *text)
307 hres = IHTMLOptionElement_get_text(option, &bstr);
308 ok_(__FILE__,line) (hres == S_OK, "get_text failed: %08x\n", hres);
309 ok_(__FILE__,line) (!strcmp_wa(bstr, text), "text=%s\n", dbgstr_w(bstr));
313 #define test_option_put_text(o,t) _test_option_put_text(__LINE__,o,t)
314 static void _test_option_put_text(unsigned line, IHTMLOptionElement *option, const char *text)
320 hres = IHTMLOptionElement_put_text(option, bstr);
322 ok(hres == S_OK, "put_text failed: %08x\n", hres);
324 _test_option_text(line, option, text);
327 #define test_option_value(o,t) _test_option_value(__LINE__,o,t)
328 static void _test_option_value(unsigned line, IHTMLOptionElement *option, const char *value)
333 hres = IHTMLOptionElement_get_value(option, &bstr);
334 ok_(__FILE__,line) (hres == S_OK, "get_value failed: %08x\n", hres);
335 ok_(__FILE__,line) (!strcmp_wa(bstr, value), "value=%s\n", dbgstr_w(bstr));
339 #define test_option_put_value(o,t) _test_option_put_value(__LINE__,o,t)
340 static void _test_option_put_value(unsigned line, IHTMLOptionElement *option, const char *value)
345 bstr = a2bstr(value);
346 hres = IHTMLOptionElement_put_value(option, bstr);
348 ok(hres == S_OK, "put_value failed: %08x\n", hres);
350 _test_option_value(line, option, value);
353 #define create_option_elem(d,t,v) _create_option_elem(__LINE__,d,t,v)
354 static IHTMLOptionElement *_create_option_elem(unsigned line, IHTMLDocument2 *doc,
355 const char *txt, const char *val)
357 IHTMLOptionElementFactory *factory;
358 IHTMLOptionElement *option;
359 IHTMLWindow2 *window;
360 VARIANT text, value, empty;
363 hres = IHTMLDocument2_get_parentWindow(doc, &window);
364 ok_(__FILE__,line) (hres == S_OK, "get_parentElement failed: %08x\n", hres);
366 hres = IHTMLWindow2_get_Option(window, &factory);
367 IHTMLWindow2_Release(window);
368 ok_(__FILE__,line) (hres == S_OK, "get_Option failed: %08x\n", hres);
370 V_VT(&text) = VT_BSTR;
371 V_BSTR(&text) = a2bstr(txt);
372 V_VT(&value) = VT_BSTR;
373 V_BSTR(&value) = a2bstr(val);
374 V_VT(&empty) = VT_EMPTY;
376 hres = IHTMLOptionElementFactory_create(factory, text, value, empty, empty, &option);
377 ok_(__FILE__,line) (hres == S_OK, "create failed: %08x\n", hres);
379 IHTMLOptionElementFactory_Release(factory);
381 VariantClear(&value);
383 _test_option_text(line, option, txt);
384 _test_option_value(line, option, val);
389 #define test_select_length(s,l) _test_select_length(__LINE__,s,l)
390 static void _test_select_length(unsigned line, IHTMLSelectElement *select, long length)
392 long len = 0xdeadbeef;
395 hres = IHTMLSelectElement_get_length(select, &len);
396 ok_(__FILE__,line) (hres == S_OK, "get_length failed: %08x\n", hres);
397 ok_(__FILE__,line) (len == length, "len=%ld, expected %ld\n", len, length);
400 #define test_select_selidx(s,i) _test_select_selidx(__LINE__,s,i)
401 static void _test_select_selidx(unsigned line, IHTMLSelectElement *select, long index)
403 long idx = 0xdeadbeef;
406 hres = IHTMLSelectElement_get_selectedIndex(select, &idx);
407 ok_(__FILE__,line) (hres == S_OK, "get_selectedIndex failed: %08x\n", hres);
408 ok_(__FILE__,line) (idx == index, "idx=%ld, expected %ld\n", idx, index);
411 #define test_select_put_selidx(s,i) _test_select_put_selidx(__LINE__,s,i)
412 static void _test_select_put_selidx(unsigned line, IHTMLSelectElement *select, long index)
416 hres = IHTMLSelectElement_put_selectedIndex(select, index);
417 ok_(__FILE__,line) (hres == S_OK, "get_selectedIndex failed: %08x\n", hres);
418 _test_select_selidx(line, select, index);
421 #define test_range_text(r,t) _test_range_text(__LINE__,r,t)
422 static void _test_range_text(unsigned line, IHTMLTxtRange *range, const char *extext)
427 hres = IHTMLTxtRange_get_text(range, &text);
428 ok_(__FILE__, line) (hres == S_OK, "get_text failed: %08x\n", hres);
431 ok_(__FILE__, line) (text != NULL, "text == NULL\n");
432 ok_(__FILE__, line) (!strcmp_wa(text, extext), "text=\"%s\", expected \"%s\"\n", dbgstr_w(text), extext);
434 ok_(__FILE__, line) (text == NULL, "text=\"%s\", expected NULL\n", dbgstr_w(text));
441 #define test_range_collapse(r,b) _test_range_collapse(__LINE__,r,b)
442 static void _test_range_collapse(unsigned line, IHTMLTxtRange *range, BOOL b)
446 hres = IHTMLTxtRange_collapse(range, b);
447 ok_(__FILE__, line) (hres == S_OK, "collapse failed: %08x\n", hres);
448 _test_range_text(line, range, NULL);
451 #define test_range_expand(r,u,b,t) _test_range_expand(__LINE__,r,u,b,t)
452 static void _test_range_expand(unsigned line, IHTMLTxtRange *range, LPWSTR unit,
453 VARIANT_BOOL exb, const char *extext)
455 VARIANT_BOOL b = 0xe0e0;
458 hres = IHTMLTxtRange_expand(range, unit, &b);
459 ok_(__FILE__,line) (hres == S_OK, "expand failed: %08x\n", hres);
460 ok_(__FILE__,line) (b == exb, "b=%x, expected %x\n", b, exb);
461 _test_range_text(line, range, extext);
464 #define test_range_move(r,u,c,e) _test_range_move(__LINE__,r,u,c,e)
465 static void _test_range_move(unsigned line, IHTMLTxtRange *range, LPWSTR unit, long cnt, long excnt)
470 hres = IHTMLTxtRange_move(range, unit, cnt, &c);
471 ok_(__FILE__,line) (hres == S_OK, "move failed: %08x\n", hres);
472 ok_(__FILE__,line) (c == excnt, "count=%ld, expected %ld\n", c, excnt);
473 _test_range_text(line, range, NULL);
476 #define test_range_movestart(r,u,c,e) _test_range_movestart(__LINE__,r,u,c,e)
477 static void _test_range_movestart(unsigned line, IHTMLTxtRange *range,
478 LPWSTR unit, long cnt, long excnt)
483 hres = IHTMLTxtRange_moveStart(range, unit, cnt, &c);
484 ok_(__FILE__,line) (hres == S_OK, "move failed: %08x\n", hres);
485 ok_(__FILE__,line) (c == excnt, "count=%ld, expected %ld\n", c, excnt);
488 #define test_range_moveend(r,u,c,e) _test_range_moveend(__LINE__,r,u,c,e)
489 static void _test_range_moveend(unsigned line, IHTMLTxtRange *range, LPWSTR unit, long cnt, long excnt)
494 hres = IHTMLTxtRange_moveEnd(range, unit, cnt, &c);
495 ok_(__FILE__,line) (hres == S_OK, "move failed: %08x\n", hres);
496 ok_(__FILE__,line) (c == excnt, "count=%ld, expected %ld\n", c, excnt);
499 #define test_range_put_text(r,t) _test_range_put_text(__LINE__,r,t)
500 static void _test_range_put_text(unsigned line, IHTMLTxtRange *range, const char *text)
503 BSTR bstr = a2bstr(text);
505 hres = IHTMLTxtRange_put_text(range, bstr);
506 ok_(__FILE__,line) (hres == S_OK, "put_text failed: %08x\n", hres);
508 _test_range_text(line, range, NULL);
511 #define test_range_inrange(r1,r2,b) _test_range_inrange(__LINE__,r1,r2,b)
512 static void _test_range_inrange(unsigned line, IHTMLTxtRange *range1, IHTMLTxtRange *range2, VARIANT_BOOL exb)
518 hres = IHTMLTxtRange_inRange(range1, range2, &b);
519 ok_(__FILE__,line) (hres == S_OK, "(1->2) isEqual failed: %08x\n", hres);
520 ok_(__FILE__,line) (b == exb, "(1->2) b=%x, expected %x\n", b, exb);
523 #define test_range_isequal(r1,r2,b) _test_range_isequal(__LINE__,r1,r2,b)
524 static void _test_range_isequal(unsigned line, IHTMLTxtRange *range1, IHTMLTxtRange *range2, VARIANT_BOOL exb)
530 hres = IHTMLTxtRange_isEqual(range1, range2, &b);
531 ok_(__FILE__,line) (hres == S_OK, "(1->2) isEqual failed: %08x\n", hres);
532 ok_(__FILE__,line) (b == exb, "(1->2) b=%x, expected %x\n", b, exb);
535 hres = IHTMLTxtRange_isEqual(range2, range1, &b);
536 ok_(__FILE__,line) (hres == S_OK, "(2->1) isEqual failed: %08x\n", hres);
537 ok_(__FILE__,line) (b == exb, "(2->1) b=%x, expected %x\n", b, exb);
540 test_range_inrange(range1, range2, VARIANT_TRUE);
541 test_range_inrange(range2, range1, VARIANT_TRUE);
545 #define test_range_parent(r,t) _test_range_parent(__LINE__,r,t)
546 static void _test_range_parent(unsigned line, IHTMLTxtRange *range, elem_type_t type)
551 hres = IHTMLTxtRange_parentElement(range, &elem);
552 ok_(__FILE__,line) (hres == S_OK, "parentElement failed: %08x\n", hres);
554 _test_elem_type(line, (IUnknown*)elem, type);
556 IHTMLElement_Release(elem);
559 static void test_elem_collection(IHTMLElementCollection *col, const elem_type_t *elem_types, long exlen)
567 hres = IHTMLElementCollection_get_length(col, &len);
568 ok(hres == S_OK, "get_length failed: %08x\n", hres);
569 ok(len == exlen, "len=%ld, expected %ld\n", len, exlen);
571 V_VT(&index) = VT_EMPTY;
574 for(i=0; i<len; i++) {
576 disp = (void*)0xdeadbeef;
577 hres = IHTMLElementCollection_item(col, name, index, &disp);
578 ok(hres == S_OK, "item(%d) failed: %08x\n", i, hres);
579 ok(disp != NULL, "item returned NULL\n");
580 if(FAILED(hres) || !disp)
583 test_elem_type((IUnknown*)disp, elem_types[i]);
584 IDispatch_Release(disp);
588 disp = (void*)0xdeadbeef;
589 hres = IHTMLElementCollection_item(col, name, index, &disp);
590 ok(hres == S_OK, "item failed: %08x\n", hres);
591 ok(disp == NULL, "disp != NULL\n");
594 disp = (void*)0xdeadbeef;
595 hres = IHTMLElementCollection_item(col, name, index, &disp);
596 ok(hres == E_INVALIDARG, "item failed: %08x, expected E_INVALIDARG\n", hres);
597 ok(disp == NULL, "disp != NULL\n");
600 static void test_elem_col_item(IHTMLElementCollection *col, LPCWSTR n,
601 const elem_type_t *elem_types, long len)
603 IHTMLElementCollection *elcol;
609 V_VT(&index) = VT_EMPTY;
610 V_VT(&name) = VT_BSTR;
611 V_BSTR(&name) = SysAllocString(n);
613 hres = IHTMLElementCollection_item(col, name, index, &disp);
614 ok(hres == S_OK, "item failed: %08x\n", hres);
616 hres = IDispatch_QueryInterface(disp, &IID_IHTMLElementCollection, (void**)&elcol);
617 IDispatch_Release(disp);
618 ok(hres == S_OK, "Could not get IHTMLElementCollection interface: %08x\n", hres);
619 test_elem_collection(elcol, elem_types, len);
620 IHTMLElementCollection_Release(elcol);
622 V_VT(&index) = VT_I4;
624 for(i=0; i<len; i++) {
626 disp = (void*)0xdeadbeef;
627 hres = IHTMLElementCollection_item(col, name, index, &disp);
628 ok(hres == S_OK, "item failed: %08x\n", hres);
629 ok(disp != NULL, "disp == NULL\n");
630 if(FAILED(hres) || !disp)
633 test_elem_type((IUnknown*)disp, elem_types[i]);
635 IDispatch_Release(disp);
639 disp = (void*)0xdeadbeef;
640 hres = IHTMLElementCollection_item(col, name, index, &disp);
641 ok(hres == S_OK, "item failed: %08x\n", hres);
642 ok(disp == NULL, "disp != NULL\n");
645 disp = (void*)0xdeadbeef;
646 hres = IHTMLElementCollection_item(col, name, index, &disp);
647 ok(hres == E_INVALIDARG, "item failed: %08x, expected E_INVALIDARG\n", hres);
648 ok(disp == NULL, "disp != NULL\n");
650 SysFreeString(V_BSTR(&name));
653 static IHTMLElement *get_elem_by_id(IHTMLDocument2 *doc, LPCWSTR id, BOOL expect_success)
655 IHTMLElementCollection *col;
657 IDispatch *disp = (void*)0xdeadbeef;
661 hres = IHTMLDocument2_get_all(doc, &col);
662 ok(hres == S_OK, "get_all failed: %08x\n", hres);
663 ok(col != NULL, "col == NULL\n");
664 if(FAILED(hres) || !col)
667 V_VT(&index) = VT_EMPTY;
668 V_VT(&name) = VT_BSTR;
669 V_BSTR(&name) = SysAllocString(id);
671 hres = IHTMLElementCollection_item(col, name, index, &disp);
672 IHTMLElementCollection_Release(col);
673 SysFreeString(V_BSTR(&name));
674 ok(hres == S_OK, "item failed: %08x\n", hres);
675 if(!expect_success) {
676 ok(disp == NULL, "disp != NULL\n");
680 ok(disp != NULL, "disp == NULL\n");
684 hres = IDispatch_QueryInterface(disp, &IID_IHTMLElement, (void**)&elem);
685 IDispatch_Release(disp);
686 ok(hres == S_OK, "Could not get IHTMLElement interface: %08x\n", hres);
691 static void test_select_elem(IHTMLSelectElement *select)
693 test_select_length(select, 2);
694 test_select_selidx(select, 0);
695 test_select_put_selidx(select, 1);
698 static void test_create_option_elem(IHTMLDocument2 *doc)
700 IHTMLOptionElement *option;
702 option = create_option_elem(doc, "test text", "test value");
704 test_option_put_text(option, "new text");
705 test_option_put_value(option, "new value");
707 IHTMLOptionElement_Release(option);
710 static void test_txtrange(IHTMLDocument2 *doc)
713 IHTMLBodyElement *body;
714 IHTMLTxtRange *body_range, *range, *range2;
715 IHTMLSelectionObject *selection;
716 IDispatch *disp_range;
719 hres = IHTMLDocument2_get_body(doc, &elem);
720 ok(hres == S_OK, "get_body failed: %08x\n", hres);
722 hres = IHTMLElement_QueryInterface(elem, &IID_IHTMLBodyElement, (void**)&body);
723 IHTMLElement_Release(elem);
725 hres = IHTMLBodyElement_createTextRange(body, &body_range);
726 IHTMLBodyElement_Release(body);
727 ok(hres == S_OK, "createTextRange failed: %08x\n", hres);
729 test_range_text(body_range, "test abc 123\r\nit's text");
731 hres = IHTMLTxtRange_duplicate(body_range, &range);
732 ok(hres == S_OK, "duplicate failed: %08x\n", hres);
734 hres = IHTMLTxtRange_duplicate(body_range, &range2);
735 ok(hres == S_OK, "duplicate failed: %08x\n", hres);
736 test_range_isequal(range, range2, VARIANT_TRUE);
738 test_range_text(range, "test abc 123\r\nit's text");
739 test_range_text(body_range, "test abc 123\r\nit's text");
741 test_range_collapse(range, TRUE);
742 test_range_isequal(range, range2, VARIANT_FALSE);
743 test_range_inrange(range, range2, VARIANT_FALSE);
744 test_range_inrange(range2, range, VARIANT_TRUE);
745 IHTMLTxtRange_Release(range2);
747 test_range_expand(range, wordW, VARIANT_TRUE, "test ");
748 test_range_expand(range, wordW, VARIANT_FALSE, "test ");
749 test_range_move(range, characterW, 2, 2);
750 test_range_expand(range, wordW, VARIANT_TRUE, "test ");
752 test_range_collapse(range, FALSE);
753 test_range_expand(range, wordW, VARIANT_TRUE, "abc ");
755 test_range_collapse(range, FALSE);
756 test_range_expand(range, wordW, VARIANT_TRUE, "123");
757 test_range_expand(range, wordW, VARIANT_FALSE, "123");
758 test_range_move(range, characterW, 2, 2);
759 test_range_expand(range, wordW, VARIANT_TRUE, "123");
760 test_range_moveend(range, characterW, -5, -5);
761 test_range_text(range, NULL);
762 test_range_moveend(range, characterW, 3, 3);
763 test_range_text(range, "c 1");
764 test_range_expand(range, texteditW, VARIANT_TRUE, "test abc 123\r\nit's text");
765 test_range_collapse(range, TRUE);
766 test_range_move(range, characterW, 4, 4);
767 test_range_moveend(range, characterW, 1, 1);
768 test_range_text(range, " ");
769 test_range_move(range, wordW, 1, 1);
770 test_range_moveend(range, characterW, 2, 2);
771 test_range_text(range, "ab");
773 IHTMLTxtRange_Release(range);
775 hres = IHTMLTxtRange_duplicate(body_range, &range);
776 ok(hres == S_OK, "duplicate failed: %08x\n", hres);
778 test_range_text(range, "test abc 123\r\nit's text");
779 test_range_move(range, characterW, 3, 3);
780 test_range_moveend(range, characterW, 1, 1);
781 test_range_text(range, "t");
782 test_range_moveend(range, characterW, 3, 3);
783 test_range_text(range, "t ab");
784 test_range_moveend(range, characterW, -2, -2);
785 test_range_text(range, "t ");
786 test_range_move(range, characterW, 6, 6);
787 test_range_moveend(range, characterW, 3, 3);
788 test_range_text(range, "123");
789 test_range_moveend(range, characterW, 2, 2);
790 test_range_text(range, "123\r\ni");
792 IHTMLTxtRange_Release(range);
794 hres = IHTMLTxtRange_duplicate(body_range, &range);
795 ok(hres == S_OK, "duplicate failed: %08x\n", hres);
797 test_range_move(range, wordW, 1, 1);
798 test_range_moveend(range, characterW, 2, 2);
799 test_range_text(range, "ab");
801 test_range_move(range, characterW, -2, -2);
802 test_range_moveend(range, characterW, 2, 2);
803 test_range_text(range, "t ");
805 test_range_move(range, wordW, 3, 3);
806 test_range_move(range, wordW, -2, -2);
807 test_range_moveend(range, characterW, 2, 2);
808 test_range_text(range, "ab");
810 test_range_move(range, characterW, -6, -5);
811 test_range_moveend(range, characterW, -1, 0);
812 test_range_moveend(range, characterW, -6, 0);
813 test_range_move(range, characterW, 2, 2);
814 test_range_moveend(range, characterW, 2, 2);
815 test_range_text(range, "st");
816 test_range_moveend(range, characterW, -6, -4);
817 test_range_moveend(range, characterW, 2, 2);
819 IHTMLTxtRange_Release(range);
821 hres = IHTMLTxtRange_duplicate(body_range, &range);
822 ok(hres == S_OK, "duplicate failed: %08x\n", hres);
824 test_range_move(range, wordW, 2, 2);
825 test_range_moveend(range, characterW, 2, 2);
826 test_range_text(range, "12");
828 test_range_move(range, characterW, 15, 14);
829 test_range_move(range, characterW, -2, -2);
830 test_range_moveend(range, characterW, 3, 2);
831 test_range_text(range, "t");
832 test_range_moveend(range, characterW, -1, -1);
833 test_range_text(range, "t");
834 test_range_expand(range, wordW, VARIANT_TRUE, "text");
835 test_range_move(range, characterW, -2, -2);
836 test_range_moveend(range, characterW, 2, 2);
837 test_range_text(range, "s ");
838 test_range_move(range, characterW, 100, 7);
839 test_range_move(range, wordW, 1, 0);
840 test_range_move(range, characterW, -2, -2);
841 test_range_moveend(range, characterW, 3, 2);
842 test_range_text(range, "t");
844 IHTMLTxtRange_Release(range);
846 hres = IHTMLTxtRange_duplicate(body_range, &range);
847 ok(hres == S_OK, "duplicate failed: %08x\n", hres);
849 test_range_collapse(range, TRUE);
850 test_range_expand(range, wordW, VARIANT_TRUE, "test ");
851 test_range_put_text(range, "word");
852 test_range_text(body_range, "wordabc 123\r\nit's text");
853 test_range_text(range, NULL);
854 test_range_moveend(range, characterW, 3, 3);
855 test_range_text(range, "abc");
856 test_range_movestart(range, characterW, -2, -2);
857 test_range_text(range, "rdabc");
858 test_range_movestart(range, characterW, 3, 3);
859 test_range_text(range, "bc");
860 test_range_movestart(range, characterW, 4, 4);
861 test_range_text(range, NULL);
862 test_range_movestart(range, characterW, -3, -3);
863 test_range_text(range, "c 1");
864 test_range_movestart(range, characterW, -7, -6);
865 test_range_text(range, "wordabc 1");
866 test_range_movestart(range, characterW, 100, 22);
867 test_range_text(range, NULL);
869 IHTMLTxtRange_Release(range);
870 IHTMLTxtRange_Release(body_range);
872 hres = IHTMLDocument2_get_selection(doc, &selection);
873 ok(hres == S_OK, "IHTMLDocument2_get_selection failed: %08x\n", hres);
875 hres = IHTMLSelectionObject_createRange(selection, &disp_range);
876 ok(hres == S_OK, "IHTMLSelectionObject_createRange failed: %08x\n", hres);
877 IHTMLSelectionObject_Release(selection);
879 hres = IDispatch_QueryInterface(disp_range, &IID_IHTMLTxtRange, (void **)&range);
880 ok(hres == S_OK, "Could not get IID_IHTMLTxtRange interface: 0x%08x\n", hres);
881 IDispatch_Release(disp_range);
883 test_range_text(range, NULL);
884 test_range_moveend(range, characterW, 3, 3);
885 test_range_text(range, "wor");
886 test_range_parent(range, ET_BODY);
887 test_range_expand(range, texteditW, VARIANT_TRUE, "wordabc 123\r\nit's text");
888 test_range_expand(range, texteditW, VARIANT_TRUE, "wordabc 123\r\nit's text");
889 test_range_move(range, characterW, 3, 3);
890 test_range_expand(range, wordW, VARIANT_TRUE, "wordabc ");
891 test_range_moveend(range, characterW, -4, -4);
892 test_range_put_text(range, "abc def ");
893 test_range_expand(range, texteditW, VARIANT_TRUE, "abc def abc 123\r\nit's text");
894 test_range_move(range, wordW, 1, 1);
895 test_range_movestart(range, characterW, -1, -1);
896 test_range_text(range, " ");
897 test_range_move(range, wordW, 1, 1);
898 test_range_moveend(range, characterW, 3, 3);
899 test_range_text(range, "def");
900 test_range_put_text(range, "xyz");
901 test_range_moveend(range, characterW, 1, 1);
902 test_range_move(range, wordW, 1, 1);
903 test_range_moveend(range, characterW, 2, 2);
904 test_range_text(range, "ab");
906 IHTMLTxtRange_Release(range);
909 static void test_compatmode(IHTMLDocument2 *doc)
911 IHTMLDocument5 *doc5;
915 hres = IHTMLDocument2_QueryInterface(doc, &IID_IHTMLDocument5, (void**)&doc5);
916 ok(hres == S_OK, "Could not get IHTMLDocument5 interface: %08x\n", hres);
920 hres = IHTMLDocument5_get_compatMode(doc5, &mode);
921 IHTMLDocument5_Release(doc5);
922 ok(hres == S_OK, "get_compatMode failed: %08x\n", hres);
923 ok(!strcmp_wa(mode, "BackCompat"), "compatMode=%s\n", dbgstr_w(mode));
927 static void test_default_style(IHTMLStyle *style)
934 str = (void*)0xdeadbeef;
935 hres = IHTMLStyle_get_fontFamily(style, &str);
936 ok(hres == S_OK, "get_fontFamily failed: %08x\n", hres);
937 ok(!str, "fontFamily = %s\n", dbgstr_w(str));
939 str = (void*)0xdeadbeef;
940 hres = IHTMLStyle_get_fontWeight(style, &str);
941 ok(hres == S_OK, "get_fontWeight failed: %08x\n", hres);
942 ok(!str, "fontWeight = %s\n", dbgstr_w(str));
944 str = (void*)0xdeadbeef;
945 hres = IHTMLStyle_get_display(style, &str);
946 ok(hres == S_OK, "get_display failed: %08x\n", hres);
947 ok(!str, "display = %s\n", dbgstr_w(str));
949 str = (void*)0xdeadbeef;
950 hres = IHTMLStyle_get_visibility(style, &str);
951 ok(hres == S_OK, "get_visibility failed: %08x\n", hres);
952 ok(!str, "visibility = %s\n", dbgstr_w(str));
955 hres = IHTMLStyle_get_fontSize(style, &v);
956 ok(hres == S_OK, "get_fontSize failed: %08x\n", hres);
957 ok(V_VT(&v) == VT_BSTR, "V_VT(fontSize) = %d\n", V_VT(&v));
958 ok(!V_BSTR(&v), "V_BSTR(fontSize) = %s\n", dbgstr_w(V_BSTR(&v)));
961 hres = IHTMLStyle_get_color(style, &v);
962 ok(hres == S_OK, "get_color failed: %08x\n", hres);
963 ok(V_VT(&v) == VT_BSTR, "V_VT(color) = %d\n", V_VT(&v));
964 ok(!V_BSTR(&v), "V_BSTR(color) = %s\n", dbgstr_w(V_BSTR(&v)));
967 hres = IHTMLStyle_get_textDecorationUnderline(style, &b);
968 ok(hres == S_OK, "get_textDecorationUnderline failed: %08x\n", hres);
969 ok(b == VARIANT_FALSE, "textDecorationUnderline = %x\n", b);
972 hres = IHTMLStyle_get_textDecorationLineThrough(style, &b);
973 ok(hres == S_OK, "get_textDecorationLineThrough failed: %08x\n", hres);
974 ok(b == VARIANT_FALSE, "textDecorationLineThrough = %x\n", b);
977 static void test_default_selection(IHTMLDocument2 *doc)
979 IHTMLSelectionObject *selection;
980 IHTMLTxtRange *range;
985 hres = IHTMLDocument2_get_selection(doc, &selection);
986 ok(hres == S_OK, "get_selection failed: %08x\n", hres);
988 hres = IHTMLSelectionObject_get_type(selection, &str);
989 ok(hres == S_OK, "get_type failed: %08x\n", hres);
990 ok(!lstrcmpW(str, noneW), "type = %s\n", dbgstr_w(str));
993 hres = IHTMLSelectionObject_createRange(selection, &disp);
994 IHTMLSelectionObject_Release(selection);
995 ok(hres == S_OK, "createRange failed: %08x\n", hres);
997 hres = IDispatch_QueryInterface(disp, &IID_IHTMLTxtRange, (void**)&range);
998 IDispatch_Release(disp);
999 ok(hres == S_OK, "Could not get IHTMLTxtRange interface: %08x\n", hres);
1001 test_range_text(range, NULL);
1002 IHTMLTxtRange_Release(range);
1005 static void test_defaults(IHTMLDocument2 *doc)
1007 IHTMLStyleSheetsCollection *stylesheetcol;
1013 hres = IHTMLDocument2_get_body(doc, &elem);
1014 ok(hres == S_OK, "get_body failed: %08x\n", hres);
1016 hres = IHTMLElement_get_style(elem, &style);
1017 IHTMLElement_Release(elem);
1018 ok(hres == S_OK, "get_style failed: %08x\n", hres);
1020 test_default_style(style);
1021 test_compatmode(doc);
1023 IHTMLStyle_Release(style);
1025 hres = IHTMLDocument2_get_styleSheets(doc, &stylesheetcol);
1026 ok(hres == S_OK, "get_styleSheets failed: %08x\n", hres);
1029 hres = IHTMLStyleSheetsCollection_get_length(stylesheetcol, &l);
1030 ok(hres == S_OK, "get_length failed: %08x\n", hres);
1031 ok(l == 0, "length = %ld\n", l);
1033 IHTMLStyleSheetsCollection_Release(stylesheetcol);
1035 test_default_selection(doc);
1038 static void test_elems(IHTMLDocument2 *doc)
1040 IHTMLElementCollection *col;
1045 static const WCHAR xW[] = {'x',0};
1046 static const WCHAR sW[] = {'s',0};
1047 static const WCHAR xxxW[] = {'x','x','x',0};
1049 static const elem_type_t all_types[] = {
1062 static const elem_type_t item_types[] = {
1068 hres = IHTMLDocument2_get_all(doc, &col);
1069 ok(hres == S_OK, "get_all failed: %08x\n", hres);
1070 test_elem_collection(col, all_types, sizeof(all_types)/sizeof(all_types[0]));
1071 test_elem_col_item(col, xW, item_types, sizeof(item_types)/sizeof(item_types[0]));
1072 IHTMLElementCollection_Release(col);
1074 elem = get_doc_elem(doc);
1075 ok(hres == S_OK, "get_documentElement failed: %08x\n", hres);
1076 hres = IHTMLElement_get_all(elem, &disp);
1077 IHTMLElement_Release(elem);
1078 ok(hres == S_OK, "get_all failed: %08x\n", hres);
1080 hres = IDispatch_QueryInterface(disp, &IID_IHTMLElementCollection, (void**)&col);
1081 IDispatch_Release(disp);
1082 ok(hres == S_OK, "Could not get IHTMLElementCollection: %08x\n", hres);
1083 test_elem_collection(col, all_types+1, sizeof(all_types)/sizeof(all_types[0])-1);
1084 IHTMLElementCollection_Release(col);
1086 get_elem_by_id(doc, xxxW, FALSE);
1087 elem = get_elem_by_id(doc, sW, TRUE);
1089 IHTMLSelectElement *select;
1091 hres = IHTMLElement_QueryInterface(elem, &IID_IHTMLSelectElement, (void**)&select);
1092 ok(hres == S_OK, "Could not get IHTMLSelectElement interface: %08x\n", hres);
1094 test_select_elem(select);
1096 IHTMLSelectElement_Release(select);
1097 IHTMLElement_Release(elem);
1100 test_create_option_elem(doc);
1103 static IHTMLDocument2 *notif_doc;
1104 static BOOL doc_complete;
1106 static HRESULT WINAPI PropertyNotifySink_QueryInterface(IPropertyNotifySink *iface,
1107 REFIID riid, void**ppv)
1109 if(IsEqualGUID(&IID_IPropertyNotifySink, riid)) {
1114 ok(0, "unexpected call\n");
1115 return E_NOINTERFACE;
1118 static ULONG WINAPI PropertyNotifySink_AddRef(IPropertyNotifySink *iface)
1123 static ULONG WINAPI PropertyNotifySink_Release(IPropertyNotifySink *iface)
1128 static HRESULT WINAPI PropertyNotifySink_OnChanged(IPropertyNotifySink *iface, DISPID dispID)
1130 if(dispID == DISPID_READYSTATE){
1134 static const WCHAR completeW[] = {'c','o','m','p','l','e','t','e',0};
1136 hres = IHTMLDocument2_get_readyState(notif_doc, &state);
1137 ok(hres == S_OK, "get_readyState failed: %08x\n", hres);
1139 if(!lstrcmpW(state, completeW))
1140 doc_complete = TRUE;
1142 SysFreeString(state);
1148 static HRESULT WINAPI PropertyNotifySink_OnRequestEdit(IPropertyNotifySink *iface, DISPID dispID)
1150 ok(0, "unexpected call\n");
1154 static IPropertyNotifySinkVtbl PropertyNotifySinkVtbl = {
1155 PropertyNotifySink_QueryInterface,
1156 PropertyNotifySink_AddRef,
1157 PropertyNotifySink_Release,
1158 PropertyNotifySink_OnChanged,
1159 PropertyNotifySink_OnRequestEdit
1162 static IPropertyNotifySink PropertyNotifySink = { &PropertyNotifySinkVtbl };
1164 static IHTMLDocument2 *create_doc_with_string(const char *str)
1166 IPersistStreamInit *init;
1168 IHTMLDocument2 *doc;
1172 notif_doc = doc = create_document();
1176 doc_complete = FALSE;
1178 mem = GlobalAlloc(0, len);
1179 memcpy(mem, str, len);
1180 CreateStreamOnHGlobal(mem, TRUE, &stream);
1182 IHTMLDocument2_QueryInterface(doc, &IID_IPersistStreamInit, (void**)&init);
1184 IPersistStreamInit_Load(init, stream);
1185 IPersistStreamInit_Release(init);
1186 IStream_Release(stream);
1191 static void do_advise(IUnknown *unk, REFIID riid, IUnknown *unk_advise)
1193 IConnectionPointContainer *container;
1194 IConnectionPoint *cp;
1198 hres = IUnknown_QueryInterface(unk, &IID_IConnectionPointContainer, (void**)&container);
1199 ok(hres == S_OK, "QueryInterface(IID_IConnectionPointContainer) failed: %08x\n", hres);
1201 hres = IConnectionPointContainer_FindConnectionPoint(container, riid, &cp);
1202 IConnectionPointContainer_Release(container);
1203 ok(hres == S_OK, "FindConnectionPoint failed: %08x\n", hres);
1205 hres = IConnectionPoint_Advise(cp, unk_advise, &cookie);
1206 IConnectionPoint_Release(cp);
1207 ok(hres == S_OK, "Advise failed: %08x\n", hres);
1210 typedef void (*domtest_t)(IHTMLDocument2*);
1212 static void run_domtest(const char *str, domtest_t test)
1214 IHTMLDocument2 *doc;
1215 IHTMLElement *body = NULL;
1220 doc = create_doc_with_string(str);
1221 do_advise((IUnknown*)doc, &IID_IPropertyNotifySink, (IUnknown*)&PropertyNotifySink);
1223 while(!doc_complete && GetMessage(&msg, NULL, 0, 0)) {
1224 TranslateMessage(&msg);
1225 DispatchMessage(&msg);
1228 hres = IHTMLDocument2_get_body(doc, &body);
1229 ok(hres == S_OK, "get_body failed: %08x\n", hres);
1232 IHTMLElement_Release(body);
1235 skip("Could not get document body. Assuming no Gecko installed.\n");
1238 ref = IHTMLDocument2_Release(doc);
1239 ok(!ref, "ref = %d\n", ref);
1242 static void gecko_installer_workaround(BOOL disable)
1247 static BOOL has_url = FALSE;
1248 static char url[2048];
1250 if(!disable && !has_url)
1253 res = RegOpenKey(HKEY_CURRENT_USER, "Software\\Wine\\MSHTML", &hkey);
1254 if(res != ERROR_SUCCESS)
1258 DWORD type, size = sizeof(url);
1260 res = RegQueryValueEx(hkey, "GeckoUrl", NULL, &type, (PVOID)url, &size);
1261 if(res == ERROR_SUCCESS && type == REG_SZ)
1264 RegDeleteValue(hkey, "GeckoUrl");
1266 RegSetValueEx(hkey, "GeckoUrl", 0, REG_SZ, (PVOID)url, lstrlenA(url)+1);
1274 gecko_installer_workaround(TRUE);
1277 run_domtest(doc_str1, test_doc_elem);
1278 run_domtest(doc_str2, test_txtrange);
1279 run_domtest(doc_str3, test_elems);
1280 run_domtest(doc_blank, test_defaults);
1283 gecko_installer_workaround(FALSE);