diff --git a/dlls/mshtml/Makefile.in b/dlls/mshtml/Makefile.in index 26311af3136..e479bf881b3 100644 --- a/dlls/mshtml/Makefile.in +++ b/dlls/mshtml/Makefile.in @@ -31,6 +31,7 @@ C_SRCS = \ htmliframe.c \ htmlimg.c \ htmlinput.c \ + htmllink.c \ htmllocation.c \ htmlmeta.c \ htmlnode.c \ diff --git a/dlls/mshtml/htmlelem.c b/dlls/mshtml/htmlelem.c index 157d7aff13f..d11ebbc1afd 100644 --- a/dlls/mshtml/htmlelem.c +++ b/dlls/mshtml/htmlelem.c @@ -45,6 +45,7 @@ static const WCHAR headW[] = {'H','E','A','D',0}; static const WCHAR iframeW[] = {'I','F','R','A','M','E',0}; static const WCHAR imgW[] = {'I','M','G',0}; static const WCHAR inputW[] = {'I','N','P','U','T',0}; +static const WCHAR linkW[] = {'L','I','N','K',0}; static const WCHAR metaW[] = {'M','E','T','A',0}; static const WCHAR objectW[] = {'O','B','J','E','C','T',0}; static const WCHAR optionW[] = {'O','P','T','I','O','N',0}; @@ -72,6 +73,7 @@ static const tag_desc_t tag_descs[] = { {iframeW, HTMLIFrame_Create}, {imgW, HTMLImgElement_Create}, {inputW, HTMLInputElement_Create}, + {linkW, HTMLLinkElement_Create}, {metaW, HTMLMetaElement_Create}, {objectW, HTMLObjectElement_Create}, {optionW, HTMLOptionElement_Create}, diff --git a/dlls/mshtml/htmllink.c b/dlls/mshtml/htmllink.c new file mode 100644 index 00000000000..837538e84b4 --- /dev/null +++ b/dlls/mshtml/htmllink.c @@ -0,0 +1,334 @@ + /* + * Copyright 2012 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 + */ + +#include + +#define COBJMACROS + +#include "windef.h" +#include "winbase.h" +#include "winuser.h" +#include "ole2.h" + +#include "wine/debug.h" + +#include "mshtml_private.h" + +WINE_DEFAULT_DEBUG_CHANNEL(mshtml); + +typedef struct { + HTMLElement element; + IHTMLLinkElement IHTMLLinkElement_iface; +} HTMLLinkElement; + +static inline HTMLLinkElement *impl_from_IHTMLLinkElement(IHTMLLinkElement *iface) +{ + return CONTAINING_RECORD(iface, HTMLLinkElement, IHTMLLinkElement_iface); +} + +static HRESULT WINAPI HTMLLinkElement_QueryInterface(IHTMLLinkElement *iface, + REFIID riid, void **ppv) +{ + HTMLLinkElement *This = impl_from_IHTMLLinkElement(iface); + + return IHTMLDOMNode_QueryInterface(&This->element.node.IHTMLDOMNode_iface, riid, ppv); +} + +static ULONG WINAPI HTMLLinkElement_AddRef(IHTMLLinkElement *iface) +{ + HTMLLinkElement *This = impl_from_IHTMLLinkElement(iface); + + return IHTMLDOMNode_AddRef(&This->element.node.IHTMLDOMNode_iface); +} + +static ULONG WINAPI HTMLLinkElement_Release(IHTMLLinkElement *iface) +{ + HTMLLinkElement *This = impl_from_IHTMLLinkElement(iface); + + return IHTMLDOMNode_Release(&This->element.node.IHTMLDOMNode_iface); +} + +static HRESULT WINAPI HTMLLinkElement_GetTypeInfoCount(IHTMLLinkElement *iface, UINT *pctinfo) +{ + HTMLLinkElement *This = impl_from_IHTMLLinkElement(iface); + + return IDispatchEx_GetTypeInfoCount(&This->element.node.dispex.IDispatchEx_iface, pctinfo); +} + +static HRESULT WINAPI HTMLLinkElement_GetTypeInfo(IHTMLLinkElement *iface, UINT iTInfo, + LCID lcid, ITypeInfo **ppTInfo) +{ + HTMLLinkElement *This = impl_from_IHTMLLinkElement(iface); + + return IDispatchEx_GetTypeInfo(&This->element.node.dispex.IDispatchEx_iface, iTInfo, lcid, + ppTInfo); +} + +static HRESULT WINAPI HTMLLinkElement_GetIDsOfNames(IHTMLLinkElement *iface, REFIID riid, + LPOLESTR *rgszNames, UINT cNames, + LCID lcid, DISPID *rgDispId) +{ + HTMLLinkElement *This = impl_from_IHTMLLinkElement(iface); + + return IDispatchEx_GetIDsOfNames(&This->element.node.dispex.IDispatchEx_iface, riid, rgszNames, + cNames, lcid, rgDispId); +} + +static HRESULT WINAPI HTMLLinkElement_Invoke(IHTMLLinkElement *iface, DISPID dispIdMember, + REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS *pDispParams, + VARIANT *pVarResult, EXCEPINFO *pExcepInfo, UINT *puArgErr) +{ + HTMLLinkElement *This = impl_from_IHTMLLinkElement(iface); + + return IDispatchEx_Invoke(&This->element.node.dispex.IDispatchEx_iface, dispIdMember, riid, + lcid, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr); +} + +static HRESULT WINAPI HTMLLinkElement_put_href(IHTMLLinkElement *iface, BSTR v) +{ + HTMLLinkElement *This = impl_from_IHTMLLinkElement(iface); + FIXME("(%p)->(%s)\n", This, debugstr_w(v)); + return E_NOTIMPL; +} + +static HRESULT WINAPI HTMLLinkElement_get_href(IHTMLLinkElement *iface, BSTR *p) +{ + HTMLLinkElement *This = impl_from_IHTMLLinkElement(iface); + FIXME("(%p)->(%p)\n", This, p); + return E_NOTIMPL; +} + +static HRESULT WINAPI HTMLLinkElement_put_rel(IHTMLLinkElement *iface, BSTR v) +{ + HTMLLinkElement *This = impl_from_IHTMLLinkElement(iface); + FIXME("(%p)->(%s)\n", This, debugstr_w(v)); + return E_NOTIMPL; +} + +static HRESULT WINAPI HTMLLinkElement_get_rel(IHTMLLinkElement *iface, BSTR *p) +{ + HTMLLinkElement *This = impl_from_IHTMLLinkElement(iface); + FIXME("(%p)->(%p)\n", This, p); + return E_NOTIMPL; +} + +static HRESULT WINAPI HTMLLinkElement_put_rev(IHTMLLinkElement *iface, BSTR v) +{ + HTMLLinkElement *This = impl_from_IHTMLLinkElement(iface); + FIXME("(%p)->(%s)\n", This, debugstr_w(v)); + return E_NOTIMPL; +} + +static HRESULT WINAPI HTMLLinkElement_get_rev(IHTMLLinkElement *iface, BSTR *p) +{ + HTMLLinkElement *This = impl_from_IHTMLLinkElement(iface); + FIXME("(%p)->(%p)\n", This, p); + return E_NOTIMPL; +} + +static HRESULT WINAPI HTMLLinkElement_put_type(IHTMLLinkElement *iface, BSTR v) +{ + HTMLLinkElement *This = impl_from_IHTMLLinkElement(iface); + FIXME("(%p)->(%s)\n", This, debugstr_w(v)); + return E_NOTIMPL; +} + +static HRESULT WINAPI HTMLLinkElement_get_type(IHTMLLinkElement *iface, BSTR *p) +{ + HTMLLinkElement *This = impl_from_IHTMLLinkElement(iface); + FIXME("(%p)->(%p)\n", This, p); + return E_NOTIMPL; +} + +static HRESULT WINAPI HTMLLinkElement_get_readyState(IHTMLLinkElement *iface, BSTR *p) +{ + HTMLLinkElement *This = impl_from_IHTMLLinkElement(iface); + FIXME("(%p)->(%p)\n", This, p); + return E_NOTIMPL; +} + +static HRESULT WINAPI HTMLLinkElement_put_onreadystatechange(IHTMLLinkElement *iface, VARIANT v) +{ + HTMLLinkElement *This = impl_from_IHTMLLinkElement(iface); + FIXME("(%p)->(%s)\n", This, debugstr_variant(&v)); + return E_NOTIMPL; +} + +static HRESULT WINAPI HTMLLinkElement_get_onreadystatechange(IHTMLLinkElement *iface, VARIANT *p) +{ + HTMLLinkElement *This = impl_from_IHTMLLinkElement(iface); + FIXME("(%p)->(%p)\n", This, p); + return E_NOTIMPL; +} + +static HRESULT WINAPI HTMLLinkElement_put_onload(IHTMLLinkElement *iface, VARIANT v) +{ + HTMLLinkElement *This = impl_from_IHTMLLinkElement(iface); + FIXME("(%p)->(%s)\n", This, debugstr_variant(&v)); + return E_NOTIMPL; +} + +static HRESULT WINAPI HTMLLinkElement_get_onload(IHTMLLinkElement *iface, VARIANT *p) +{ + HTMLLinkElement *This = impl_from_IHTMLLinkElement(iface); + FIXME("(%p)->(%p)\n", This, p); + return E_NOTIMPL; +} + +static HRESULT WINAPI HTMLLinkElement_put_onerror(IHTMLLinkElement *iface, VARIANT v) +{ + HTMLLinkElement *This = impl_from_IHTMLLinkElement(iface); + FIXME("(%p)->(%s)\n", This, debugstr_variant(&v)); + return E_NOTIMPL; +} + +static HRESULT WINAPI HTMLLinkElement_get_onerror(IHTMLLinkElement *iface, VARIANT *p) +{ + HTMLLinkElement *This = impl_from_IHTMLLinkElement(iface); + FIXME("(%p)->(%p)\n", This, p); + return E_NOTIMPL; +} + +static HRESULT WINAPI HTMLLinkElement_get_styleSheet(IHTMLLinkElement *iface, IHTMLStyleSheet **p) +{ + HTMLLinkElement *This = impl_from_IHTMLLinkElement(iface); + FIXME("(%p)->(%p)\n", This, p); + return E_NOTIMPL; +} + +static HRESULT WINAPI HTMLLinkElement_put_disabled(IHTMLLinkElement *iface, VARIANT_BOOL v) +{ + HTMLLinkElement *This = impl_from_IHTMLLinkElement(iface); + FIXME("(%p)->(%x)\n", This, v); + return E_NOTIMPL; +} + +static HRESULT WINAPI HTMLLinkElement_get_disabled(IHTMLLinkElement *iface, VARIANT_BOOL *p) +{ + HTMLLinkElement *This = impl_from_IHTMLLinkElement(iface); + FIXME("(%p)->(%p)\n", This, p); + return E_NOTIMPL; +} + +static HRESULT WINAPI HTMLLinkElement_put_media(IHTMLLinkElement *iface, BSTR v) +{ + HTMLLinkElement *This = impl_from_IHTMLLinkElement(iface); + FIXME("(%p)->(%s)\n", This, debugstr_w(v)); + return E_NOTIMPL; +} + +static HRESULT WINAPI HTMLLinkElement_get_media(IHTMLLinkElement *iface, BSTR *p) +{ + HTMLLinkElement *This = impl_from_IHTMLLinkElement(iface); + FIXME("(%p)->(%p)\n", This, p); + return E_NOTIMPL; +} + +static const IHTMLLinkElementVtbl HTMLLinkElementVtbl = { + HTMLLinkElement_QueryInterface, + HTMLLinkElement_AddRef, + HTMLLinkElement_Release, + HTMLLinkElement_GetTypeInfoCount, + HTMLLinkElement_GetTypeInfo, + HTMLLinkElement_GetIDsOfNames, + HTMLLinkElement_Invoke, + HTMLLinkElement_put_href, + HTMLLinkElement_get_href, + HTMLLinkElement_put_rel, + HTMLLinkElement_get_rel, + HTMLLinkElement_put_rev, + HTMLLinkElement_get_rev, + HTMLLinkElement_put_type, + HTMLLinkElement_get_type, + HTMLLinkElement_get_readyState, + HTMLLinkElement_put_onreadystatechange, + HTMLLinkElement_get_onreadystatechange, + HTMLLinkElement_put_onload, + HTMLLinkElement_get_onload, + HTMLLinkElement_put_onerror, + HTMLLinkElement_get_onerror, + HTMLLinkElement_get_styleSheet, + HTMLLinkElement_put_disabled, + HTMLLinkElement_get_disabled, + HTMLLinkElement_put_media, + HTMLLinkElement_get_media +}; + +static inline HTMLLinkElement *impl_from_HTMLDOMNode(HTMLDOMNode *iface) +{ + return CONTAINING_RECORD(iface, HTMLLinkElement, element.node); +} + +static HRESULT HTMLLinkElement_QI(HTMLDOMNode *iface, REFIID riid, void **ppv) +{ + HTMLLinkElement *This = impl_from_HTMLDOMNode(iface); + + if(IsEqualGUID(&IID_IHTMLLinkElement, riid)) { + TRACE("(%p)->(IID_IHTMLLinkElement %p)\n", This, ppv); + *ppv = &This->IHTMLLinkElement_iface; + }else { + return HTMLElement_QI(&This->element.node, riid, ppv); + } + + IUnknown_AddRef((IUnknown*)*ppv); + return S_OK; +} + +static void HTMLLinkElement_destructor(HTMLDOMNode *iface) +{ + HTMLLinkElement *This = impl_from_HTMLDOMNode(iface); + + HTMLElement_destructor(&This->element.node); +} + +static const NodeImplVtbl HTMLLinkElementImplVtbl = { + HTMLLinkElement_QI, + HTMLLinkElement_destructor, + HTMLElement_clone, + HTMLElement_handle_event, + HTMLElement_get_attr_col +}; + +static const tid_t HTMLLinkElement_iface_tids[] = { + HTMLELEMENT_TIDS, + IHTMLLinkElement_tid, + 0 +}; +static dispex_static_data_t HTMLLinkElement_dispex = { + NULL, + DispHTMLLinkElement_tid, + NULL, + HTMLLinkElement_iface_tids +}; + +HRESULT HTMLLinkElement_Create(HTMLDocumentNode *doc, nsIDOMHTMLElement *nselem, HTMLElement **elem) +{ + HTMLLinkElement *ret; + + ret = heap_alloc_zero(sizeof(*ret)); + if(!ret) + return E_OUTOFMEMORY; + + ret->IHTMLLinkElement_iface.lpVtbl = &HTMLLinkElementVtbl; + ret->element.node.vtbl = &HTMLLinkElementImplVtbl; + + HTMLElement_Init(&ret->element, doc, nselem, &HTMLLinkElement_dispex); + + *elem = &ret->element; + return S_OK; +} diff --git a/dlls/mshtml/mshtml_private.h b/dlls/mshtml/mshtml_private.h index d199d21cf51..dc51565a570 100644 --- a/dlls/mshtml/mshtml_private.h +++ b/dlls/mshtml/mshtml_private.h @@ -92,6 +92,7 @@ typedef struct event_target_t event_target_t; XDIID(DispHTMLIFrame) \ XDIID(DispHTMLImg) \ XDIID(DispHTMLInputElement) \ + XDIID(DispHTMLLinkElement) \ XDIID(DispHTMLLocation) \ XDIID(DispHTMLMetaElement) \ XDIID(DispHTMLNavigator) \ @@ -151,6 +152,7 @@ typedef struct event_target_t event_target_t; XIID(IHTMLImageElementFactory) \ XIID(IHTMLImgElement) \ XIID(IHTMLInputElement) \ + XIID(IHTMLLinkElement) \ XIID(IHTMLLocation) \ XIID(IHTMLMetaElement) \ XIID(IHTMLMimeTypesCollection) \ @@ -879,6 +881,7 @@ HRESULT HTMLIFrame_Create(HTMLDocumentNode*,nsIDOMHTMLElement*,HTMLElement**) DE HRESULT HTMLStyleElement_Create(HTMLDocumentNode*,nsIDOMHTMLElement*,HTMLElement**) DECLSPEC_HIDDEN; HRESULT HTMLImgElement_Create(HTMLDocumentNode*,nsIDOMHTMLElement*,HTMLElement**) DECLSPEC_HIDDEN; HRESULT HTMLInputElement_Create(HTMLDocumentNode*,nsIDOMHTMLElement*,HTMLElement**) DECLSPEC_HIDDEN; +HRESULT HTMLLinkElement_Create(HTMLDocumentNode*,nsIDOMHTMLElement*,HTMLElement**) DECLSPEC_HIDDEN; HRESULT HTMLMetaElement_Create(HTMLDocumentNode*,nsIDOMHTMLElement*,HTMLElement**) DECLSPEC_HIDDEN; HRESULT HTMLObjectElement_Create(HTMLDocumentNode*,nsIDOMHTMLElement*,HTMLElement**) DECLSPEC_HIDDEN; HRESULT HTMLOptionElement_Create(HTMLDocumentNode*,nsIDOMHTMLElement*,HTMLElement**) DECLSPEC_HIDDEN; diff --git a/dlls/mshtml/tests/dom.c b/dlls/mshtml/tests/dom.c index d55c3d76964..7c84d0ce869 100644 --- a/dlls/mshtml/tests/dom.c +++ b/dlls/mshtml/tests/dom.c @@ -48,6 +48,7 @@ static const char range_test2_str[] = static const char elem_test_str[] = "test" "" + "" "text test" "link" "" @@ -119,7 +120,8 @@ typedef enum { ET_EMBED, ET_DIV, ET_META, - ET_NOSCRIPT + ET_NOSCRIPT, + ET_LINK } elem_type_t; static const IID * const none_iids[] = { @@ -320,6 +322,13 @@ static const IID * const meta_iids[] = { NULL }; +static const IID * const link_iids[] = { + ELEM_IFACES, + &IID_IHTMLLinkElement, + &IID_IConnectionPointContainer, + NULL +}; + static const IID * const object_iids[] = { ELEM_IFACES, &IID_IHTMLObjectElement, @@ -432,7 +441,8 @@ static const elem_type_info_t elem_type_infos[] = { {"EMBED", embed_iids, &DIID_DispHTMLEmbed}, {"DIV", elem_iids, NULL}, {"META", meta_iids, &DIID_DispHTMLMetaElement}, - {"NOSCRIPT", elem_iids, NULL /*&DIID_DispHTMLNoShowElement*/} + {"NOSCRIPT", elem_iids, NULL /*&DIID_DispHTMLNoShowElement*/}, + {"LINK", link_iids, &DIID_DispHTMLLinkElement} }; static const char *dbgstr_guid(REFIID riid) @@ -5277,6 +5287,7 @@ static void test_elems(IHTMLDocument2 *doc) ET_TITLE, ET_STYLE, ET_META, + ET_LINK, ET_BODY, ET_COMMENT, ET_A,