/* * DOM Document implementation * * Copyright 2005 Mike McCormack * * 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 #include "config.h" #include #ifdef HAVE_LIBXML2 # include # include #endif #include "windef.h" #include "winbase.h" #include "winuser.h" #include "winnls.h" #include "ole2.h" #include "msxml6.h" #include "msxml_private.h" #include "wine/debug.h" WINE_DEFAULT_DEBUG_CHANNEL(msxml); #ifdef HAVE_LIBXML2 static const xmlChar DT_prefix[] = "dt"; static const xmlChar DT_nsURI[] = "urn:schemas-microsoft-com:datatypes"; typedef struct _domelem { xmlnode node; IXMLDOMElement IXMLDOMElement_iface; LONG ref; } domelem; static inline domelem *impl_from_IXMLDOMElement( IXMLDOMElement *iface ) { return CONTAINING_RECORD(iface, domelem, IXMLDOMElement_iface); } static inline xmlNodePtr get_element( const domelem *This ) { return This->node.node; } static HRESULT WINAPI domelem_QueryInterface( IXMLDOMElement *iface, REFIID riid, void** ppvObject ) { domelem *This = impl_from_IXMLDOMElement( iface ); TRACE("(%p)->(%s %p)\n", This, debugstr_guid(riid), ppvObject); if ( IsEqualGUID( riid, &IID_IXMLDOMElement ) || IsEqualGUID( riid, &IID_IXMLDOMNode ) || IsEqualGUID( riid, &IID_IDispatch ) || IsEqualGUID( riid, &IID_IUnknown ) ) { *ppvObject = &This->IXMLDOMElement_iface; } else if(node_query_interface(&This->node, riid, ppvObject)) { return *ppvObject ? S_OK : E_NOINTERFACE; } else { TRACE("interface %s not implemented\n", debugstr_guid(riid)); *ppvObject = NULL; return E_NOINTERFACE; } IUnknown_AddRef( (IUnknown*)*ppvObject ); return S_OK; } static ULONG WINAPI domelem_AddRef( IXMLDOMElement *iface ) { domelem *This = impl_from_IXMLDOMElement( iface ); LONG ref = InterlockedIncrement(&This->ref); TRACE("(%p)->(%d)\n", This, ref); return ref; } static ULONG WINAPI domelem_Release( IXMLDOMElement *iface ) { domelem *This = impl_from_IXMLDOMElement( iface ); ULONG ref = InterlockedDecrement(&This->ref); TRACE("(%p)->(%d)\n", This, ref); if(!ref) { destroy_xmlnode(&This->node); heap_free(This); } return ref; } static HRESULT WINAPI domelem_GetTypeInfoCount( IXMLDOMElement *iface, UINT* pctinfo ) { domelem *This = impl_from_IXMLDOMElement( iface ); TRACE("(%p)->(%p)\n", This, pctinfo); *pctinfo = 1; return S_OK; } static HRESULT WINAPI domelem_GetTypeInfo( IXMLDOMElement *iface, UINT iTInfo, LCID lcid, ITypeInfo** ppTInfo ) { domelem *This = impl_from_IXMLDOMElement( iface ); HRESULT hr; TRACE("(%p)->(%u %u %p)\n", This, iTInfo, lcid, ppTInfo); hr = get_typeinfo(IXMLDOMElement_tid, ppTInfo); return hr; } static HRESULT WINAPI domelem_GetIDsOfNames( IXMLDOMElement *iface, REFIID riid, LPOLESTR* rgszNames, UINT cNames, LCID lcid, DISPID* rgDispId ) { domelem *This = impl_from_IXMLDOMElement( iface ); ITypeInfo *typeinfo; HRESULT hr; TRACE("(%p)->(%s %p %u %u %p)\n", This, debugstr_guid(riid), rgszNames, cNames, lcid, rgDispId); if(!rgszNames || cNames == 0 || !rgDispId) return E_INVALIDARG; hr = get_typeinfo(IXMLDOMElement_tid, &typeinfo); if(SUCCEEDED(hr)) { hr = ITypeInfo_GetIDsOfNames(typeinfo, rgszNames, cNames, rgDispId); ITypeInfo_Release(typeinfo); } return hr; } static HRESULT WINAPI domelem_Invoke( IXMLDOMElement *iface, DISPID dispIdMember, REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS* pDispParams, VARIANT* pVarResult, EXCEPINFO* pExcepInfo, UINT* puArgErr ) { domelem *This = impl_from_IXMLDOMElement( iface ); ITypeInfo *typeinfo; HRESULT hr; TRACE("(%p)->(%d %s %d %d %p %p %p %p)\n", This, dispIdMember, debugstr_guid(riid), lcid, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr); hr = get_typeinfo(IXMLDOMElement_tid, &typeinfo); if(SUCCEEDED(hr)) { hr = ITypeInfo_Invoke(typeinfo, &This->IXMLDOMElement_iface, dispIdMember, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr); ITypeInfo_Release(typeinfo); } return hr; } static HRESULT WINAPI domelem_get_nodeName( IXMLDOMElement *iface, BSTR* p ) { domelem *This = impl_from_IXMLDOMElement( iface ); TRACE("(%p)->(%p)\n", This, p); return node_get_nodeName(&This->node, p); } static HRESULT WINAPI domelem_get_nodeValue( IXMLDOMElement *iface, VARIANT* value) { domelem *This = impl_from_IXMLDOMElement( iface ); TRACE("(%p)->(%p)\n", This, value); if(!value) return E_INVALIDARG; V_VT(value) = VT_NULL; V_BSTR(value) = NULL; /* tests show that we should do this */ return S_FALSE; } static HRESULT WINAPI domelem_put_nodeValue( IXMLDOMElement *iface, VARIANT value) { domelem *This = impl_from_IXMLDOMElement( iface ); TRACE("(%p)->(%s)\n", This, debugstr_variant(&value)); return E_FAIL; } static HRESULT WINAPI domelem_get_nodeType( IXMLDOMElement *iface, DOMNodeType* domNodeType ) { domelem *This = impl_from_IXMLDOMElement( iface ); TRACE("(%p)->(%p)\n", This, domNodeType); *domNodeType = NODE_ELEMENT; return S_OK; } static HRESULT WINAPI domelem_get_parentNode( IXMLDOMElement *iface, IXMLDOMNode** parent ) { domelem *This = impl_from_IXMLDOMElement( iface ); TRACE("(%p)->(%p)\n", This, parent); return node_get_parent(&This->node, parent); } static HRESULT WINAPI domelem_get_childNodes( IXMLDOMElement *iface, IXMLDOMNodeList** outList) { domelem *This = impl_from_IXMLDOMElement( iface ); TRACE("(%p)->(%p)\n", This, outList); return node_get_child_nodes(&This->node, outList); } static HRESULT WINAPI domelem_get_firstChild( IXMLDOMElement *iface, IXMLDOMNode** domNode) { domelem *This = impl_from_IXMLDOMElement( iface ); TRACE("(%p)->(%p)\n", This, domNode); return node_get_first_child(&This->node, domNode); } static HRESULT WINAPI domelem_get_lastChild( IXMLDOMElement *iface, IXMLDOMNode** domNode) { domelem *This = impl_from_IXMLDOMElement( iface ); TRACE("(%p)->(%p)\n", This, domNode); return node_get_last_child(&This->node, domNode); } static HRESULT WINAPI domelem_get_previousSibling( IXMLDOMElement *iface, IXMLDOMNode** domNode) { domelem *This = impl_from_IXMLDOMElement( iface ); TRACE("(%p)->(%p)\n", This, domNode); return node_get_previous_sibling(&This->node, domNode); } static HRESULT WINAPI domelem_get_nextSibling( IXMLDOMElement *iface, IXMLDOMNode** domNode) { domelem *This = impl_from_IXMLDOMElement( iface ); TRACE("(%p)->(%p)\n", This, domNode); return node_get_next_sibling(&This->node, domNode); } static HRESULT WINAPI domelem_get_attributes( IXMLDOMElement *iface, IXMLDOMNamedNodeMap** map) { domelem *This = impl_from_IXMLDOMElement( iface ); TRACE("(%p)->(%p)\n", This, map); *map = create_nodemap(This->node.node); return S_OK; } static HRESULT WINAPI domelem_insertBefore( IXMLDOMElement *iface, IXMLDOMNode* newNode, VARIANT refChild, IXMLDOMNode** outOldNode) { domelem *This = impl_from_IXMLDOMElement( iface ); TRACE("(%p)->(%p %s %p)\n", This, newNode, debugstr_variant(&refChild), outOldNode); return node_insert_before(&This->node, newNode, &refChild, outOldNode); } static HRESULT WINAPI domelem_replaceChild( IXMLDOMElement *iface, IXMLDOMNode* newNode, IXMLDOMNode* oldNode, IXMLDOMNode** outOldNode) { domelem *This = impl_from_IXMLDOMElement( iface ); TRACE("(%p)->(%p %p %p)\n", This, newNode, oldNode, outOldNode); return node_replace_child(&This->node, newNode, oldNode, outOldNode); } static HRESULT WINAPI domelem_removeChild( IXMLDOMElement *iface, IXMLDOMNode *child, IXMLDOMNode **oldChild) { domelem *This = impl_from_IXMLDOMElement( iface ); TRACE("(%p)->(%p %p)\n", This, child, oldChild); return node_remove_child(&This->node, child, oldChild); } static HRESULT WINAPI domelem_appendChild( IXMLDOMElement *iface, IXMLDOMNode *child, IXMLDOMNode **outChild) { domelem *This = impl_from_IXMLDOMElement( iface ); TRACE("(%p)->(%p %p)\n", This, child, outChild); return node_append_child(&This->node, child, outChild); } static HRESULT WINAPI domelem_hasChildNodes( IXMLDOMElement *iface, VARIANT_BOOL *ret) { domelem *This = impl_from_IXMLDOMElement( iface ); TRACE("(%p)->(%p)\n", This, ret); return node_has_childnodes(&This->node, ret); } static HRESULT WINAPI domelem_get_ownerDocument( IXMLDOMElement *iface, IXMLDOMDocument **doc) { domelem *This = impl_from_IXMLDOMElement( iface ); TRACE("(%p)->(%p)\n", This, doc); return node_get_owner_doc(&This->node, doc); } static HRESULT WINAPI domelem_cloneNode( IXMLDOMElement *iface, VARIANT_BOOL deep, IXMLDOMNode** outNode) { domelem *This = impl_from_IXMLDOMElement( iface ); TRACE("(%p)->(%d %p)\n", This, deep, outNode); return node_clone( &This->node, deep, outNode ); } static HRESULT WINAPI domelem_get_nodeTypeString( IXMLDOMElement *iface, BSTR* p) { domelem *This = impl_from_IXMLDOMElement( iface ); static const WCHAR elementW[] = {'e','l','e','m','e','n','t',0}; TRACE("(%p)->(%p)\n", This, p); return return_bstr(elementW, p); } static HRESULT WINAPI domelem_get_text( IXMLDOMElement *iface, BSTR* p) { domelem *This = impl_from_IXMLDOMElement( iface ); TRACE("(%p)->(%p)\n", This, p); return node_get_text(&This->node, p); } static HRESULT WINAPI domelem_put_text( IXMLDOMElement *iface, BSTR p) { domelem *This = impl_from_IXMLDOMElement( iface ); TRACE("(%p)->(%s)\n", This, debugstr_w(p)); return node_put_text( &This->node, p ); } static HRESULT WINAPI domelem_get_specified( IXMLDOMElement *iface, VARIANT_BOOL* isSpecified) { domelem *This = impl_from_IXMLDOMElement( iface ); FIXME("(%p)->(%p) stub!\n", This, isSpecified); *isSpecified = VARIANT_TRUE; return S_OK; } static HRESULT WINAPI domelem_get_definition( IXMLDOMElement *iface, IXMLDOMNode** definitionNode) { domelem *This = impl_from_IXMLDOMElement( iface ); FIXME("(%p)->(%p)\n", This, definitionNode); return E_NOTIMPL; } static inline BYTE hex_to_byte(xmlChar c) { if(c <= '9') return c-'0'; if(c <= 'F') return c-'A'+10; return c-'a'+10; } static inline BYTE base64_to_byte(xmlChar c) { if(c == '+') return 62; if(c == '/') return 63; if(c <= '9') return c-'0'+52; if(c <= 'Z') return c-'A'; return c-'a'+26; } static inline HRESULT variant_from_dt(XDR_DT dt, xmlChar* str, VARIANT* v) { VARIANT src; HRESULT hr = S_OK; BOOL handled = FALSE; VariantInit(&src); switch (dt) { case DT_INVALID: case DT_STRING: case DT_NMTOKEN: case DT_NMTOKENS: case DT_NUMBER: case DT_URI: case DT_UUID: { V_VT(v) = VT_BSTR; V_BSTR(v) = bstr_from_xmlChar(str); if(!V_BSTR(v)) return E_OUTOFMEMORY; handled = TRUE; } break; case DT_DATE: case DT_DATE_TZ: case DT_DATETIME: case DT_DATETIME_TZ: case DT_TIME: case DT_TIME_TZ: { WCHAR *p, *e; SYSTEMTIME st; DOUBLE date = 0.0; st.wYear = 1899; st.wMonth = 12; st.wDay = 30; st.wDayOfWeek = st.wHour = st.wMinute = st.wSecond = st.wMilliseconds = 0; V_VT(&src) = VT_BSTR; V_BSTR(&src) = bstr_from_xmlChar(str); if(!V_BSTR(&src)) return E_OUTOFMEMORY; p = V_BSTR(&src); e = p + SysStringLen(V_BSTR(&src)); if(p+4pvData)[i] = (hex_to_byte(str[2*i])<<4) + hex_to_byte(str[2*i+1]); handled = TRUE; } break; case DT_BIN_BASE64: { SAFEARRAYBOUND sab; xmlChar *c1, *c2; int i, len; /* remove all formatting chars */ c1 = c2 = str; len = 0; while (*c2) { if ( *c2 == ' ' || *c2 == '\t' || *c2 == '\n' || *c2 == '\r' ) { c2++; continue; } *c1++ = *c2++; len++; } /* skip padding */ if(str[len-2] == '=') i = 2; else if(str[len-1] == '=') i = 1; else i = 0; sab.lLbound = 0; sab.cElements = len/4*3-i; V_VT(v) = (VT_ARRAY|VT_UI1); V_ARRAY(v) = SafeArrayCreate(VT_UI1, 1, &sab); if(!V_ARRAY(v)) return E_OUTOFMEMORY; for(i=0; ipvData)[3*i] = (base64_to_byte(str[4*i])<<2) + (base64_to_byte(str[4*i+1])>>4); if(3*i+1 < sab.cElements) ((BYTE*)V_ARRAY(v)->pvData)[3*i+1] = (base64_to_byte(str[4*i+1])<<4) + (base64_to_byte(str[4*i+2])>>2); if(3*i+2 < sab.cElements) ((BYTE*)V_ARRAY(v)->pvData)[3*i+2] = (base64_to_byte(str[4*i+2])<<6) + base64_to_byte(str[4*i+3]); } handled = TRUE; } break; case DT_BOOLEAN: V_VT(v) = VT_BOOL; break; case DT_FIXED_14_4: V_VT(v) = VT_CY; break; case DT_I1: V_VT(v) = VT_I1; break; case DT_I2: V_VT(v) = VT_I2; break; case DT_I4: case DT_INT: V_VT(v) = VT_I4; break; case DT_I8: V_VT(v) = VT_I8; break; case DT_R4: V_VT(v) = VT_R4; break; case DT_FLOAT: case DT_R8: V_VT(v) = VT_R8; break; case DT_UI1: V_VT(v) = VT_UI1; break; case DT_UI2: V_VT(v) = VT_UI2; break; case DT_UI4: V_VT(v) = VT_UI4; break; case DT_UI8: V_VT(v) = VT_UI8; break; case DT_CHAR: case DT_ENTITY: case DT_ENTITIES: case DT_ENUMERATION: case DT_ID: case DT_IDREF: case DT_IDREFS: case DT_NOTATION: FIXME("need to handle dt:%s\n", debugstr_dt(dt)); V_VT(v) = VT_BSTR; V_BSTR(v) = bstr_from_xmlChar(str); if (!V_BSTR(v)) return E_OUTOFMEMORY; handled = TRUE; break; } if (!handled) { V_VT(&src) = VT_BSTR; V_BSTR(&src) = bstr_from_xmlChar(str); if(!V_BSTR(&src)) return E_OUTOFMEMORY; hr = VariantChangeTypeEx(v, &src, MAKELCID(MAKELANGID(LANG_ENGLISH,SUBLANG_ENGLISH_US),SORT_DEFAULT),0, V_VT(v)); VariantClear(&src); } return hr; } static XDR_DT element_get_dt(xmlNodePtr node) { XDR_DT dt = DT_INVALID; TRACE("(%p)\n", node); if(node->type != XML_ELEMENT_NODE) { FIXME("invalid element node\n"); return dt; } if (node->ns && xmlStrEqual(node->ns->href, DT_nsURI)) { dt = str_to_dt(node->name, -1); } else { xmlChar* pVal = xmlGetNsProp(node, BAD_CAST "dt", DT_nsURI); if (pVal) { dt = str_to_dt(pVal, -1); xmlFree(pVal); } else if (node->doc) { IXMLDOMDocument3* doc = (IXMLDOMDocument3*)create_domdoc((xmlNodePtr)node->doc); if (doc) { VARIANT v; VariantInit(&v); if (IXMLDOMDocument3_get_schemas(doc, &v) == S_OK && V_VT(&v) == VT_DISPATCH) { dt = SchemaCache_get_node_dt((IXMLDOMSchemaCollection2*)V_DISPATCH(&v), node); } VariantClear(&v); IXMLDOMDocument3_Release(doc); } } } TRACE("=> dt:%s\n", debugstr_dt(dt)); return dt; } static HRESULT WINAPI domelem_get_nodeTypedValue( IXMLDOMElement *iface, VARIANT* v) { domelem *This = impl_from_IXMLDOMElement( iface ); XDR_DT dt; xmlChar* content; HRESULT hr; TRACE("(%p)->(%p)\n", This, v); if(!v) return E_INVALIDARG; V_VT(v) = VT_NULL; dt = element_get_dt(get_element(This)); content = xmlNodeGetContent(get_element(This)); hr = variant_from_dt(dt, content, v); xmlFree(content); return hr; } static HRESULT WINAPI domelem_put_nodeTypedValue( IXMLDOMElement *iface, VARIANT value) { domelem *This = impl_from_IXMLDOMElement( iface ); XDR_DT dt; HRESULT hr; TRACE("(%p)->(%s)\n", This, debugstr_variant(&value)); dt = element_get_dt(get_element(This)); /* for untyped node coerce to BSTR and set */ if (dt == DT_INVALID) { if (V_VT(&value) != VT_BSTR) { VARIANT content; VariantInit(&content); hr = VariantChangeType(&content, &value, 0, VT_BSTR); if (hr == S_OK) { hr = node_set_content(&This->node, V_BSTR(&content)); VariantClear(&content); } } else hr = node_set_content(&This->node, V_BSTR(&value)); } else { FIXME("not implemented for dt:%s\n", debugstr_dt(dt)); return E_NOTIMPL; } return hr; } static HRESULT WINAPI domelem_get_dataType( IXMLDOMElement *iface, VARIANT* typename) { domelem *This = impl_from_IXMLDOMElement( iface ); XDR_DT dt; TRACE("(%p)->(%p)\n", This, typename); if (!typename) return E_INVALIDARG; dt = element_get_dt(get_element(This)); switch (dt) { case DT_BIN_BASE64: case DT_BIN_HEX: case DT_BOOLEAN: case DT_CHAR: case DT_DATE: case DT_DATE_TZ: case DT_DATETIME: case DT_DATETIME_TZ: case DT_FIXED_14_4: case DT_FLOAT: case DT_I1: case DT_I2: case DT_I4: case DT_I8: case DT_INT: case DT_NUMBER: case DT_R4: case DT_R8: case DT_TIME: case DT_TIME_TZ: case DT_UI1: case DT_UI2: case DT_UI4: case DT_UI8: case DT_URI: case DT_UUID: V_VT(typename) = VT_BSTR; V_BSTR(typename) = SysAllocString(dt_to_bstr(dt)); if (!V_BSTR(typename)) return E_OUTOFMEMORY; break; default: /* Other types (DTD equivalents) do not return anything here, * but the pointer part of the VARIANT is set to NULL */ V_VT(typename) = VT_NULL; V_BSTR(typename) = NULL; break; } return (V_VT(typename) != VT_NULL) ? S_OK : S_FALSE; } static HRESULT WINAPI domelem_put_dataType( IXMLDOMElement *iface, BSTR dtName) { domelem *This = impl_from_IXMLDOMElement( iface ); HRESULT hr = E_FAIL; xmlChar *str; XDR_DT dt; TRACE("(%p)->(%s)\n", This, debugstr_w(dtName)); if(dtName == NULL) return E_INVALIDARG; dt = bstr_to_dt(dtName, -1); /* An example of this is. The Text in the node needs to be a 0 or 1 for a boolean type. This applies to changing types (string->bool) or setting a new one */ str = xmlNodeGetContent(get_element(This)); hr = dt_validate(dt, str); xmlFree(str); /* Check all supported types. */ if (hr == S_OK) { switch (dt) { case DT_BIN_BASE64: case DT_BIN_HEX: case DT_BOOLEAN: case DT_CHAR: case DT_DATE: case DT_DATE_TZ: case DT_DATETIME: case DT_DATETIME_TZ: case DT_FIXED_14_4: case DT_FLOAT: case DT_I1: case DT_I2: case DT_I4: case DT_I8: case DT_INT: case DT_NMTOKEN: case DT_NMTOKENS: case DT_NUMBER: case DT_R4: case DT_R8: case DT_STRING: case DT_TIME: case DT_TIME_TZ: case DT_UI1: case DT_UI2: case DT_UI4: case DT_UI8: case DT_URI: case DT_UUID: { xmlAttrPtr attr = xmlHasNsProp(get_element(This), DT_prefix, DT_nsURI); if (attr) { attr = xmlSetNsProp(get_element(This), attr->ns, DT_prefix, dt_to_str(dt)); hr = S_OK; } else { xmlNsPtr ns = xmlNewNs(get_element(This), DT_nsURI, DT_prefix); if (ns) { attr = xmlNewNsProp(get_element(This), ns, DT_prefix, dt_to_str(dt)); if (attr) { xmlAddChild(get_element(This), (xmlNodePtr)attr); hr = S_OK; } else ERR("Failed to create Attribute\n"); } else ERR("Failed to create Namespace\n"); } } break; default: FIXME("need to handle dt:%s\n", debugstr_dt(dt)); break; } } return hr; } static HRESULT WINAPI domelem_get_xml( IXMLDOMElement *iface, BSTR* p) { domelem *This = impl_from_IXMLDOMElement( iface ); TRACE("(%p)->(%p)\n", This, p); return node_get_xml(&This->node, TRUE, FALSE, p); } static HRESULT WINAPI domelem_transformNode( IXMLDOMElement *iface, IXMLDOMNode *node, BSTR *p) { domelem *This = impl_from_IXMLDOMElement( iface ); TRACE("(%p)->(%p %p)\n", This, node, p); return node_transform_node(&This->node, node, p); } static HRESULT WINAPI domelem_selectNodes( IXMLDOMElement *iface, BSTR p, IXMLDOMNodeList** outList) { domelem *This = impl_from_IXMLDOMElement( iface ); TRACE("(%p)->(%s %p)\n", This, debugstr_w(p), outList); return node_select_nodes(&This->node, p, outList); } static HRESULT WINAPI domelem_selectSingleNode( IXMLDOMElement *iface, BSTR p, IXMLDOMNode** outNode) { domelem *This = impl_from_IXMLDOMElement( iface ); TRACE("(%p)->(%s %p)\n", This, debugstr_w(p), outNode); return node_select_singlenode(&This->node, p, outNode); } static HRESULT WINAPI domelem_get_parsed( IXMLDOMElement *iface, VARIANT_BOOL* isParsed) { domelem *This = impl_from_IXMLDOMElement( iface ); FIXME("(%p)->(%p) stub!\n", This, isParsed); *isParsed = VARIANT_TRUE; return S_OK; } static HRESULT WINAPI domelem_get_namespaceURI( IXMLDOMElement *iface, BSTR* p) { domelem *This = impl_from_IXMLDOMElement( iface ); TRACE("(%p)->(%p)\n", This, p); return node_get_namespaceURI(&This->node, p); } static HRESULT WINAPI domelem_get_prefix( IXMLDOMElement *iface, BSTR* prefix) { domelem *This = impl_from_IXMLDOMElement( iface ); TRACE("(%p)->(%p)\n", This, prefix); return node_get_prefix( &This->node, prefix ); } static HRESULT WINAPI domelem_get_baseName( IXMLDOMElement *iface, BSTR* name) { domelem *This = impl_from_IXMLDOMElement( iface ); TRACE("(%p)->(%p)\n", This, name); return node_get_base_name( &This->node, name ); } static HRESULT WINAPI domelem_transformNodeToObject( IXMLDOMElement *iface, IXMLDOMNode* domNode, VARIANT var1) { domelem *This = impl_from_IXMLDOMElement( iface ); FIXME("(%p)->(%p %s)\n", This, domNode, debugstr_variant(&var1)); return E_NOTIMPL; } static HRESULT WINAPI domelem_get_tagName( IXMLDOMElement *iface, BSTR* p) { domelem *This = impl_from_IXMLDOMElement( iface ); xmlNodePtr element; const xmlChar *prefix; xmlChar *qname; TRACE("(%p)->(%p)\n", This, p ); if (!p) return E_INVALIDARG; element = get_element( This ); if ( !element ) return E_FAIL; prefix = element->ns ? element->ns->prefix : NULL; qname = xmlBuildQName(element->name, prefix, NULL, 0); *p = bstr_from_xmlChar(qname); if (qname != element->name) xmlFree(qname); return *p ? S_OK : E_OUTOFMEMORY; } static HRESULT WINAPI domelem_getAttribute( IXMLDOMElement *iface, BSTR name, VARIANT* value) { domelem *This = impl_from_IXMLDOMElement( iface ); xmlNodePtr element; xmlChar *xml_name, *xml_value = NULL; HRESULT hr = S_FALSE; TRACE("(%p)->(%s %p)\n", This, debugstr_w(name), value); if(!value || !name) return E_INVALIDARG; element = get_element( This ); if ( !element ) return E_FAIL; V_BSTR(value) = NULL; V_VT(value) = VT_NULL; xml_name = xmlchar_from_wchar( name ); if(!xmlValidateNameValue(xml_name)) hr = E_FAIL; else xml_value = xmlGetNsProp(element, xml_name, NULL); heap_free(xml_name); if(xml_value) { V_VT(value) = VT_BSTR; V_BSTR(value) = bstr_from_xmlChar( xml_value ); xmlFree(xml_value); hr = S_OK; } return hr; } static HRESULT WINAPI domelem_setAttribute( IXMLDOMElement *iface, BSTR name, VARIANT value) { domelem *This = impl_from_IXMLDOMElement( iface ); xmlChar *xml_name, *xml_value, *local, *prefix; xmlNodePtr element; HRESULT hr; VARIANT var; TRACE("(%p)->(%s %s)\n", This, debugstr_w(name), debugstr_variant(&value)); element = get_element( This ); if ( !element ) return E_FAIL; VariantInit(&var); hr = VariantChangeType(&var, &value, 0, VT_BSTR); if(hr != S_OK) { FIXME("VariantChangeType failed\n"); return hr; } xml_name = xmlchar_from_wchar( name ); xml_value = xmlchar_from_wchar( V_BSTR(&var) ); if ((local = xmlSplitQName2(xml_name, &prefix))) { static const xmlChar* xmlnsA = (const xmlChar*)"xmlns"; xmlNsPtr ns = NULL; /* it's not allowed to modify existing namespace definition */ if (xmlStrEqual(prefix, xmlnsA)) ns = xmlSearchNs(element->doc, element, local); xmlFree(prefix); xmlFree(local); if (ns) return xmlStrEqual(ns->href, xml_value) ? S_OK : E_INVALIDARG; } if (!xmlSetNsProp(element, NULL, xml_name, xml_value)) hr = E_FAIL; heap_free(xml_value); heap_free(xml_name); VariantClear(&var); return hr; } static HRESULT WINAPI domelem_removeAttribute( IXMLDOMElement *iface, BSTR p) { domelem *This = impl_from_IXMLDOMElement( iface ); IXMLDOMNamedNodeMap *attr; HRESULT hr; TRACE("(%p)->(%s)\n", This, debugstr_w(p)); hr = IXMLDOMElement_get_attributes(iface, &attr); if (hr != S_OK) return hr; hr = IXMLDOMNamedNodeMap_removeNamedItem(attr, p, NULL); IXMLDOMNamedNodeMap_Release(attr); return hr; } static HRESULT WINAPI domelem_getAttributeNode( IXMLDOMElement *iface, BSTR p, IXMLDOMAttribute** attributeNode ) { domelem *This = impl_from_IXMLDOMElement( iface ); xmlChar *xml_name; xmlNodePtr element; xmlAttrPtr attr; IUnknown *unk; HRESULT hr = S_FALSE; TRACE("(%p)->(%s %p)\n", This, debugstr_w(p), attributeNode); if(!attributeNode) return E_FAIL; *attributeNode = NULL; element = get_element( This ); if ( !element ) return E_FAIL; xml_name = xmlchar_from_wchar(p); if(!xmlValidateNameValue(xml_name)) { heap_free(xml_name); return E_FAIL; } attr = xmlHasProp(element, xml_name); if(attr) { unk = create_attribute((xmlNodePtr)attr); hr = IUnknown_QueryInterface(unk, &IID_IXMLDOMAttribute, (void**)attributeNode); IUnknown_Release(unk); } heap_free(xml_name); return hr; } static HRESULT WINAPI domelem_setAttributeNode( IXMLDOMElement *iface, IXMLDOMAttribute* attribute, IXMLDOMAttribute** old) { domelem *This = impl_from_IXMLDOMElement( iface ); static const WCHAR xmlnsW[] = {'x','m','l','n','s',0}; xmlChar *name, *value; BSTR nameW, prefix; xmlnode *attr_node; xmlAttrPtr attr; VARIANT valueW; HRESULT hr; FIXME("(%p)->(%p %p): semi-stub\n", This, attribute, old); if (!attribute) return E_INVALIDARG; attr_node = get_node_obj((IXMLDOMNode*)attribute); if (!attr_node) return E_FAIL; if (attr_node->parent) { WARN("attempt to add already used attribute\n"); return E_FAIL; } hr = IXMLDOMAttribute_get_nodeName(attribute, &nameW); if (hr != S_OK) return hr; /* adding xmlns attribute doesn't change a tree or existing namespace definition */ if (!strcmpW(nameW, xmlnsW)) { SysFreeString(nameW); return DISP_E_UNKNOWNNAME; } hr = IXMLDOMAttribute_get_nodeValue(attribute, &valueW); if (hr != S_OK) { SysFreeString(nameW); return hr; } if (old) *old = NULL; TRACE("attribute: %s=%s\n", debugstr_w(nameW), debugstr_w(V_BSTR(&valueW))); hr = IXMLDOMAttribute_get_prefix(attribute, &prefix); if (hr == S_OK) { FIXME("namespaces not supported: %s\n", debugstr_w(prefix)); SysFreeString(prefix); } name = xmlchar_from_wchar(nameW); value = xmlchar_from_wchar(V_BSTR(&valueW)); if (!name || !value) { SysFreeString(nameW); VariantClear(&valueW); heap_free(name); heap_free(value); return E_OUTOFMEMORY; } attr = xmlSetNsProp(get_element(This), NULL, name, value); if (attr) attr_node->parent = (IXMLDOMNode*)iface; SysFreeString(nameW); VariantClear(&valueW); heap_free(name); heap_free(value); return attr ? S_OK : E_FAIL; } static HRESULT WINAPI domelem_removeAttributeNode( IXMLDOMElement *iface, IXMLDOMAttribute* domAttribute, IXMLDOMAttribute** attributeNode) { domelem *This = impl_from_IXMLDOMElement( iface ); FIXME("(%p)->(%p %p)\n", This, domAttribute, attributeNode); return E_NOTIMPL; } static HRESULT WINAPI domelem_getElementsByTagName( IXMLDOMElement *iface, BSTR tagName, IXMLDOMNodeList** resultList) { domelem *This = impl_from_IXMLDOMElement( iface ); xmlChar *query; HRESULT hr; BOOL XPath; TRACE("(%p)->(%s, %p)\n", This, debugstr_w(tagName), resultList); if (!tagName || !resultList) return E_INVALIDARG; XPath = is_xpathmode(get_element(This)->doc); set_xpathmode(get_element(This)->doc, TRUE); query = tagName_to_XPath(tagName); hr = create_selection(get_element(This), query, resultList); xmlFree(query); set_xpathmode(get_element(This)->doc, XPath); return hr; } static HRESULT WINAPI domelem_normalize( IXMLDOMElement *iface ) { domelem *This = impl_from_IXMLDOMElement( iface ); FIXME("%p\n", This); return E_NOTIMPL; } static const struct IXMLDOMElementVtbl domelem_vtbl = { domelem_QueryInterface, domelem_AddRef, domelem_Release, domelem_GetTypeInfoCount, domelem_GetTypeInfo, domelem_GetIDsOfNames, domelem_Invoke, domelem_get_nodeName, domelem_get_nodeValue, domelem_put_nodeValue, domelem_get_nodeType, domelem_get_parentNode, domelem_get_childNodes, domelem_get_firstChild, domelem_get_lastChild, domelem_get_previousSibling, domelem_get_nextSibling, domelem_get_attributes, domelem_insertBefore, domelem_replaceChild, domelem_removeChild, domelem_appendChild, domelem_hasChildNodes, domelem_get_ownerDocument, domelem_cloneNode, domelem_get_nodeTypeString, domelem_get_text, domelem_put_text, domelem_get_specified, domelem_get_definition, domelem_get_nodeTypedValue, domelem_put_nodeTypedValue, domelem_get_dataType, domelem_put_dataType, domelem_get_xml, domelem_transformNode, domelem_selectNodes, domelem_selectSingleNode, domelem_get_parsed, domelem_get_namespaceURI, domelem_get_prefix, domelem_get_baseName, domelem_transformNodeToObject, domelem_get_tagName, domelem_getAttribute, domelem_setAttribute, domelem_removeAttribute, domelem_getAttributeNode, domelem_setAttributeNode, domelem_removeAttributeNode, domelem_getElementsByTagName, domelem_normalize, }; static const tid_t domelem_iface_tids[] = { IXMLDOMElement_tid, 0 }; static dispex_static_data_t domelem_dispex = { NULL, IXMLDOMElement_tid, NULL, domelem_iface_tids }; IUnknown* create_element( xmlNodePtr element ) { domelem *This; This = heap_alloc( sizeof *This ); if ( !This ) return NULL; This->IXMLDOMElement_iface.lpVtbl = &domelem_vtbl; This->ref = 1; init_xmlnode(&This->node, element, (IXMLDOMNode*)&This->IXMLDOMElement_iface, &domelem_dispex); return (IUnknown*)&This->IXMLDOMElement_iface; } #endif