2 * Node map implementation
4 * Copyright 2005 Mike McCormack
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
27 # include <libxml/parser.h>
28 # include <libxml/xmlerror.h>
37 #include "msxml2did.h"
39 #include "msxml_private.h"
41 #include "wine/debug.h"
43 WINE_DEFAULT_DEBUG_CHANNEL(msxml
);
47 typedef struct _xmlnodemap
50 IXMLDOMNamedNodeMap IXMLDOMNamedNodeMap_iface
;
51 ISupportErrorInfo ISupportErrorInfo_iface
;
58 static inline xmlnodemap
*impl_from_IXMLDOMNamedNodeMap( IXMLDOMNamedNodeMap
*iface
)
60 return CONTAINING_RECORD(iface
, xmlnodemap
, IXMLDOMNamedNodeMap_iface
);
63 static inline xmlnodemap
*impl_from_ISupportErrorInfo( ISupportErrorInfo
*iface
)
65 return CONTAINING_RECORD(iface
, xmlnodemap
, ISupportErrorInfo_iface
);
68 static HRESULT WINAPI
xmlnodemap_QueryInterface(
69 IXMLDOMNamedNodeMap
*iface
,
70 REFIID riid
, void** ppvObject
)
72 xmlnodemap
*This
= impl_from_IXMLDOMNamedNodeMap( iface
);
73 TRACE("(%p)->(%s %p)\n", iface
, debugstr_guid(riid
), ppvObject
);
75 if( IsEqualGUID( riid
, &IID_IUnknown
) ||
76 IsEqualGUID( riid
, &IID_IDispatch
) ||
77 IsEqualGUID( riid
, &IID_IXMLDOMNamedNodeMap
) )
81 else if (dispex_query_interface(&This
->dispex
, riid
, ppvObject
))
83 return *ppvObject
? S_OK
: E_NOINTERFACE
;
85 else if( IsEqualGUID( riid
, &IID_ISupportErrorInfo
))
87 *ppvObject
= &This
->ISupportErrorInfo_iface
;
91 TRACE("interface %s not implemented\n", debugstr_guid(riid
));
96 IXMLDOMElement_AddRef( iface
);
101 static ULONG WINAPI
xmlnodemap_AddRef(
102 IXMLDOMNamedNodeMap
*iface
)
104 xmlnodemap
*This
= impl_from_IXMLDOMNamedNodeMap( iface
);
105 ULONG ref
= InterlockedIncrement( &This
->ref
);
106 TRACE("(%p)->(%d)\n", This
, ref
);
110 static ULONG WINAPI
xmlnodemap_Release(
111 IXMLDOMNamedNodeMap
*iface
)
113 xmlnodemap
*This
= impl_from_IXMLDOMNamedNodeMap( iface
);
114 ULONG ref
= InterlockedDecrement( &This
->ref
);
116 TRACE("(%p)->(%d)\n", This
, ref
);
119 xmldoc_release( This
->node
->doc
);
120 release_dispex(&This
->dispex
);
127 static HRESULT WINAPI
xmlnodemap_GetTypeInfoCount(
128 IXMLDOMNamedNodeMap
*iface
,
131 xmlnodemap
*This
= impl_from_IXMLDOMNamedNodeMap( iface
);
133 TRACE("(%p)->(%p)\n", This
, pctinfo
);
140 static HRESULT WINAPI
xmlnodemap_GetTypeInfo(
141 IXMLDOMNamedNodeMap
*iface
,
142 UINT iTInfo
, LCID lcid
,
143 ITypeInfo
** ppTInfo
)
145 xmlnodemap
*This
= impl_from_IXMLDOMNamedNodeMap( iface
);
146 TRACE("(%p)->(%u %u %p)\n", This
, iTInfo
, lcid
, ppTInfo
);
147 return get_typeinfo(IXMLDOMNamedNodeMap_tid
, ppTInfo
);
150 static HRESULT WINAPI
xmlnodemap_GetIDsOfNames(
151 IXMLDOMNamedNodeMap
*iface
,
152 REFIID riid
, LPOLESTR
* rgszNames
,
153 UINT cNames
, LCID lcid
, DISPID
* rgDispId
)
155 xmlnodemap
*This
= impl_from_IXMLDOMNamedNodeMap( iface
);
159 TRACE("(%p)->(%s %p %u %u %p)\n", This
, debugstr_guid(riid
), rgszNames
, cNames
,
162 if(!rgszNames
|| cNames
== 0 || !rgDispId
)
165 hr
= get_typeinfo(IXMLDOMNamedNodeMap_tid
, &typeinfo
);
168 hr
= ITypeInfo_GetIDsOfNames(typeinfo
, rgszNames
, cNames
, rgDispId
);
169 ITypeInfo_Release(typeinfo
);
175 static HRESULT WINAPI
xmlnodemap_Invoke(
176 IXMLDOMNamedNodeMap
*iface
,
177 DISPID dispIdMember
, REFIID riid
, LCID lcid
,
178 WORD wFlags
, DISPPARAMS
* pDispParams
, VARIANT
* pVarResult
,
179 EXCEPINFO
* pExcepInfo
, UINT
* puArgErr
)
181 xmlnodemap
*This
= impl_from_IXMLDOMNamedNodeMap( iface
);
185 TRACE("(%p)->(%d %s %d %d %p %p %p %p)\n", This
, dispIdMember
, debugstr_guid(riid
),
186 lcid
, wFlags
, pDispParams
, pVarResult
, pExcepInfo
, puArgErr
);
188 hr
= get_typeinfo(IXMLDOMNamedNodeMap_tid
, &typeinfo
);
191 hr
= ITypeInfo_Invoke(typeinfo
, &This
->IXMLDOMNamedNodeMap_iface
, dispIdMember
, wFlags
,
192 pDispParams
, pVarResult
, pExcepInfo
, puArgErr
);
193 ITypeInfo_Release(typeinfo
);
199 static HRESULT WINAPI
xmlnodemap_getNamedItem(
200 IXMLDOMNamedNodeMap
*iface
,
204 xmlnodemap
*This
= impl_from_IXMLDOMNamedNodeMap( iface
);
205 xmlChar
*nameA
, *local
, *prefix
;
210 TRACE("(%p)->(%s %p)\n", This
, debugstr_w(name
), item
);
212 nameA
= xmlchar_from_wchar(name
);
213 local
= xmlSplitQName2(nameA
, &prefix
);
217 return IXMLDOMNamedNodeMap_getQualifiedItem(iface
, name
, NULL
, item
);
219 /* try to get namespace uri for supplied qualified name */
220 ns
= xmlSearchNs(This
->node
->doc
, This
->node
, prefix
);
227 if (item
) *item
= NULL
;
228 return item
? S_FALSE
: E_INVALIDARG
;
231 uriW
= bstr_from_xmlChar(ns
->href
);
232 localW
= bstr_from_xmlChar(local
);
235 TRACE("got qualified node %s, uri=%s\n", debugstr_w(localW
), debugstr_w(uriW
));
237 hr
= IXMLDOMNamedNodeMap_getQualifiedItem(iface
, localW
, uriW
, item
);
239 SysFreeString(localW
);
245 static HRESULT WINAPI
xmlnodemap_setNamedItem(
246 IXMLDOMNamedNodeMap
*iface
,
247 IXMLDOMNode
* newItem
,
248 IXMLDOMNode
** namedItem
)
250 xmlnodemap
*This
= impl_from_IXMLDOMNamedNodeMap( iface
);
254 TRACE("(%p)->(%p %p)\n", This
, newItem
, namedItem
);
259 if(namedItem
) *namedItem
= NULL
;
261 /* Must be an Attribute */
262 ThisNew
= get_node_obj( newItem
);
263 if(!ThisNew
) return E_FAIL
;
265 if(ThisNew
->node
->type
!= XML_ATTRIBUTE_NODE
)
268 if(!ThisNew
->node
->parent
)
269 if(xmldoc_remove_orphan(ThisNew
->node
->doc
, ThisNew
->node
) != S_OK
)
270 WARN("%p is not an orphan of %p\n", ThisNew
->node
, ThisNew
->node
->doc
);
272 nodeNew
= xmlAddChild(This
->node
, ThisNew
->node
);
275 *namedItem
= create_node( nodeNew
);
279 static HRESULT WINAPI
xmlnodemap_removeNamedItem(
280 IXMLDOMNamedNodeMap
*iface
,
282 IXMLDOMNode
** namedItem
)
284 xmlnodemap
*This
= impl_from_IXMLDOMNamedNodeMap( iface
);
285 TRACE("(%p)->(%s %p)\n", This
, debugstr_w(name
), namedItem
);
286 return IXMLDOMNamedNodeMap_removeQualifiedItem(iface
, name
, NULL
, namedItem
);
289 static HRESULT WINAPI
xmlnodemap_get_item(
290 IXMLDOMNamedNodeMap
*iface
,
292 IXMLDOMNode
** listItem
)
294 xmlnodemap
*This
= impl_from_IXMLDOMNamedNodeMap( iface
);
298 TRACE("(%p)->(%d %p)\n", This
, index
, listItem
);
305 curr
= This
->node
->properties
;
307 for (attrIndex
= 0; attrIndex
< index
; attrIndex
++) {
308 if (curr
->next
== NULL
)
314 *listItem
= create_node( (xmlNodePtr
) curr
);
319 static HRESULT WINAPI
xmlnodemap_get_length(
320 IXMLDOMNamedNodeMap
*iface
,
327 xmlnodemap
*This
= impl_from_IXMLDOMNamedNodeMap( iface
);
329 TRACE("(%p)->(%p)\n", This
, listLength
);
334 first
= This
->node
->properties
;
346 *listLength
= attrCount
;
351 static HRESULT WINAPI
xmlnodemap_getQualifiedItem(
352 IXMLDOMNamedNodeMap
*iface
,
355 IXMLDOMNode
** qualifiedItem
)
357 xmlnodemap
*This
= impl_from_IXMLDOMNamedNodeMap( iface
);
362 TRACE("(%p)->(%s %s %p)\n", This
, debugstr_w(baseName
), debugstr_w(namespaceURI
), qualifiedItem
);
364 if (!baseName
|| !qualifiedItem
) return E_INVALIDARG
;
366 if (namespaceURI
&& *namespaceURI
)
368 href
= xmlchar_from_wchar(namespaceURI
);
369 if (!href
) return E_OUTOFMEMORY
;
374 name
= xmlchar_from_wchar(baseName
);
378 return E_OUTOFMEMORY
;
381 attr
= xmlHasNsProp(This
->node
, name
, href
);
388 *qualifiedItem
= NULL
;
392 *qualifiedItem
= create_node((xmlNodePtr
)attr
);
397 static HRESULT WINAPI
xmlnodemap_removeQualifiedItem(
398 IXMLDOMNamedNodeMap
*iface
,
401 IXMLDOMNode
** qualifiedItem
)
403 xmlnodemap
*This
= impl_from_IXMLDOMNamedNodeMap( iface
);
408 TRACE("(%p)->(%s %s %p)\n", This
, debugstr_w(baseName
), debugstr_w(namespaceURI
), qualifiedItem
);
410 if (!baseName
) return E_INVALIDARG
;
412 if (namespaceURI
&& *namespaceURI
)
414 href
= xmlchar_from_wchar(namespaceURI
);
415 if (!href
) return E_OUTOFMEMORY
;
420 name
= xmlchar_from_wchar(baseName
);
424 return E_OUTOFMEMORY
;
427 attr
= xmlHasNsProp( This
->node
, name
, href
);
434 if (qualifiedItem
) *qualifiedItem
= NULL
;
440 xmlUnlinkNode( (xmlNodePtr
) attr
);
441 xmldoc_add_orphan( attr
->doc
, (xmlNodePtr
) attr
);
442 *qualifiedItem
= create_node( (xmlNodePtr
) attr
);
446 if (xmlRemoveProp(attr
) == -1)
447 ERR("xmlRemoveProp failed\n");
453 static HRESULT WINAPI
xmlnodemap_nextNode(
454 IXMLDOMNamedNodeMap
*iface
,
455 IXMLDOMNode
** nextItem
)
457 xmlnodemap
*This
= impl_from_IXMLDOMNamedNodeMap( iface
);
461 TRACE("(%p)->(%p: %d)\n", This
, nextItem
, This
->iterator
);
465 curr
= This
->node
->properties
;
467 for (attrIndex
= 0; attrIndex
< This
->iterator
; attrIndex
++) {
468 if (curr
->next
== NULL
)
476 *nextItem
= create_node( (xmlNodePtr
) curr
);
481 static HRESULT WINAPI
xmlnodemap_reset(
482 IXMLDOMNamedNodeMap
*iface
)
484 xmlnodemap
*This
= impl_from_IXMLDOMNamedNodeMap( iface
);
486 TRACE("(%p: %d)\n", This
, This
->iterator
);
493 static HRESULT WINAPI
xmlnodemap__newEnum(
494 IXMLDOMNamedNodeMap
*iface
,
497 xmlnodemap
*This
= impl_from_IXMLDOMNamedNodeMap( iface
);
498 FIXME("(%p)->(%p)\n", This
, ppUnk
);
502 static const struct IXMLDOMNamedNodeMapVtbl xmlnodemap_vtbl
=
504 xmlnodemap_QueryInterface
,
507 xmlnodemap_GetTypeInfoCount
,
508 xmlnodemap_GetTypeInfo
,
509 xmlnodemap_GetIDsOfNames
,
511 xmlnodemap_getNamedItem
,
512 xmlnodemap_setNamedItem
,
513 xmlnodemap_removeNamedItem
,
515 xmlnodemap_get_length
,
516 xmlnodemap_getQualifiedItem
,
517 xmlnodemap_removeQualifiedItem
,
523 static HRESULT WINAPI
support_error_QueryInterface(
524 ISupportErrorInfo
*iface
,
525 REFIID riid
, void** ppvObject
)
527 xmlnodemap
*This
= impl_from_ISupportErrorInfo( iface
);
528 TRACE("%p %s %p\n", iface
, debugstr_guid(riid
), ppvObject
);
529 return IXMLDOMNamedNodeMap_QueryInterface(&This
->IXMLDOMNamedNodeMap_iface
, riid
, ppvObject
);
532 static ULONG WINAPI
support_error_AddRef(
533 ISupportErrorInfo
*iface
)
535 xmlnodemap
*This
= impl_from_ISupportErrorInfo( iface
);
536 return IXMLDOMNamedNodeMap_AddRef(&This
->IXMLDOMNamedNodeMap_iface
);
539 static ULONG WINAPI
support_error_Release(
540 ISupportErrorInfo
*iface
)
542 xmlnodemap
*This
= impl_from_ISupportErrorInfo( iface
);
543 return IXMLDOMNamedNodeMap_Release(&This
->IXMLDOMNamedNodeMap_iface
);
546 static HRESULT WINAPI
support_error_InterfaceSupportsErrorInfo(
547 ISupportErrorInfo
*iface
,
550 FIXME("(%p)->(%s)\n", iface
, debugstr_guid(riid
));
554 static const struct ISupportErrorInfoVtbl support_error_vtbl
=
556 support_error_QueryInterface
,
557 support_error_AddRef
,
558 support_error_Release
,
559 support_error_InterfaceSupportsErrorInfo
562 static HRESULT
xmlnodemap_get_dispid(IUnknown
*iface
, BSTR name
, DWORD flags
, DISPID
*dispid
)
567 for(ptr
= name
; *ptr
&& isdigitW(*ptr
); ptr
++)
568 idx
= idx
*10 + (*ptr
-'0');
570 return DISP_E_UNKNOWNNAME
;
572 *dispid
= DISPID_DOM_COLLECTION_BASE
+ idx
;
573 TRACE("ret %x\n", *dispid
);
577 static HRESULT
xmlnodemap_invoke(IUnknown
*iface
, DISPID id
, LCID lcid
, WORD flags
, DISPPARAMS
*params
,
578 VARIANT
*res
, EXCEPINFO
*ei
)
580 xmlnodemap
*This
= impl_from_IXMLDOMNamedNodeMap( (IXMLDOMNamedNodeMap
*)iface
);
582 TRACE("(%p)->(%x %x %x %p %p %p)\n", This
, id
, lcid
, flags
, params
, res
, ei
);
584 V_VT(res
) = VT_DISPATCH
;
585 V_DISPATCH(res
) = NULL
;
587 if (id
< DISPID_DOM_COLLECTION_BASE
|| id
> DISPID_DOM_COLLECTION_MAX
)
588 return DISP_E_UNKNOWNNAME
;
592 case INVOKE_PROPERTYGET
:
594 IXMLDOMNode
*disp
= NULL
;
596 IXMLDOMNamedNodeMap_get_item(&This
->IXMLDOMNamedNodeMap_iface
, id
- DISPID_DOM_COLLECTION_BASE
, &disp
);
597 V_DISPATCH(res
) = (IDispatch
*)disp
;
602 FIXME("unimplemented flags %x\n", flags
);
607 TRACE("ret %p\n", V_DISPATCH(res
));
612 static const dispex_static_data_vtbl_t xmlnodemap_dispex_vtbl
= {
613 xmlnodemap_get_dispid
,
617 static const tid_t xmlnodemap_iface_tids
[] = {
618 IXMLDOMNamedNodeMap_tid
,
622 static dispex_static_data_t xmlnodemap_dispex
= {
623 &xmlnodemap_dispex_vtbl
,
624 IXMLDOMNamedNodeMap_tid
,
626 xmlnodemap_iface_tids
629 IXMLDOMNamedNodeMap
*create_nodemap( const xmlNodePtr node
)
633 This
= heap_alloc( sizeof *This
);
637 This
->IXMLDOMNamedNodeMap_iface
.lpVtbl
= &xmlnodemap_vtbl
;
638 This
->ISupportErrorInfo_iface
.lpVtbl
= &support_error_vtbl
;
643 init_dispex(&This
->dispex
, (IUnknown
*)&This
->IXMLDOMNamedNodeMap_iface
, &xmlnodemap_dispex
);
645 xmldoc_add_ref(node
->doc
);
647 return &This
->IXMLDOMNamedNodeMap_iface
;