| /* |
| * Copyright 2007 Jacek Caban for CodeWeavers |
| * |
| * This library is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Lesser General Public |
| * License as published by the Free Software Foundation; either |
| * version 2.1 of the License, or (at your option) any later version. |
| * |
| * This library is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * Lesser General Public License for more details. |
| * |
| * You should have received a copy of the GNU Lesser General Public |
| * License along with this library; if not, write to the Free Software |
| * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA |
| */ |
| |
| #define COBJMACROS |
| #define CONST_VTABLE |
| |
| #include <wine/test.h> |
| #include <stdarg.h> |
| #include <stdio.h> |
| |
| #include "windef.h" |
| #include "winbase.h" |
| #include "ole2.h" |
| #include "mshtml.h" |
| #include "mshtmcid.h" |
| #include "mshtmhst.h" |
| #include "docobj.h" |
| |
| static const char doc_blank[] = "<html></html>"; |
| static const char doc_str1[] = "<html><body>test</body></html>"; |
| static const char doc_str2[] = |
| "<html><body>test \na<font size=\"2\">bc\t123<br /> it's\r\n \t</font>text<br /></body></html>"; |
| static const char elem_test_str[] = |
| "<html><head><title>test</title><style>.body { margin-right: 0px; }</style>" |
| "<body><a href=\"http://test\" name=\"x\">link</a><input />" |
| "<select id=\"s\"><option id=\"x\">opt1</option><option id=\"y\">opt2</option></select>" |
| "<textarea id=\"X\">text text</textarea>" |
| "<table><tbody></tbody></table>" |
| "</body></html>"; |
| static const char indent_test_str[] = |
| "<html><head><title>test</title></head><body>abc<br /><a href=\"about:blank\">123</a></body></html>"; |
| |
| static const WCHAR noneW[] = {'N','o','n','e',0}; |
| |
| static WCHAR characterW[] = {'c','h','a','r','a','c','t','e','r',0}; |
| static WCHAR texteditW[] = {'t','e','x','t','e','d','i','t',0}; |
| static WCHAR wordW[] = {'w','o','r','d',0}; |
| |
| typedef enum { |
| ET_NONE, |
| ET_HTML, |
| ET_HEAD, |
| ET_TITLE, |
| ET_BODY, |
| ET_A, |
| ET_INPUT, |
| ET_SELECT, |
| ET_TEXTAREA, |
| ET_OPTION, |
| ET_STYLE, |
| ET_BLOCKQUOTE, |
| ET_P, |
| ET_BR, |
| ET_TABLE, |
| ET_TBODY |
| } elem_type_t; |
| |
| static REFIID const none_iids[] = { |
| &IID_IUnknown, |
| NULL |
| }; |
| |
| static REFIID const elem_iids[] = { |
| &IID_IHTMLDOMNode, |
| &IID_IHTMLElement, |
| &IID_IHTMLElement2, |
| &IID_IConnectionPointContainer, |
| NULL |
| }; |
| |
| static REFIID const body_iids[] = { |
| &IID_IHTMLDOMNode, |
| &IID_IHTMLElement, |
| &IID_IHTMLElement2, |
| &IID_IHTMLTextContainer, |
| &IID_IHTMLBodyElement, |
| &IID_IConnectionPointContainer, |
| NULL |
| }; |
| |
| static REFIID const anchor_iids[] = { |
| &IID_IHTMLDOMNode, |
| &IID_IHTMLElement, |
| &IID_IHTMLElement2, |
| &IID_IHTMLAnchorElement, |
| &IID_IConnectionPointContainer, |
| NULL |
| }; |
| |
| static REFIID const input_iids[] = { |
| &IID_IHTMLDOMNode, |
| &IID_IHTMLElement, |
| &IID_IHTMLElement2, |
| &IID_IHTMLInputElement, |
| &IID_IHTMLInputTextElement, |
| &IID_IConnectionPointContainer, |
| NULL |
| }; |
| |
| static REFIID const select_iids[] = { |
| &IID_IHTMLDOMNode, |
| &IID_IHTMLElement, |
| &IID_IHTMLElement2, |
| &IID_IHTMLSelectElement, |
| &IID_IConnectionPointContainer, |
| NULL |
| }; |
| |
| static REFIID const textarea_iids[] = { |
| &IID_IHTMLDOMNode, |
| &IID_IHTMLElement, |
| &IID_IHTMLElement2, |
| &IID_IHTMLTextAreaElement, |
| &IID_IConnectionPointContainer, |
| NULL |
| }; |
| |
| static REFIID const option_iids[] = { |
| &IID_IHTMLDOMNode, |
| &IID_IHTMLElement, |
| &IID_IHTMLElement2, |
| &IID_IHTMLOptionElement, |
| &IID_IConnectionPointContainer, |
| NULL |
| }; |
| |
| static REFIID const table_iids[] = { |
| &IID_IHTMLDOMNode, |
| &IID_IHTMLElement, |
| &IID_IHTMLElement2, |
| &IID_IHTMLTable, |
| &IID_IConnectionPointContainer, |
| NULL |
| }; |
| |
| typedef struct { |
| const char *tag; |
| REFIID *iids; |
| } elem_type_info_t; |
| |
| static const elem_type_info_t elem_type_infos[] = { |
| {"", none_iids}, |
| {"HTML", elem_iids}, |
| {"HEAD", elem_iids}, |
| {"TITLE", elem_iids}, |
| {"BODY", body_iids}, |
| {"A", anchor_iids}, |
| {"INPUT", input_iids}, |
| {"SELECT", select_iids}, |
| {"TEXTAREA", textarea_iids}, |
| {"OPTION", option_iids}, |
| {"STYLE", elem_iids}, |
| {"BLOCKQUOTE",elem_iids}, |
| {"P", elem_iids}, |
| {"BR", elem_iids}, |
| {"TABLE", table_iids}, |
| {"TBODY", elem_iids} |
| }; |
| |
| static const char *dbgstr_w(LPCWSTR str) |
| { |
| static char buf[512]; |
| if(!str) |
| return "(null)"; |
| WideCharToMultiByte(CP_ACP, 0, str, -1, buf, sizeof(buf), NULL, NULL); |
| return buf; |
| } |
| |
| static const char *dbgstr_guid(REFIID riid) |
| { |
| static char buf[50]; |
| |
| sprintf(buf, "{%08X-%04X-%04X-%02X%02X-%02X%02X%02X%02X%02X%02X}", |
| riid->Data1, riid->Data2, riid->Data3, riid->Data4[0], |
| riid->Data4[1], riid->Data4[2], riid->Data4[3], riid->Data4[4], |
| riid->Data4[5], riid->Data4[6], riid->Data4[7]); |
| |
| return buf; |
| } |
| |
| static int strcmp_wa(LPCWSTR strw, const char *stra) |
| { |
| WCHAR buf[512]; |
| MultiByteToWideChar(CP_ACP, 0, stra, -1, buf, sizeof(buf)/sizeof(WCHAR)); |
| return lstrcmpW(strw, buf); |
| } |
| |
| static BSTR a2bstr(const char *str) |
| { |
| BSTR ret; |
| int len; |
| |
| len = MultiByteToWideChar(CP_ACP, 0, str, -1, NULL, 0); |
| ret = SysAllocStringLen(NULL, len); |
| MultiByteToWideChar(CP_ACP, 0, str, -1, ret, len); |
| |
| return ret; |
| } |
| |
| static IHTMLDocument2 *create_document(void) |
| { |
| IHTMLDocument2 *doc; |
| HRESULT hres; |
| |
| hres = CoCreateInstance(&CLSID_HTMLDocument, NULL, CLSCTX_INPROC_SERVER|CLSCTX_INPROC_HANDLER, |
| &IID_IHTMLDocument2, (void**)&doc); |
| ok(hres == S_OK, "CoCreateInstance failed: %08x\n", hres); |
| |
| return doc; |
| } |
| |
| #define test_ifaces(i,ids) _test_ifaces(__LINE__,i,ids) |
| static void _test_ifaces(unsigned line, IUnknown *iface, REFIID *iids) |
| { |
| const IID * const *piid; |
| IUnknown *unk; |
| HRESULT hres; |
| |
| for(piid = iids; *piid; piid++) { |
| hres = IDispatch_QueryInterface(iface, *piid, (void**)&unk); |
| ok_(__FILE__,line) (hres == S_OK, "Could not get %s interface: %08x\n", dbgstr_guid(*piid), hres); |
| if(SUCCEEDED(hres)) |
| IUnknown_Release(unk); |
| } |
| } |
| |
| #define test_node_name(u,n) _test_node_name(__LINE__,u,n) |
| static void _test_node_name(unsigned line, IUnknown *unk, const char *exname) |
| { |
| IHTMLDOMNode *node; |
| BSTR name; |
| HRESULT hres; |
| |
| hres = IUnknown_QueryInterface(unk, &IID_IHTMLDOMNode, (void**)&node); |
| ok_(__FILE__, line) (hres == S_OK, "QueryInterface(IID_IHTMLNode) failed: %08x\n", hres); |
| |
| hres = IHTMLDOMNode_get_nodeName(node, &name); |
| IHTMLDOMNode_Release(node); |
| ok_(__FILE__, line) (hres == S_OK, "get_nodeName failed: %08x\n", hres); |
| ok_(__FILE__, line) (!strcmp_wa(name, exname), "got name: %s, expected %s\n", dbgstr_w(name), exname); |
| |
| SysFreeString(name); |
| } |
| |
| #define test_elem_tag(u,n) _test_elem_tag(__LINE__,u,n) |
| static void _test_elem_tag(unsigned line, IUnknown *unk, const char *extag) |
| { |
| IHTMLElement *elem; |
| BSTR tag; |
| HRESULT hres; |
| |
| hres = IUnknown_QueryInterface(unk, &IID_IHTMLElement, (void**)&elem); |
| ok_(__FILE__, line) (hres == S_OK, "QueryInterface(IID_IHTMLElement) failed: %08x\n", hres); |
| |
| hres = IHTMLElement_get_tagName(elem, &tag); |
| IHTMLElement_Release(elem); |
| ok_(__FILE__, line) (hres == S_OK, "get_tagName failed: %08x\n", hres); |
| ok_(__FILE__, line) (!strcmp_wa(tag, extag), "got tag: %s, expected %s\n", dbgstr_w(tag), extag); |
| |
| SysFreeString(tag); |
| } |
| |
| #define test_elem_type(ifc,t) _test_elem_type(__LINE__,ifc,t) |
| static void _test_elem_type(unsigned line, IUnknown *unk, elem_type_t type) |
| { |
| _test_elem_tag(line, unk, elem_type_infos[type].tag); |
| _test_ifaces(line, unk, elem_type_infos[type].iids); |
| } |
| |
| static void test_doc_elem(IHTMLDocument2 *doc) |
| { |
| IHTMLElement *elem; |
| IHTMLDocument3 *doc3; |
| HRESULT hres; |
| |
| hres = IHTMLDocument2_QueryInterface(doc, &IID_IHTMLDocument3, (void**)&doc3); |
| ok(hres == S_OK, "QueryInterface(IID_IHTMLDocument3) failed: %08x\n", hres); |
| |
| hres = IHTMLDocument3_get_documentElement(doc3, &elem); |
| IHTMLDocument3_Release(doc3); |
| ok(hres == S_OK, "get_documentElement failed: %08x\n", hres); |
| |
| test_node_name((IUnknown*)elem, "HTML"); |
| test_elem_tag((IUnknown*)elem, "HTML"); |
| |
| IHTMLElement_Release(elem); |
| } |
| |
| #define get_doc_elem(d) _get_doc_elem(__LINE__,d) |
| static IHTMLElement *_get_doc_elem(unsigned line, IHTMLDocument2 *doc) |
| { |
| IHTMLElement *elem; |
| IHTMLDocument3 *doc3; |
| HRESULT hres; |
| |
| hres = IHTMLDocument2_QueryInterface(doc, &IID_IHTMLDocument3, (void**)&doc3); |
| ok_(__FILE__,line) (hres == S_OK, "Could not get IHTMLDocument3 interface: %08x\n", hres); |
| hres = IHTMLDocument3_get_documentElement(doc3, &elem); |
| ok_(__FILE__,line) (hres == S_OK, "get_documentElement failed: %08x\n", hres); |
| IHTMLDocument3_Release(doc3); |
| |
| return elem; |
| } |
| |
| #define test_option_text(o,t) _test_option_text(__LINE__,o,t) |
| static void _test_option_text(unsigned line, IHTMLOptionElement *option, const char *text) |
| { |
| BSTR bstr; |
| HRESULT hres; |
| |
| hres = IHTMLOptionElement_get_text(option, &bstr); |
| ok_(__FILE__,line) (hres == S_OK, "get_text failed: %08x\n", hres); |
| ok_(__FILE__,line) (!strcmp_wa(bstr, text), "text=%s\n", dbgstr_w(bstr)); |
| SysFreeString(bstr); |
| } |
| |
| #define test_option_put_text(o,t) _test_option_put_text(__LINE__,o,t) |
| static void _test_option_put_text(unsigned line, IHTMLOptionElement *option, const char *text) |
| { |
| BSTR bstr; |
| HRESULT hres; |
| |
| bstr = a2bstr(text); |
| hres = IHTMLOptionElement_put_text(option, bstr); |
| SysFreeString(bstr); |
| ok(hres == S_OK, "put_text failed: %08x\n", hres); |
| |
| _test_option_text(line, option, text); |
| } |
| |
| #define test_option_value(o,t) _test_option_value(__LINE__,o,t) |
| static void _test_option_value(unsigned line, IHTMLOptionElement *option, const char *value) |
| { |
| BSTR bstr; |
| HRESULT hres; |
| |
| hres = IHTMLOptionElement_get_value(option, &bstr); |
| ok_(__FILE__,line) (hres == S_OK, "get_value failed: %08x\n", hres); |
| ok_(__FILE__,line) (!strcmp_wa(bstr, value), "value=%s\n", dbgstr_w(bstr)); |
| SysFreeString(bstr); |
| } |
| |
| #define test_option_put_value(o,t) _test_option_put_value(__LINE__,o,t) |
| static void _test_option_put_value(unsigned line, IHTMLOptionElement *option, const char *value) |
| { |
| BSTR bstr; |
| HRESULT hres; |
| |
| bstr = a2bstr(value); |
| hres = IHTMLOptionElement_put_value(option, bstr); |
| SysFreeString(bstr); |
| ok(hres == S_OK, "put_value failed: %08x\n", hres); |
| |
| _test_option_value(line, option, value); |
| } |
| |
| #define create_option_elem(d,t,v) _create_option_elem(__LINE__,d,t,v) |
| static IHTMLOptionElement *_create_option_elem(unsigned line, IHTMLDocument2 *doc, |
| const char *txt, const char *val) |
| { |
| IHTMLOptionElementFactory *factory; |
| IHTMLOptionElement *option; |
| IHTMLWindow2 *window; |
| VARIANT text, value, empty; |
| HRESULT hres; |
| |
| hres = IHTMLDocument2_get_parentWindow(doc, &window); |
| ok_(__FILE__,line) (hres == S_OK, "get_parentElement failed: %08x\n", hres); |
| |
| hres = IHTMLWindow2_get_Option(window, &factory); |
| IHTMLWindow2_Release(window); |
| ok_(__FILE__,line) (hres == S_OK, "get_Option failed: %08x\n", hres); |
| |
| V_VT(&text) = VT_BSTR; |
| V_BSTR(&text) = a2bstr(txt); |
| V_VT(&value) = VT_BSTR; |
| V_BSTR(&value) = a2bstr(val); |
| V_VT(&empty) = VT_EMPTY; |
| |
| hres = IHTMLOptionElementFactory_create(factory, text, value, empty, empty, &option); |
| ok_(__FILE__,line) (hres == S_OK, "create failed: %08x\n", hres); |
| |
| IHTMLOptionElementFactory_Release(factory); |
| VariantClear(&text); |
| VariantClear(&value); |
| |
| _test_option_text(line, option, txt); |
| _test_option_value(line, option, val); |
| |
| return option; |
| } |
| |
| #define test_select_length(s,l) _test_select_length(__LINE__,s,l) |
| static void _test_select_length(unsigned line, IHTMLSelectElement *select, long length) |
| { |
| long len = 0xdeadbeef; |
| HRESULT hres; |
| |
| hres = IHTMLSelectElement_get_length(select, &len); |
| ok_(__FILE__,line) (hres == S_OK, "get_length failed: %08x\n", hres); |
| ok_(__FILE__,line) (len == length, "len=%ld, expected %ld\n", len, length); |
| } |
| |
| #define test_select_selidx(s,i) _test_select_selidx(__LINE__,s,i) |
| static void _test_select_selidx(unsigned line, IHTMLSelectElement *select, long index) |
| { |
| long idx = 0xdeadbeef; |
| HRESULT hres; |
| |
| hres = IHTMLSelectElement_get_selectedIndex(select, &idx); |
| ok_(__FILE__,line) (hres == S_OK, "get_selectedIndex failed: %08x\n", hres); |
| ok_(__FILE__,line) (idx == index, "idx=%ld, expected %ld\n", idx, index); |
| } |
| |
| #define test_select_put_selidx(s,i) _test_select_put_selidx(__LINE__,s,i) |
| static void _test_select_put_selidx(unsigned line, IHTMLSelectElement *select, long index) |
| { |
| HRESULT hres; |
| |
| hres = IHTMLSelectElement_put_selectedIndex(select, index); |
| ok_(__FILE__,line) (hres == S_OK, "get_selectedIndex failed: %08x\n", hres); |
| _test_select_selidx(line, select, index); |
| } |
| |
| #define test_range_text(r,t) _test_range_text(__LINE__,r,t) |
| static void _test_range_text(unsigned line, IHTMLTxtRange *range, const char *extext) |
| { |
| BSTR text; |
| HRESULT hres; |
| |
| hres = IHTMLTxtRange_get_text(range, &text); |
| ok_(__FILE__, line) (hres == S_OK, "get_text failed: %08x\n", hres); |
| |
| if(extext) { |
| ok_(__FILE__, line) (text != NULL, "text == NULL\n"); |
| ok_(__FILE__, line) (!strcmp_wa(text, extext), "text=\"%s\", expected \"%s\"\n", dbgstr_w(text), extext); |
| }else { |
| ok_(__FILE__, line) (text == NULL, "text=\"%s\", expected NULL\n", dbgstr_w(text)); |
| } |
| |
| SysFreeString(text); |
| |
| } |
| |
| #define test_range_collapse(r,b) _test_range_collapse(__LINE__,r,b) |
| static void _test_range_collapse(unsigned line, IHTMLTxtRange *range, BOOL b) |
| { |
| HRESULT hres; |
| |
| hres = IHTMLTxtRange_collapse(range, b); |
| ok_(__FILE__, line) (hres == S_OK, "collapse failed: %08x\n", hres); |
| _test_range_text(line, range, NULL); |
| } |
| |
| #define test_range_expand(r,u,b,t) _test_range_expand(__LINE__,r,u,b,t) |
| static void _test_range_expand(unsigned line, IHTMLTxtRange *range, LPWSTR unit, |
| VARIANT_BOOL exb, const char *extext) |
| { |
| VARIANT_BOOL b = 0xe0e0; |
| HRESULT hres; |
| |
| hres = IHTMLTxtRange_expand(range, unit, &b); |
| ok_(__FILE__,line) (hres == S_OK, "expand failed: %08x\n", hres); |
| ok_(__FILE__,line) (b == exb, "b=%x, expected %x\n", b, exb); |
| _test_range_text(line, range, extext); |
| } |
| |
| #define test_range_move(r,u,c,e) _test_range_move(__LINE__,r,u,c,e) |
| static void _test_range_move(unsigned line, IHTMLTxtRange *range, LPWSTR unit, long cnt, long excnt) |
| { |
| long c = 0xdeadbeef; |
| HRESULT hres; |
| |
| hres = IHTMLTxtRange_move(range, unit, cnt, &c); |
| ok_(__FILE__,line) (hres == S_OK, "move failed: %08x\n", hres); |
| ok_(__FILE__,line) (c == excnt, "count=%ld, expected %ld\n", c, excnt); |
| _test_range_text(line, range, NULL); |
| } |
| |
| #define test_range_movestart(r,u,c,e) _test_range_movestart(__LINE__,r,u,c,e) |
| static void _test_range_movestart(unsigned line, IHTMLTxtRange *range, |
| LPWSTR unit, long cnt, long excnt) |
| { |
| long c = 0xdeadbeef; |
| HRESULT hres; |
| |
| hres = IHTMLTxtRange_moveStart(range, unit, cnt, &c); |
| ok_(__FILE__,line) (hres == S_OK, "move failed: %08x\n", hres); |
| ok_(__FILE__,line) (c == excnt, "count=%ld, expected %ld\n", c, excnt); |
| } |
| |
| #define test_range_moveend(r,u,c,e) _test_range_moveend(__LINE__,r,u,c,e) |
| static void _test_range_moveend(unsigned line, IHTMLTxtRange *range, LPWSTR unit, long cnt, long excnt) |
| { |
| long c = 0xdeadbeef; |
| HRESULT hres; |
| |
| hres = IHTMLTxtRange_moveEnd(range, unit, cnt, &c); |
| ok_(__FILE__,line) (hres == S_OK, "move failed: %08x\n", hres); |
| ok_(__FILE__,line) (c == excnt, "count=%ld, expected %ld\n", c, excnt); |
| } |
| |
| #define test_range_put_text(r,t) _test_range_put_text(__LINE__,r,t) |
| static void _test_range_put_text(unsigned line, IHTMLTxtRange *range, const char *text) |
| { |
| HRESULT hres; |
| BSTR bstr = a2bstr(text); |
| |
| hres = IHTMLTxtRange_put_text(range, bstr); |
| ok_(__FILE__,line) (hres == S_OK, "put_text failed: %08x\n", hres); |
| SysFreeString(bstr); |
| _test_range_text(line, range, NULL); |
| } |
| |
| #define test_range_inrange(r1,r2,b) _test_range_inrange(__LINE__,r1,r2,b) |
| static void _test_range_inrange(unsigned line, IHTMLTxtRange *range1, IHTMLTxtRange *range2, VARIANT_BOOL exb) |
| { |
| VARIANT_BOOL b; |
| HRESULT hres; |
| |
| b = 0xe0e0; |
| hres = IHTMLTxtRange_inRange(range1, range2, &b); |
| ok_(__FILE__,line) (hres == S_OK, "(1->2) isEqual failed: %08x\n", hres); |
| ok_(__FILE__,line) (b == exb, "(1->2) b=%x, expected %x\n", b, exb); |
| } |
| |
| #define test_range_isequal(r1,r2,b) _test_range_isequal(__LINE__,r1,r2,b) |
| static void _test_range_isequal(unsigned line, IHTMLTxtRange *range1, IHTMLTxtRange *range2, VARIANT_BOOL exb) |
| { |
| VARIANT_BOOL b; |
| HRESULT hres; |
| |
| b = 0xe0e0; |
| hres = IHTMLTxtRange_isEqual(range1, range2, &b); |
| ok_(__FILE__,line) (hres == S_OK, "(1->2) isEqual failed: %08x\n", hres); |
| ok_(__FILE__,line) (b == exb, "(1->2) b=%x, expected %x\n", b, exb); |
| |
| b = 0xe0e0; |
| hres = IHTMLTxtRange_isEqual(range2, range1, &b); |
| ok_(__FILE__,line) (hres == S_OK, "(2->1) isEqual failed: %08x\n", hres); |
| ok_(__FILE__,line) (b == exb, "(2->1) b=%x, expected %x\n", b, exb); |
| |
| if(exb) { |
| test_range_inrange(range1, range2, VARIANT_TRUE); |
| test_range_inrange(range2, range1, VARIANT_TRUE); |
| } |
| } |
| |
| #define test_range_parent(r,t) _test_range_parent(__LINE__,r,t) |
| static void _test_range_parent(unsigned line, IHTMLTxtRange *range, elem_type_t type) |
| { |
| IHTMLElement *elem; |
| HRESULT hres; |
| |
| hres = IHTMLTxtRange_parentElement(range, &elem); |
| ok_(__FILE__,line) (hres == S_OK, "parentElement failed: %08x\n", hres); |
| |
| _test_elem_type(line, (IUnknown*)elem, type); |
| |
| IHTMLElement_Release(elem); |
| } |
| |
| #define test_elem_collection(c,t,l) _test_elem_collection(__LINE__,c,t,l) |
| static void _test_elem_collection(unsigned line, IHTMLElementCollection *col, |
| const elem_type_t *elem_types, long exlen) |
| { |
| long len; |
| DWORD i; |
| VARIANT name, index; |
| IDispatch *disp; |
| HRESULT hres; |
| |
| hres = IHTMLElementCollection_get_length(col, &len); |
| ok_(__FILE__,line) (hres == S_OK, "get_length failed: %08x\n", hres); |
| ok_(__FILE__,line) (len == exlen, "len=%ld, expected %ld\n", len, exlen); |
| |
| if(len > exlen) |
| len = exlen; |
| |
| V_VT(&index) = VT_EMPTY; |
| V_VT(&name) = VT_I4; |
| |
| for(i=0; i<len; i++) { |
| V_I4(&name) = i; |
| disp = (void*)0xdeadbeef; |
| hres = IHTMLElementCollection_item(col, name, index, &disp); |
| ok_(__FILE__,line) (hres == S_OK, "item(%d) failed: %08x\n", i, hres); |
| ok_(__FILE__,line) (disp != NULL, "item returned NULL\n"); |
| if(FAILED(hres) || !disp) |
| continue; |
| |
| _test_elem_type(line, (IUnknown*)disp, elem_types[i]); |
| IDispatch_Release(disp); |
| } |
| |
| V_I4(&name) = len; |
| disp = (void*)0xdeadbeef; |
| hres = IHTMLElementCollection_item(col, name, index, &disp); |
| ok_(__FILE__,line) (hres == S_OK, "item failed: %08x\n", hres); |
| ok_(__FILE__,line) (disp == NULL, "disp != NULL\n"); |
| |
| V_I4(&name) = -1; |
| disp = (void*)0xdeadbeef; |
| hres = IHTMLElementCollection_item(col, name, index, &disp); |
| ok_(__FILE__,line) (hres == E_INVALIDARG, "item failed: %08x, expected E_INVALIDARG\n", hres); |
| ok_(__FILE__,line) (disp == NULL, "disp != NULL\n"); |
| } |
| |
| static void test_elem_col_item(IHTMLElementCollection *col, LPCWSTR n, |
| const elem_type_t *elem_types, long len) |
| { |
| IHTMLElementCollection *elcol; |
| IDispatch *disp; |
| VARIANT name, index; |
| DWORD i; |
| HRESULT hres; |
| |
| V_VT(&index) = VT_EMPTY; |
| V_VT(&name) = VT_BSTR; |
| V_BSTR(&name) = SysAllocString(n); |
| |
| hres = IHTMLElementCollection_item(col, name, index, &disp); |
| ok(hres == S_OK, "item failed: %08x\n", hres); |
| |
| hres = IDispatch_QueryInterface(disp, &IID_IHTMLElementCollection, (void**)&elcol); |
| IDispatch_Release(disp); |
| ok(hres == S_OK, "Could not get IHTMLElementCollection interface: %08x\n", hres); |
| test_elem_collection(elcol, elem_types, len); |
| IHTMLElementCollection_Release(elcol); |
| |
| V_VT(&index) = VT_I4; |
| |
| for(i=0; i<len; i++) { |
| V_I4(&index) = i; |
| disp = (void*)0xdeadbeef; |
| hres = IHTMLElementCollection_item(col, name, index, &disp); |
| ok(hres == S_OK, "item failed: %08x\n", hres); |
| ok(disp != NULL, "disp == NULL\n"); |
| if(FAILED(hres) || !disp) |
| continue; |
| |
| test_elem_type((IUnknown*)disp, elem_types[i]); |
| |
| IDispatch_Release(disp); |
| } |
| |
| V_I4(&index) = len; |
| disp = (void*)0xdeadbeef; |
| hres = IHTMLElementCollection_item(col, name, index, &disp); |
| ok(hres == S_OK, "item failed: %08x\n", hres); |
| ok(disp == NULL, "disp != NULL\n"); |
| |
| V_I4(&index) = -1; |
| disp = (void*)0xdeadbeef; |
| hres = IHTMLElementCollection_item(col, name, index, &disp); |
| ok(hres == E_INVALIDARG, "item failed: %08x, expected E_INVALIDARG\n", hres); |
| ok(disp == NULL, "disp != NULL\n"); |
| |
| SysFreeString(V_BSTR(&name)); |
| } |
| |
| static IHTMLElement *get_elem_by_id(IHTMLDocument2 *doc, LPCWSTR id, BOOL expect_success) |
| { |
| IHTMLElementCollection *col; |
| IHTMLElement *elem; |
| IDispatch *disp = (void*)0xdeadbeef; |
| VARIANT name, index; |
| HRESULT hres; |
| |
| hres = IHTMLDocument2_get_all(doc, &col); |
| ok(hres == S_OK, "get_all failed: %08x\n", hres); |
| ok(col != NULL, "col == NULL\n"); |
| if(FAILED(hres) || !col) |
| return NULL; |
| |
| V_VT(&index) = VT_EMPTY; |
| V_VT(&name) = VT_BSTR; |
| V_BSTR(&name) = SysAllocString(id); |
| |
| hres = IHTMLElementCollection_item(col, name, index, &disp); |
| IHTMLElementCollection_Release(col); |
| SysFreeString(V_BSTR(&name)); |
| ok(hres == S_OK, "item failed: %08x\n", hres); |
| if(!expect_success) { |
| ok(disp == NULL, "disp != NULL\n"); |
| return NULL; |
| } |
| |
| ok(disp != NULL, "disp == NULL\n"); |
| if(!disp) |
| return NULL; |
| |
| hres = IDispatch_QueryInterface(disp, &IID_IHTMLElement, (void**)&elem); |
| IDispatch_Release(disp); |
| ok(hres == S_OK, "Could not get IHTMLElement interface: %08x\n", hres); |
| |
| return elem; |
| } |
| |
| static void test_select_elem(IHTMLSelectElement *select) |
| { |
| test_select_length(select, 2); |
| test_select_selidx(select, 0); |
| test_select_put_selidx(select, 1); |
| } |
| |
| static void test_create_option_elem(IHTMLDocument2 *doc) |
| { |
| IHTMLOptionElement *option; |
| |
| option = create_option_elem(doc, "test text", "test value"); |
| |
| test_option_put_text(option, "new text"); |
| test_option_put_value(option, "new value"); |
| |
| IHTMLOptionElement_Release(option); |
| } |
| |
| static IHTMLTxtRange *test_create_body_range(IHTMLDocument2 *doc) |
| { |
| IHTMLBodyElement *body; |
| IHTMLTxtRange *range; |
| IHTMLElement *elem; |
| HRESULT hres; |
| |
| hres = IHTMLDocument2_get_body(doc, &elem); |
| ok(hres == S_OK, "get_body failed: %08x\n", hres); |
| |
| hres = IHTMLElement_QueryInterface(elem, &IID_IHTMLBodyElement, (void**)&body); |
| IHTMLElement_Release(elem); |
| |
| hres = IHTMLBodyElement_createTextRange(body, &range); |
| IHTMLBodyElement_Release(body); |
| ok(hres == S_OK, "createTextRange failed: %08x\n", hres); |
| |
| return range; |
| } |
| |
| static void test_txtrange(IHTMLDocument2 *doc) |
| { |
| IHTMLTxtRange *body_range, *range, *range2; |
| IHTMLSelectionObject *selection; |
| IDispatch *disp_range; |
| HRESULT hres; |
| |
| body_range = test_create_body_range(doc); |
| |
| test_range_text(body_range, "test abc 123\r\nit's text"); |
| |
| hres = IHTMLTxtRange_duplicate(body_range, &range); |
| ok(hres == S_OK, "duplicate failed: %08x\n", hres); |
| |
| hres = IHTMLTxtRange_duplicate(body_range, &range2); |
| ok(hres == S_OK, "duplicate failed: %08x\n", hres); |
| test_range_isequal(range, range2, VARIANT_TRUE); |
| |
| test_range_text(range, "test abc 123\r\nit's text"); |
| test_range_text(body_range, "test abc 123\r\nit's text"); |
| |
| test_range_collapse(range, TRUE); |
| test_range_isequal(range, range2, VARIANT_FALSE); |
| test_range_inrange(range, range2, VARIANT_FALSE); |
| test_range_inrange(range2, range, VARIANT_TRUE); |
| IHTMLTxtRange_Release(range2); |
| |
| test_range_expand(range, wordW, VARIANT_TRUE, "test "); |
| test_range_expand(range, wordW, VARIANT_FALSE, "test "); |
| test_range_move(range, characterW, 2, 2); |
| test_range_expand(range, wordW, VARIANT_TRUE, "test "); |
| |
| test_range_collapse(range, FALSE); |
| test_range_expand(range, wordW, VARIANT_TRUE, "abc "); |
| |
| test_range_collapse(range, FALSE); |
| test_range_expand(range, wordW, VARIANT_TRUE, "123"); |
| test_range_expand(range, wordW, VARIANT_FALSE, "123"); |
| test_range_move(range, characterW, 2, 2); |
| test_range_expand(range, wordW, VARIANT_TRUE, "123"); |
| test_range_moveend(range, characterW, -5, -5); |
| test_range_text(range, NULL); |
| test_range_moveend(range, characterW, 3, 3); |
| test_range_text(range, "c 1"); |
| test_range_expand(range, texteditW, VARIANT_TRUE, "test abc 123\r\nit's text"); |
| test_range_collapse(range, TRUE); |
| test_range_move(range, characterW, 4, 4); |
| test_range_moveend(range, characterW, 1, 1); |
| test_range_text(range, " "); |
| test_range_move(range, wordW, 1, 1); |
| test_range_moveend(range, characterW, 2, 2); |
| test_range_text(range, "ab"); |
| |
| IHTMLTxtRange_Release(range); |
| |
| hres = IHTMLTxtRange_duplicate(body_range, &range); |
| ok(hres == S_OK, "duplicate failed: %08x\n", hres); |
| |
| test_range_text(range, "test abc 123\r\nit's text"); |
| test_range_move(range, characterW, 3, 3); |
| test_range_moveend(range, characterW, 1, 1); |
| test_range_text(range, "t"); |
| test_range_moveend(range, characterW, 3, 3); |
| test_range_text(range, "t ab"); |
| test_range_moveend(range, characterW, -2, -2); |
| test_range_text(range, "t "); |
| test_range_move(range, characterW, 6, 6); |
| test_range_moveend(range, characterW, 3, 3); |
| test_range_text(range, "123"); |
| test_range_moveend(range, characterW, 2, 2); |
| test_range_text(range, "123\r\ni"); |
| |
| IHTMLTxtRange_Release(range); |
| |
| hres = IHTMLTxtRange_duplicate(body_range, &range); |
| ok(hres == S_OK, "duplicate failed: %08x\n", hres); |
| |
| test_range_move(range, wordW, 1, 1); |
| test_range_moveend(range, characterW, 2, 2); |
| test_range_text(range, "ab"); |
| |
| test_range_move(range, characterW, -2, -2); |
| test_range_moveend(range, characterW, 2, 2); |
| test_range_text(range, "t "); |
| |
| test_range_move(range, wordW, 3, 3); |
| test_range_move(range, wordW, -2, -2); |
| test_range_moveend(range, characterW, 2, 2); |
| test_range_text(range, "ab"); |
| |
| test_range_move(range, characterW, -6, -5); |
| test_range_moveend(range, characterW, -1, 0); |
| test_range_moveend(range, characterW, -6, 0); |
| test_range_move(range, characterW, 2, 2); |
| test_range_moveend(range, characterW, 2, 2); |
| test_range_text(range, "st"); |
| test_range_moveend(range, characterW, -6, -4); |
| test_range_moveend(range, characterW, 2, 2); |
| |
| IHTMLTxtRange_Release(range); |
| |
| hres = IHTMLTxtRange_duplicate(body_range, &range); |
| ok(hres == S_OK, "duplicate failed: %08x\n", hres); |
| |
| test_range_move(range, wordW, 2, 2); |
| test_range_moveend(range, characterW, 2, 2); |
| test_range_text(range, "12"); |
| |
| test_range_move(range, characterW, 15, 14); |
| test_range_move(range, characterW, -2, -2); |
| test_range_moveend(range, characterW, 3, 2); |
| test_range_text(range, "t"); |
| test_range_moveend(range, characterW, -1, -1); |
| test_range_text(range, "t"); |
| test_range_expand(range, wordW, VARIANT_TRUE, "text"); |
| test_range_move(range, characterW, -2, -2); |
| test_range_moveend(range, characterW, 2, 2); |
| test_range_text(range, "s "); |
| test_range_move(range, characterW, 100, 7); |
| test_range_move(range, wordW, 1, 0); |
| test_range_move(range, characterW, -2, -2); |
| test_range_moveend(range, characterW, 3, 2); |
| test_range_text(range, "t"); |
| |
| IHTMLTxtRange_Release(range); |
| |
| hres = IHTMLTxtRange_duplicate(body_range, &range); |
| ok(hres == S_OK, "duplicate failed: %08x\n", hres); |
| |
| test_range_collapse(range, TRUE); |
| test_range_expand(range, wordW, VARIANT_TRUE, "test "); |
| test_range_put_text(range, "word"); |
| test_range_text(body_range, "wordabc 123\r\nit's text"); |
| test_range_text(range, NULL); |
| test_range_moveend(range, characterW, 3, 3); |
| test_range_text(range, "abc"); |
| test_range_movestart(range, characterW, -2, -2); |
| test_range_text(range, "rdabc"); |
| test_range_movestart(range, characterW, 3, 3); |
| test_range_text(range, "bc"); |
| test_range_movestart(range, characterW, 4, 4); |
| test_range_text(range, NULL); |
| test_range_movestart(range, characterW, -3, -3); |
| test_range_text(range, "c 1"); |
| test_range_movestart(range, characterW, -7, -6); |
| test_range_text(range, "wordabc 1"); |
| test_range_movestart(range, characterW, 100, 22); |
| test_range_text(range, NULL); |
| |
| IHTMLTxtRange_Release(range); |
| IHTMLTxtRange_Release(body_range); |
| |
| hres = IHTMLDocument2_get_selection(doc, &selection); |
| ok(hres == S_OK, "IHTMLDocument2_get_selection failed: %08x\n", hres); |
| |
| hres = IHTMLSelectionObject_createRange(selection, &disp_range); |
| ok(hres == S_OK, "IHTMLSelectionObject_createRange failed: %08x\n", hres); |
| IHTMLSelectionObject_Release(selection); |
| |
| hres = IDispatch_QueryInterface(disp_range, &IID_IHTMLTxtRange, (void **)&range); |
| ok(hres == S_OK, "Could not get IID_IHTMLTxtRange interface: 0x%08x\n", hres); |
| IDispatch_Release(disp_range); |
| |
| test_range_text(range, NULL); |
| test_range_moveend(range, characterW, 3, 3); |
| test_range_text(range, "wor"); |
| test_range_parent(range, ET_BODY); |
| test_range_expand(range, texteditW, VARIANT_TRUE, "wordabc 123\r\nit's text"); |
| test_range_expand(range, texteditW, VARIANT_TRUE, "wordabc 123\r\nit's text"); |
| test_range_move(range, characterW, 3, 3); |
| test_range_expand(range, wordW, VARIANT_TRUE, "wordabc "); |
| test_range_moveend(range, characterW, -4, -4); |
| test_range_put_text(range, "abc def "); |
| test_range_expand(range, texteditW, VARIANT_TRUE, "abc def abc 123\r\nit's text"); |
| test_range_move(range, wordW, 1, 1); |
| test_range_movestart(range, characterW, -1, -1); |
| test_range_text(range, " "); |
| test_range_move(range, wordW, 1, 1); |
| test_range_moveend(range, characterW, 3, 3); |
| test_range_text(range, "def"); |
| test_range_put_text(range, "xyz"); |
| test_range_moveend(range, characterW, 1, 1); |
| test_range_move(range, wordW, 1, 1); |
| test_range_moveend(range, characterW, 2, 2); |
| test_range_text(range, "ab"); |
| |
| IHTMLTxtRange_Release(range); |
| } |
| |
| static void test_compatmode(IHTMLDocument2 *doc) |
| { |
| IHTMLDocument5 *doc5; |
| BSTR mode; |
| HRESULT hres; |
| |
| hres = IHTMLDocument2_QueryInterface(doc, &IID_IHTMLDocument5, (void**)&doc5); |
| ok(hres == S_OK, "Could not get IHTMLDocument5 interface: %08x\n", hres); |
| if(FAILED(hres)) |
| return; |
| |
| hres = IHTMLDocument5_get_compatMode(doc5, &mode); |
| IHTMLDocument5_Release(doc5); |
| ok(hres == S_OK, "get_compatMode failed: %08x\n", hres); |
| ok(!strcmp_wa(mode, "BackCompat"), "compatMode=%s\n", dbgstr_w(mode)); |
| SysFreeString(mode); |
| } |
| |
| static void test_default_style(IHTMLStyle *style) |
| { |
| VARIANT_BOOL b; |
| VARIANT v; |
| BSTR str; |
| HRESULT hres; |
| |
| str = (void*)0xdeadbeef; |
| hres = IHTMLStyle_get_fontFamily(style, &str); |
| ok(hres == S_OK, "get_fontFamily failed: %08x\n", hres); |
| ok(!str, "fontFamily = %s\n", dbgstr_w(str)); |
| |
| str = (void*)0xdeadbeef; |
| hres = IHTMLStyle_get_fontWeight(style, &str); |
| ok(hres == S_OK, "get_fontWeight failed: %08x\n", hres); |
| ok(!str, "fontWeight = %s\n", dbgstr_w(str)); |
| |
| str = (void*)0xdeadbeef; |
| hres = IHTMLStyle_get_display(style, &str); |
| ok(hres == S_OK, "get_display failed: %08x\n", hres); |
| ok(!str, "display = %s\n", dbgstr_w(str)); |
| |
| str = (void*)0xdeadbeef; |
| hres = IHTMLStyle_get_visibility(style, &str); |
| ok(hres == S_OK, "get_visibility failed: %08x\n", hres); |
| ok(!str, "visibility = %s\n", dbgstr_w(str)); |
| |
| V_VT(&v) = VT_NULL; |
| hres = IHTMLStyle_get_fontSize(style, &v); |
| ok(hres == S_OK, "get_fontSize failed: %08x\n", hres); |
| ok(V_VT(&v) == VT_BSTR, "V_VT(fontSize) = %d\n", V_VT(&v)); |
| ok(!V_BSTR(&v), "V_BSTR(fontSize) = %s\n", dbgstr_w(V_BSTR(&v))); |
| |
| V_VT(&v) = VT_NULL; |
| hres = IHTMLStyle_get_color(style, &v); |
| ok(hres == S_OK, "get_color failed: %08x\n", hres); |
| ok(V_VT(&v) == VT_BSTR, "V_VT(color) = %d\n", V_VT(&v)); |
| ok(!V_BSTR(&v), "V_BSTR(color) = %s\n", dbgstr_w(V_BSTR(&v))); |
| |
| b = 0xfefe; |
| hres = IHTMLStyle_get_textDecorationUnderline(style, &b); |
| ok(hres == S_OK, "get_textDecorationUnderline failed: %08x\n", hres); |
| ok(b == VARIANT_FALSE, "textDecorationUnderline = %x\n", b); |
| |
| b = 0xfefe; |
| hres = IHTMLStyle_get_textDecorationLineThrough(style, &b); |
| ok(hres == S_OK, "get_textDecorationLineThrough failed: %08x\n", hres); |
| ok(b == VARIANT_FALSE, "textDecorationLineThrough = %x\n", b); |
| } |
| |
| static void test_default_selection(IHTMLDocument2 *doc) |
| { |
| IHTMLSelectionObject *selection; |
| IHTMLTxtRange *range; |
| IDispatch *disp; |
| BSTR str; |
| HRESULT hres; |
| |
| hres = IHTMLDocument2_get_selection(doc, &selection); |
| ok(hres == S_OK, "get_selection failed: %08x\n", hres); |
| |
| hres = IHTMLSelectionObject_get_type(selection, &str); |
| ok(hres == S_OK, "get_type failed: %08x\n", hres); |
| ok(!lstrcmpW(str, noneW), "type = %s\n", dbgstr_w(str)); |
| SysFreeString(str); |
| |
| hres = IHTMLSelectionObject_createRange(selection, &disp); |
| IHTMLSelectionObject_Release(selection); |
| ok(hres == S_OK, "createRange failed: %08x\n", hres); |
| |
| hres = IDispatch_QueryInterface(disp, &IID_IHTMLTxtRange, (void**)&range); |
| IDispatch_Release(disp); |
| ok(hres == S_OK, "Could not get IHTMLTxtRange interface: %08x\n", hres); |
| |
| test_range_text(range, NULL); |
| IHTMLTxtRange_Release(range); |
| } |
| |
| static void test_default_body(IHTMLBodyElement *body) |
| { |
| BSTR bstr; |
| HRESULT hres; |
| |
| bstr = (void*)0xdeadbeef; |
| hres = IHTMLBodyElement_get_background(body, &bstr); |
| ok(hres == S_OK, "get_background failed: %08x\n", hres); |
| ok(bstr == NULL, "bstr != NULL\n"); |
| } |
| |
| static void test_defaults(IHTMLDocument2 *doc) |
| { |
| IHTMLStyleSheetsCollection *stylesheetcol; |
| IHTMLBodyElement *body; |
| IHTMLElement *elem; |
| IHTMLStyle *style; |
| long l; |
| HRESULT hres; |
| |
| hres = IHTMLDocument2_get_body(doc, &elem); |
| ok(hres == S_OK, "get_body failed: %08x\n", hres); |
| |
| hres = IHTMLElement_QueryInterface(elem, &IID_IHTMLBodyElement, (void**)&body); |
| ok(hres == S_OK, "Could not get IHTMBodyElement: %08x\n", hres); |
| test_default_body(body); |
| IHTMLBodyElement_Release(body); |
| |
| hres = IHTMLElement_get_style(elem, &style); |
| IHTMLElement_Release(elem); |
| ok(hres == S_OK, "get_style failed: %08x\n", hres); |
| |
| test_default_style(style); |
| test_compatmode(doc); |
| |
| IHTMLStyle_Release(style); |
| |
| hres = IHTMLDocument2_get_styleSheets(doc, &stylesheetcol); |
| ok(hres == S_OK, "get_styleSheets failed: %08x\n", hres); |
| |
| l = 0xdeadbeef; |
| hres = IHTMLStyleSheetsCollection_get_length(stylesheetcol, &l); |
| ok(hres == S_OK, "get_length failed: %08x\n", hres); |
| ok(l == 0, "length = %ld\n", l); |
| |
| IHTMLStyleSheetsCollection_Release(stylesheetcol); |
| |
| test_default_selection(doc); |
| } |
| |
| static void test_stylesheet(IDispatch *disp) |
| { |
| IHTMLStyleSheetRulesCollection *col = NULL; |
| IHTMLStyleSheet *stylesheet; |
| HRESULT hres; |
| |
| hres = IDispatch_QueryInterface(disp, &IID_IHTMLStyleSheet, (void**)&stylesheet); |
| ok(hres == S_OK, "Could not get IHTMLStyleSheet: %08x\n", hres); |
| |
| hres = IHTMLStyleSheet_get_rules(stylesheet, &col); |
| ok(hres == S_OK, "get_rules failed: %08x\n", hres); |
| ok(col != NULL, "col == NULL\n"); |
| |
| IHTMLStyleSheetRulesCollection_Release(col); |
| IHTMLStyleSheet_Release(stylesheet); |
| } |
| |
| static void test_stylesheets(IHTMLDocument2 *doc) |
| { |
| IHTMLStyleSheetsCollection *col = NULL; |
| VARIANT idx, res; |
| long len = 0; |
| HRESULT hres; |
| |
| hres = IHTMLDocument2_get_styleSheets(doc, &col); |
| ok(hres == S_OK, "get_styleSheets failed: %08x\n", hres); |
| ok(col != NULL, "col == NULL\n"); |
| |
| hres = IHTMLStyleSheetsCollection_get_length(col, &len); |
| ok(hres == S_OK, "get_length failed: %08x\n", hres); |
| ok(len == 1, "len=%ld\n", len); |
| |
| VariantInit(&res); |
| V_VT(&idx) = VT_I4; |
| V_I4(&idx) = 0; |
| |
| hres = IHTMLStyleSheetsCollection_item(col, &idx, &res); |
| ok(hres == S_OK, "item failed: %08x\n", hres); |
| ok(V_VT(&res) == VT_DISPATCH, "V_VT(res) = %d\n", V_VT(&res)); |
| ok(V_DISPATCH(&res) != NULL, "V_DISPATCH(&res) == NULL\n"); |
| test_stylesheet(V_DISPATCH(&res)); |
| VariantClear(&res); |
| |
| V_VT(&res) = VT_I4; |
| V_VT(&idx) = VT_I4; |
| V_I4(&idx) = 1; |
| |
| hres = IHTMLStyleSheetsCollection_item(col, &idx, &res); |
| ok(hres == E_INVALIDARG, "item failed: %08x, expected E_INVALIDARG\n", hres); |
| ok(V_VT(&res) == VT_EMPTY, "V_VT(res) = %d\n", V_VT(&res)); |
| ok(V_DISPATCH(&res) != NULL, "V_DISPATCH(&res) == NULL\n"); |
| VariantClear(&res); |
| |
| IHTMLStyleSheetsCollection_Release(col); |
| } |
| |
| static void test_elems(IHTMLDocument2 *doc) |
| { |
| IHTMLElementCollection *col; |
| IHTMLElement *elem; |
| IDispatch *disp; |
| HRESULT hres; |
| |
| static const WCHAR xW[] = {'x',0}; |
| static const WCHAR sW[] = {'s',0}; |
| static const WCHAR xxxW[] = {'x','x','x',0}; |
| |
| static const elem_type_t all_types[] = { |
| ET_HTML, |
| ET_HEAD, |
| ET_TITLE, |
| ET_STYLE, |
| ET_BODY, |
| ET_A, |
| ET_INPUT, |
| ET_SELECT, |
| ET_OPTION, |
| ET_OPTION, |
| ET_TEXTAREA, |
| ET_TABLE, |
| ET_TBODY |
| }; |
| |
| static const elem_type_t item_types[] = { |
| ET_A, |
| ET_OPTION, |
| ET_TEXTAREA |
| }; |
| |
| hres = IHTMLDocument2_get_all(doc, &col); |
| ok(hres == S_OK, "get_all failed: %08x\n", hres); |
| test_elem_collection(col, all_types, sizeof(all_types)/sizeof(all_types[0])); |
| test_elem_col_item(col, xW, item_types, sizeof(item_types)/sizeof(item_types[0])); |
| IHTMLElementCollection_Release(col); |
| |
| elem = get_doc_elem(doc); |
| ok(hres == S_OK, "get_documentElement failed: %08x\n", hres); |
| hres = IHTMLElement_get_all(elem, &disp); |
| IHTMLElement_Release(elem); |
| ok(hres == S_OK, "get_all failed: %08x\n", hres); |
| |
| hres = IDispatch_QueryInterface(disp, &IID_IHTMLElementCollection, (void**)&col); |
| IDispatch_Release(disp); |
| ok(hres == S_OK, "Could not get IHTMLElementCollection: %08x\n", hres); |
| test_elem_collection(col, all_types+1, sizeof(all_types)/sizeof(all_types[0])-1); |
| IHTMLElementCollection_Release(col); |
| |
| get_elem_by_id(doc, xxxW, FALSE); |
| elem = get_elem_by_id(doc, sW, TRUE); |
| if(elem) { |
| IHTMLSelectElement *select; |
| |
| hres = IHTMLElement_QueryInterface(elem, &IID_IHTMLSelectElement, (void**)&select); |
| ok(hres == S_OK, "Could not get IHTMLSelectElement interface: %08x\n", hres); |
| |
| test_select_elem(select); |
| |
| IHTMLSelectElement_Release(select); |
| IHTMLElement_Release(elem); |
| } |
| |
| test_stylesheets(doc); |
| test_create_option_elem(doc); |
| } |
| |
| static void test_exec(IUnknown *unk, const GUID *grpid, DWORD cmdid, VARIANT *in, VARIANT *out) |
| { |
| IOleCommandTarget *cmdtrg; |
| HRESULT hres; |
| |
| hres = IHTMLTxtRange_QueryInterface(unk, &IID_IOleCommandTarget, (void**)&cmdtrg); |
| ok(hres == S_OK, "Could not get IOleCommandTarget interface: %08x\n", hres); |
| |
| hres = IOleCommandTarget_Exec(cmdtrg, grpid, cmdid, 0, in, out); |
| ok(hres == S_OK, "Exec failed: %08x\n", hres); |
| |
| IOleCommandTarget_Release(cmdtrg); |
| } |
| |
| static void test_indent(IHTMLDocument2 *doc) |
| { |
| IHTMLElementCollection *col; |
| IHTMLTxtRange *range; |
| HRESULT hres; |
| |
| static const elem_type_t all_types[] = { |
| ET_HTML, |
| ET_HEAD, |
| ET_TITLE, |
| ET_BODY, |
| ET_BR, |
| ET_A, |
| }; |
| |
| static const elem_type_t indent_types[] = { |
| ET_HTML, |
| ET_HEAD, |
| ET_TITLE, |
| ET_BODY, |
| ET_BLOCKQUOTE, |
| ET_P, |
| ET_BR, |
| ET_A, |
| }; |
| |
| hres = IHTMLDocument2_get_all(doc, &col); |
| ok(hres == S_OK, "get_all failed: %08x\n", hres); |
| test_elem_collection(col, all_types, sizeof(all_types)/sizeof(all_types[0])); |
| IHTMLElementCollection_Release(col); |
| |
| range = test_create_body_range(doc); |
| test_exec((IUnknown*)range, &CGID_MSHTML, IDM_INDENT, NULL, NULL); |
| IHTMLTxtRange_Release(range); |
| |
| hres = IHTMLDocument2_get_all(doc, &col); |
| ok(hres == S_OK, "get_all failed: %08x\n", hres); |
| test_elem_collection(col, indent_types, sizeof(indent_types)/sizeof(indent_types[0])); |
| IHTMLElementCollection_Release(col); |
| } |
| |
| static IHTMLDocument2 *notif_doc; |
| static BOOL doc_complete; |
| |
| static HRESULT WINAPI PropertyNotifySink_QueryInterface(IPropertyNotifySink *iface, |
| REFIID riid, void**ppv) |
| { |
| if(IsEqualGUID(&IID_IPropertyNotifySink, riid)) { |
| *ppv = iface; |
| return S_OK; |
| } |
| |
| ok(0, "unexpected call\n"); |
| return E_NOINTERFACE; |
| } |
| |
| static ULONG WINAPI PropertyNotifySink_AddRef(IPropertyNotifySink *iface) |
| { |
| return 2; |
| } |
| |
| static ULONG WINAPI PropertyNotifySink_Release(IPropertyNotifySink *iface) |
| { |
| return 1; |
| } |
| |
| static HRESULT WINAPI PropertyNotifySink_OnChanged(IPropertyNotifySink *iface, DISPID dispID) |
| { |
| if(dispID == DISPID_READYSTATE){ |
| BSTR state; |
| HRESULT hres; |
| |
| static const WCHAR completeW[] = {'c','o','m','p','l','e','t','e',0}; |
| |
| hres = IHTMLDocument2_get_readyState(notif_doc, &state); |
| ok(hres == S_OK, "get_readyState failed: %08x\n", hres); |
| |
| if(!lstrcmpW(state, completeW)) |
| doc_complete = TRUE; |
| |
| SysFreeString(state); |
| } |
| |
| return S_OK; |
| } |
| |
| static HRESULT WINAPI PropertyNotifySink_OnRequestEdit(IPropertyNotifySink *iface, DISPID dispID) |
| { |
| ok(0, "unexpected call\n"); |
| return E_NOTIMPL; |
| } |
| |
| static IPropertyNotifySinkVtbl PropertyNotifySinkVtbl = { |
| PropertyNotifySink_QueryInterface, |
| PropertyNotifySink_AddRef, |
| PropertyNotifySink_Release, |
| PropertyNotifySink_OnChanged, |
| PropertyNotifySink_OnRequestEdit |
| }; |
| |
| static IPropertyNotifySink PropertyNotifySink = { &PropertyNotifySinkVtbl }; |
| |
| static IHTMLDocument2 *create_doc_with_string(const char *str) |
| { |
| IPersistStreamInit *init; |
| IStream *stream; |
| IHTMLDocument2 *doc; |
| HGLOBAL mem; |
| SIZE_T len; |
| |
| notif_doc = doc = create_document(); |
| if(!doc) |
| return NULL; |
| |
| doc_complete = FALSE; |
| len = strlen(str); |
| mem = GlobalAlloc(0, len); |
| memcpy(mem, str, len); |
| CreateStreamOnHGlobal(mem, TRUE, &stream); |
| |
| IHTMLDocument2_QueryInterface(doc, &IID_IPersistStreamInit, (void**)&init); |
| |
| IPersistStreamInit_Load(init, stream); |
| IPersistStreamInit_Release(init); |
| IStream_Release(stream); |
| |
| return doc; |
| } |
| |
| static void do_advise(IUnknown *unk, REFIID riid, IUnknown *unk_advise) |
| { |
| IConnectionPointContainer *container; |
| IConnectionPoint *cp; |
| DWORD cookie; |
| HRESULT hres; |
| |
| hres = IUnknown_QueryInterface(unk, &IID_IConnectionPointContainer, (void**)&container); |
| ok(hres == S_OK, "QueryInterface(IID_IConnectionPointContainer) failed: %08x\n", hres); |
| |
| hres = IConnectionPointContainer_FindConnectionPoint(container, riid, &cp); |
| IConnectionPointContainer_Release(container); |
| ok(hres == S_OK, "FindConnectionPoint failed: %08x\n", hres); |
| |
| hres = IConnectionPoint_Advise(cp, unk_advise, &cookie); |
| IConnectionPoint_Release(cp); |
| ok(hres == S_OK, "Advise failed: %08x\n", hres); |
| } |
| |
| typedef void (*domtest_t)(IHTMLDocument2*); |
| |
| static void run_domtest(const char *str, domtest_t test) |
| { |
| IHTMLDocument2 *doc; |
| IHTMLElement *body = NULL; |
| ULONG ref; |
| MSG msg; |
| HRESULT hres; |
| |
| doc = create_doc_with_string(str); |
| do_advise((IUnknown*)doc, &IID_IPropertyNotifySink, (IUnknown*)&PropertyNotifySink); |
| |
| while(!doc_complete && GetMessage(&msg, NULL, 0, 0)) { |
| TranslateMessage(&msg); |
| DispatchMessage(&msg); |
| } |
| |
| hres = IHTMLDocument2_get_body(doc, &body); |
| ok(hres == S_OK, "get_body failed: %08x\n", hres); |
| |
| if(body) { |
| IHTMLElement_Release(body); |
| test(doc); |
| }else { |
| skip("Could not get document body. Assuming no Gecko installed.\n"); |
| } |
| |
| ref = IHTMLDocument2_Release(doc); |
| ok(!ref, "ref = %d\n", ref); |
| } |
| |
| static void gecko_installer_workaround(BOOL disable) |
| { |
| HKEY hkey; |
| DWORD res; |
| |
| static BOOL has_url = FALSE; |
| static char url[2048]; |
| |
| if(!disable && !has_url) |
| return; |
| |
| res = RegOpenKey(HKEY_CURRENT_USER, "Software\\Wine\\MSHTML", &hkey); |
| if(res != ERROR_SUCCESS) |
| return; |
| |
| if(disable) { |
| DWORD type, size = sizeof(url); |
| |
| res = RegQueryValueEx(hkey, "GeckoUrl", NULL, &type, (PVOID)url, &size); |
| if(res == ERROR_SUCCESS && type == REG_SZ) |
| has_url = TRUE; |
| |
| RegDeleteValue(hkey, "GeckoUrl"); |
| }else { |
| RegSetValueEx(hkey, "GeckoUrl", 0, REG_SZ, (PVOID)url, lstrlenA(url)+1); |
| } |
| |
| RegCloseKey(hkey); |
| } |
| |
| START_TEST(dom) |
| { |
| gecko_installer_workaround(TRUE); |
| CoInitialize(NULL); |
| |
| run_domtest(doc_str1, test_doc_elem); |
| run_domtest(doc_str2, test_txtrange); |
| run_domtest(elem_test_str, test_elems); |
| run_domtest(doc_blank, test_defaults); |
| run_domtest(indent_test_str, test_indent); |
| |
| CoUninitialize(); |
| gecko_installer_workaround(FALSE); |
| } |