msxml3: Don't use xmlnode's IXMLDOMNode iface in insertBefore implementations.
[wine] / dlls / msxml3 / attribute.c
1 /*
2  *    DOM Attribute implementation
3  *
4  * Copyright 2006 Huw Davies
5  *
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.
10  *
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.
15  *
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
19  */
20
21 #define COBJMACROS
22
23 #include "config.h"
24
25 #include <stdarg.h>
26 #include "windef.h"
27 #include "winbase.h"
28 #include "winuser.h"
29 #include "ole2.h"
30 #include "msxml6.h"
31
32 #include "msxml_private.h"
33
34 #include "wine/debug.h"
35
36 WINE_DEFAULT_DEBUG_CHANNEL(msxml);
37
38 #ifdef HAVE_LIBXML2
39
40 typedef struct _domattr
41 {
42     xmlnode node;
43     const struct IXMLDOMAttributeVtbl *lpVtbl;
44     LONG ref;
45 } domattr;
46
47 static inline domattr *impl_from_IXMLDOMAttribute( IXMLDOMAttribute *iface )
48 {
49     return (domattr *)((char*)iface - FIELD_OFFSET(domattr, lpVtbl));
50 }
51
52 static HRESULT WINAPI domattr_QueryInterface(
53     IXMLDOMAttribute *iface,
54     REFIID riid,
55     void** ppvObject )
56 {
57     domattr *This = impl_from_IXMLDOMAttribute( iface );
58     TRACE("(%p)->(%s %p)\n", This, debugstr_guid(riid), ppvObject);
59
60     if ( IsEqualGUID( riid, &IID_IXMLDOMAttribute ) ||
61          IsEqualGUID( riid, &IID_IXMLDOMNode ) ||
62          IsEqualGUID( riid, &IID_IDispatch ) ||
63          IsEqualGUID( riid, &IID_IUnknown ) )
64     {
65         *ppvObject = iface;
66     }
67     else if(node_query_interface(&This->node, riid, ppvObject))
68     {
69         return *ppvObject ? S_OK : E_NOINTERFACE;
70     }
71     else
72     {
73         FIXME("Unsupported interface %s\n", debugstr_guid(riid));
74         return E_NOINTERFACE;
75     }
76
77     IXMLDOMText_AddRef((IUnknown*)*ppvObject);
78     return S_OK;
79 }
80
81 static ULONG WINAPI domattr_AddRef(
82     IXMLDOMAttribute *iface )
83 {
84     domattr *This = impl_from_IXMLDOMAttribute( iface );
85     return InterlockedIncrement( &This->ref );
86 }
87
88 static ULONG WINAPI domattr_Release(
89     IXMLDOMAttribute *iface )
90 {
91     domattr *This = impl_from_IXMLDOMAttribute( iface );
92     ULONG ref;
93
94     ref = InterlockedDecrement( &This->ref );
95     if ( ref == 0 )
96     {
97         destroy_xmlnode(&This->node);
98         heap_free( This );
99     }
100
101     return ref;
102 }
103
104 static HRESULT WINAPI domattr_GetTypeInfoCount(
105     IXMLDOMAttribute *iface,
106     UINT* pctinfo )
107 {
108     domattr *This = impl_from_IXMLDOMAttribute( iface );
109
110     TRACE("(%p)->(%p)\n", This, pctinfo);
111
112     *pctinfo = 1;
113
114     return S_OK;
115 }
116
117 static HRESULT WINAPI domattr_GetTypeInfo(
118     IXMLDOMAttribute *iface,
119     UINT iTInfo, LCID lcid,
120     ITypeInfo** ppTInfo )
121 {
122     domattr *This = impl_from_IXMLDOMAttribute( iface );
123     HRESULT hr;
124
125     TRACE("(%p)->(%u %u %p)\n", This, iTInfo, lcid, ppTInfo);
126
127     hr = get_typeinfo(IXMLDOMAttribute_tid, ppTInfo);
128
129     return hr;
130 }
131
132 static HRESULT WINAPI domattr_GetIDsOfNames(
133     IXMLDOMAttribute *iface,
134     REFIID riid, LPOLESTR* rgszNames,
135     UINT cNames, LCID lcid, DISPID* rgDispId )
136 {
137     domattr *This = impl_from_IXMLDOMAttribute( iface );
138     ITypeInfo *typeinfo;
139     HRESULT hr;
140
141     TRACE("(%p)->(%s %p %u %u %p)\n", This, debugstr_guid(riid), rgszNames, cNames,
142           lcid, rgDispId);
143
144     if(!rgszNames || cNames == 0 || !rgDispId)
145         return E_INVALIDARG;
146
147     hr = get_typeinfo(IXMLDOMAttribute_tid, &typeinfo);
148     if(SUCCEEDED(hr))
149     {
150         hr = ITypeInfo_GetIDsOfNames(typeinfo, rgszNames, cNames, rgDispId);
151         ITypeInfo_Release(typeinfo);
152     }
153
154     return hr;
155 }
156
157 static HRESULT WINAPI domattr_Invoke(
158     IXMLDOMAttribute *iface,
159     DISPID dispIdMember, REFIID riid, LCID lcid,
160     WORD wFlags, DISPPARAMS* pDispParams, VARIANT* pVarResult,
161     EXCEPINFO* pExcepInfo, UINT* puArgErr )
162 {
163     domattr *This = impl_from_IXMLDOMAttribute( iface );
164     ITypeInfo *typeinfo;
165     HRESULT hr;
166
167     TRACE("(%p)->(%d %s %d %d %p %p %p %p)\n", This, dispIdMember, debugstr_guid(riid),
168           lcid, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr);
169
170     hr = get_typeinfo(IXMLDOMAttribute_tid, &typeinfo);
171     if(SUCCEEDED(hr))
172     {
173         hr = ITypeInfo_Invoke(typeinfo, &(This->lpVtbl), dispIdMember, wFlags, pDispParams,
174                 pVarResult, pExcepInfo, puArgErr);
175         ITypeInfo_Release(typeinfo);
176     }
177     return hr;
178 }
179
180 static HRESULT WINAPI domattr_get_nodeName(
181     IXMLDOMAttribute *iface,
182     BSTR* p )
183 {
184     domattr *This = impl_from_IXMLDOMAttribute( iface );
185
186     TRACE("(%p)->(%p)\n", This, p);
187
188     return node_get_nodeName(&This->node, p);
189 }
190
191 static HRESULT WINAPI domattr_get_nodeValue(
192     IXMLDOMAttribute *iface,
193     VARIANT* value)
194 {
195     domattr *This = impl_from_IXMLDOMAttribute( iface );
196
197     TRACE("(%p)->(%p)\n", This, value);
198
199     return node_get_content(&This->node, value);
200 }
201
202 static HRESULT WINAPI domattr_put_nodeValue(
203     IXMLDOMAttribute *iface,
204     VARIANT value)
205 {
206     domattr *This = impl_from_IXMLDOMAttribute( iface );
207
208     TRACE("(%p)->(v%d)\n", This, V_VT(&value));
209
210     return node_put_value(&This->node, &value);
211 }
212
213 static HRESULT WINAPI domattr_get_nodeType(
214     IXMLDOMAttribute *iface,
215     DOMNodeType* domNodeType )
216 {
217     domattr *This = impl_from_IXMLDOMAttribute( iface );
218
219     TRACE("(%p)->(%p)\n", This, domNodeType);
220
221     *domNodeType = NODE_ATTRIBUTE;
222     return S_OK;
223 }
224
225 static HRESULT WINAPI domattr_get_parentNode(
226     IXMLDOMAttribute *iface,
227     IXMLDOMNode** parent )
228 {
229     domattr *This = impl_from_IXMLDOMAttribute( iface );
230     TRACE("(%p)->(%p)\n", This, parent);
231     if (!parent) return E_INVALIDARG;
232     *parent = NULL;
233     return S_FALSE;
234 }
235
236 static HRESULT WINAPI domattr_get_childNodes(
237     IXMLDOMAttribute *iface,
238     IXMLDOMNodeList** outList)
239 {
240     domattr *This = impl_from_IXMLDOMAttribute( iface );
241
242     TRACE("(%p)->(%p)\n", This, outList);
243
244     return node_get_child_nodes(&This->node, outList);
245 }
246
247 static HRESULT WINAPI domattr_get_firstChild(
248     IXMLDOMAttribute *iface,
249     IXMLDOMNode** domNode)
250 {
251     domattr *This = impl_from_IXMLDOMAttribute( iface );
252
253     TRACE("(%p)->(%p)\n", This, domNode);
254
255     return node_get_first_child(&This->node, domNode);
256 }
257
258 static HRESULT WINAPI domattr_get_lastChild(
259     IXMLDOMAttribute *iface,
260     IXMLDOMNode** domNode)
261 {
262     domattr *This = impl_from_IXMLDOMAttribute( iface );
263
264     TRACE("(%p)->(%p)\n", This, domNode);
265
266     return node_get_last_child(&This->node, domNode);
267 }
268
269 static HRESULT WINAPI domattr_get_previousSibling(
270     IXMLDOMAttribute *iface,
271     IXMLDOMNode** domNode)
272 {
273     domattr *This = impl_from_IXMLDOMAttribute( iface );
274
275     TRACE("(%p)->(%p)\n", This, domNode);
276
277     return return_null_node(domNode);
278 }
279
280 static HRESULT WINAPI domattr_get_nextSibling(
281     IXMLDOMAttribute *iface,
282     IXMLDOMNode** domNode)
283 {
284     domattr *This = impl_from_IXMLDOMAttribute( iface );
285
286     TRACE("(%p)->(%p)\n", This, domNode);
287
288     return return_null_node(domNode);
289 }
290
291 static HRESULT WINAPI domattr_get_attributes(
292     IXMLDOMAttribute *iface,
293     IXMLDOMNamedNodeMap** attributeMap)
294 {
295     domattr *This = impl_from_IXMLDOMAttribute( iface );
296
297     TRACE("(%p)->(%p)\n", This, attributeMap);
298
299     return return_null_ptr((void**)attributeMap);
300 }
301
302 static HRESULT WINAPI domattr_insertBefore(
303     IXMLDOMAttribute *iface,
304     IXMLDOMNode* newNode, VARIANT refChild,
305     IXMLDOMNode** outOldNode)
306 {
307     domattr *This = impl_from_IXMLDOMAttribute( iface );
308
309     FIXME("(%p)->(%p x%d %p) needs test\n", This, newNode, V_VT(&refChild), outOldNode);
310
311     return node_insert_before(&This->node, newNode, &refChild, outOldNode);
312 }
313
314 static HRESULT WINAPI domattr_replaceChild(
315     IXMLDOMAttribute *iface,
316     IXMLDOMNode* newNode,
317     IXMLDOMNode* oldNode,
318     IXMLDOMNode** outOldNode)
319 {
320     domattr *This = impl_from_IXMLDOMAttribute( iface );
321     return IXMLDOMNode_replaceChild( IXMLDOMNode_from_impl(&This->node), newNode, oldNode, outOldNode );
322 }
323
324 static HRESULT WINAPI domattr_removeChild(
325     IXMLDOMAttribute *iface,
326     IXMLDOMNode* domNode, IXMLDOMNode** oldNode)
327 {
328     domattr *This = impl_from_IXMLDOMAttribute( iface );
329     return IXMLDOMNode_removeChild( IXMLDOMNode_from_impl(&This->node), domNode, oldNode );
330 }
331
332 static HRESULT WINAPI domattr_appendChild(
333     IXMLDOMAttribute *iface,
334     IXMLDOMNode* newNode, IXMLDOMNode** outNewNode)
335 {
336     domattr *This = impl_from_IXMLDOMAttribute( iface );
337     return IXMLDOMNode_appendChild( IXMLDOMNode_from_impl(&This->node), newNode, outNewNode );
338 }
339
340 static HRESULT WINAPI domattr_hasChildNodes(
341     IXMLDOMAttribute *iface,
342     VARIANT_BOOL* pbool)
343 {
344     domattr *This = impl_from_IXMLDOMAttribute( iface );
345     return IXMLDOMNode_hasChildNodes( IXMLDOMNode_from_impl(&This->node), pbool );
346 }
347
348 static HRESULT WINAPI domattr_get_ownerDocument(
349     IXMLDOMAttribute *iface,
350     IXMLDOMDocument** domDocument)
351 {
352     domattr *This = impl_from_IXMLDOMAttribute( iface );
353     return IXMLDOMNode_get_ownerDocument( IXMLDOMNode_from_impl(&This->node), domDocument );
354 }
355
356 static HRESULT WINAPI domattr_cloneNode(
357     IXMLDOMAttribute *iface,
358     VARIANT_BOOL pbool, IXMLDOMNode** outNode)
359 {
360     domattr *This = impl_from_IXMLDOMAttribute( iface );
361     return IXMLDOMNode_cloneNode( IXMLDOMNode_from_impl(&This->node), pbool, outNode );
362 }
363
364 static HRESULT WINAPI domattr_get_nodeTypeString(
365     IXMLDOMAttribute *iface,
366     BSTR* p)
367 {
368     domattr *This = impl_from_IXMLDOMAttribute( iface );
369     return IXMLDOMNode_get_nodeTypeString( IXMLDOMNode_from_impl(&This->node), p );
370 }
371
372 static HRESULT WINAPI domattr_get_text(
373     IXMLDOMAttribute *iface,
374     BSTR* p)
375 {
376     domattr *This = impl_from_IXMLDOMAttribute( iface );
377     return IXMLDOMNode_get_text( IXMLDOMNode_from_impl(&This->node), p );
378 }
379
380 static HRESULT WINAPI domattr_put_text(
381     IXMLDOMAttribute *iface,
382     BSTR p)
383 {
384     domattr *This = impl_from_IXMLDOMAttribute( iface );
385     return IXMLDOMNode_put_text( IXMLDOMNode_from_impl(&This->node), p );
386 }
387
388 static HRESULT WINAPI domattr_get_specified(
389     IXMLDOMAttribute *iface,
390     VARIANT_BOOL* pbool)
391 {
392     domattr *This = impl_from_IXMLDOMAttribute( iface );
393     return IXMLDOMNode_get_specified( IXMLDOMNode_from_impl(&This->node), pbool );
394 }
395
396 static HRESULT WINAPI domattr_get_definition(
397     IXMLDOMAttribute *iface,
398     IXMLDOMNode** domNode)
399 {
400     domattr *This = impl_from_IXMLDOMAttribute( iface );
401     return IXMLDOMNode_get_definition( IXMLDOMNode_from_impl(&This->node), domNode );
402 }
403
404 static HRESULT WINAPI domattr_get_nodeTypedValue(
405     IXMLDOMAttribute *iface,
406     VARIANT* var1)
407 {
408     domattr *This = impl_from_IXMLDOMAttribute( iface );
409     return IXMLDOMNode_get_nodeTypedValue( IXMLDOMNode_from_impl(&This->node), var1 );
410 }
411
412 static HRESULT WINAPI domattr_put_nodeTypedValue(
413     IXMLDOMAttribute *iface,
414     VARIANT var1)
415 {
416     domattr *This = impl_from_IXMLDOMAttribute( iface );
417     return IXMLDOMNode_put_nodeTypedValue( IXMLDOMNode_from_impl(&This->node), var1 );
418 }
419
420 static HRESULT WINAPI domattr_get_dataType(
421     IXMLDOMAttribute *iface,
422     VARIANT* var1)
423 {
424     domattr *This = impl_from_IXMLDOMAttribute( iface );
425     return IXMLDOMNode_get_dataType( IXMLDOMNode_from_impl(&This->node), var1 );
426 }
427
428 static HRESULT WINAPI domattr_put_dataType(
429     IXMLDOMAttribute *iface,
430     BSTR p)
431 {
432     domattr *This = impl_from_IXMLDOMAttribute( iface );
433     return IXMLDOMNode_put_dataType( IXMLDOMNode_from_impl(&This->node), p );
434 }
435
436 static HRESULT WINAPI domattr_get_xml(
437     IXMLDOMAttribute *iface,
438     BSTR* p)
439 {
440     domattr *This = impl_from_IXMLDOMAttribute( iface );
441     return IXMLDOMNode_get_xml( IXMLDOMNode_from_impl(&This->node), p );
442 }
443
444 static HRESULT WINAPI domattr_transformNode(
445     IXMLDOMAttribute *iface,
446     IXMLDOMNode* domNode, BSTR* p)
447 {
448     domattr *This = impl_from_IXMLDOMAttribute( iface );
449     return IXMLDOMNode_transformNode( IXMLDOMNode_from_impl(&This->node), domNode, p );
450 }
451
452 static HRESULT WINAPI domattr_selectNodes(
453     IXMLDOMAttribute *iface,
454     BSTR p, IXMLDOMNodeList** outList)
455 {
456     domattr *This = impl_from_IXMLDOMAttribute( iface );
457     return IXMLDOMNode_selectNodes( IXMLDOMNode_from_impl(&This->node), p, outList );
458 }
459
460 static HRESULT WINAPI domattr_selectSingleNode(
461     IXMLDOMAttribute *iface,
462     BSTR p, IXMLDOMNode** outNode)
463 {
464     domattr *This = impl_from_IXMLDOMAttribute( iface );
465     return IXMLDOMNode_selectSingleNode( IXMLDOMNode_from_impl(&This->node), p, outNode );
466 }
467
468 static HRESULT WINAPI domattr_get_parsed(
469     IXMLDOMAttribute *iface,
470     VARIANT_BOOL* pbool)
471 {
472     domattr *This = impl_from_IXMLDOMAttribute( iface );
473     return IXMLDOMNode_get_parsed( IXMLDOMNode_from_impl(&This->node), pbool );
474 }
475
476 static HRESULT WINAPI domattr_get_namespaceURI(
477     IXMLDOMAttribute *iface,
478     BSTR* p)
479 {
480     domattr *This = impl_from_IXMLDOMAttribute( iface );
481     return IXMLDOMNode_get_namespaceURI( IXMLDOMNode_from_impl(&This->node), p );
482 }
483
484 static HRESULT WINAPI domattr_get_prefix(
485     IXMLDOMAttribute *iface,
486     BSTR* p)
487 {
488     domattr *This = impl_from_IXMLDOMAttribute( iface );
489     return IXMLDOMNode_get_prefix( IXMLDOMNode_from_impl(&This->node), p );
490 }
491
492 static HRESULT WINAPI domattr_get_baseName(
493     IXMLDOMAttribute *iface,
494     BSTR* p)
495 {
496     domattr *This = impl_from_IXMLDOMAttribute( iface );
497     return IXMLDOMNode_get_baseName( IXMLDOMNode_from_impl(&This->node), p );
498 }
499
500 static HRESULT WINAPI domattr_transformNodeToObject(
501     IXMLDOMAttribute *iface,
502     IXMLDOMNode* domNode, VARIANT var1)
503 {
504     domattr *This = impl_from_IXMLDOMAttribute( iface );
505     return IXMLDOMNode_transformNodeToObject( IXMLDOMNode_from_impl(&This->node), domNode, var1 );
506 }
507
508 static HRESULT WINAPI domattr_get_name(
509     IXMLDOMAttribute *iface,
510     BSTR *p)
511 {
512     domattr *This = impl_from_IXMLDOMAttribute( iface );
513
514     TRACE("(%p)->(%p)\n", This, p);
515
516     return node_get_nodeName(&This->node, p);
517 }
518
519 static HRESULT WINAPI domattr_get_value(
520     IXMLDOMAttribute *iface,
521     VARIANT *value)
522 {
523     domattr *This = impl_from_IXMLDOMAttribute( iface );
524
525     TRACE("(%p)->(%p)\n", This, value);
526
527     return node_get_content(&This->node, value);
528 }
529
530 static HRESULT WINAPI domattr_put_value(
531     IXMLDOMAttribute *iface,
532     VARIANT value)
533 {
534     domattr *This = impl_from_IXMLDOMAttribute( iface );
535
536     TRACE("(%p)->(v%d)\n", This, V_VT(&value));
537
538     return node_put_value(&This->node, &value);
539 }
540
541 static const struct IXMLDOMAttributeVtbl domattr_vtbl =
542 {
543     domattr_QueryInterface,
544     domattr_AddRef,
545     domattr_Release,
546     domattr_GetTypeInfoCount,
547     domattr_GetTypeInfo,
548     domattr_GetIDsOfNames,
549     domattr_Invoke,
550     domattr_get_nodeName,
551     domattr_get_nodeValue,
552     domattr_put_nodeValue,
553     domattr_get_nodeType,
554     domattr_get_parentNode,
555     domattr_get_childNodes,
556     domattr_get_firstChild,
557     domattr_get_lastChild,
558     domattr_get_previousSibling,
559     domattr_get_nextSibling,
560     domattr_get_attributes,
561     domattr_insertBefore,
562     domattr_replaceChild,
563     domattr_removeChild,
564     domattr_appendChild,
565     domattr_hasChildNodes,
566     domattr_get_ownerDocument,
567     domattr_cloneNode,
568     domattr_get_nodeTypeString,
569     domattr_get_text,
570     domattr_put_text,
571     domattr_get_specified,
572     domattr_get_definition,
573     domattr_get_nodeTypedValue,
574     domattr_put_nodeTypedValue,
575     domattr_get_dataType,
576     domattr_put_dataType,
577     domattr_get_xml,
578     domattr_transformNode,
579     domattr_selectNodes,
580     domattr_selectSingleNode,
581     domattr_get_parsed,
582     domattr_get_namespaceURI,
583     domattr_get_prefix,
584     domattr_get_baseName,
585     domattr_transformNodeToObject,
586     domattr_get_name,
587     domattr_get_value,
588     domattr_put_value
589 };
590
591 IUnknown* create_attribute( xmlNodePtr attribute )
592 {
593     domattr *This;
594
595     This = heap_alloc( sizeof *This );
596     if ( !This )
597         return NULL;
598
599     This->lpVtbl = &domattr_vtbl;
600     This->ref = 1;
601
602     init_xmlnode(&This->node, attribute, (IXMLDOMNode*)&This->lpVtbl, NULL);
603
604     return (IUnknown*) &This->lpVtbl;
605 }
606
607 #endif