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