msxml3: COM cleanup for the IXMLDOMNode iface in xmlnode.
[wine] / dlls / msxml3 / docfrag.c
1 /*
2  *    DOM Document Fragment implementation
3  *
4  * Copyright 2007 Alistair Leslie-Hughes
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 _domfrag
41 {
42     xmlnode node;
43     IXMLDOMDocumentFragment IXMLDOMDocumentFragment_iface;
44     LONG ref;
45 } domfrag;
46
47 static inline domfrag *impl_from_IXMLDOMDocumentFragment( IXMLDOMDocumentFragment *iface )
48 {
49     return CONTAINING_RECORD(iface, domfrag, IXMLDOMDocumentFragment_iface);
50 }
51
52 static HRESULT WINAPI domfrag_QueryInterface(
53     IXMLDOMDocumentFragment *iface,
54     REFIID riid,
55     void** ppvObject )
56 {
57     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
58     TRACE("(%p)->(%s %p)\n", This, debugstr_guid(riid), ppvObject);
59
60     if ( IsEqualGUID( riid, &IID_IXMLDOMDocumentFragment ) ||
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 domfrag_AddRef(
82     IXMLDOMDocumentFragment *iface )
83 {
84     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
85     return InterlockedIncrement( &This->ref );
86 }
87
88 static ULONG WINAPI domfrag_Release(
89     IXMLDOMDocumentFragment *iface )
90 {
91     domfrag *This = impl_from_IXMLDOMDocumentFragment( 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 domfrag_GetTypeInfoCount(
105     IXMLDOMDocumentFragment *iface,
106     UINT* pctinfo )
107 {
108     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
109
110     TRACE("(%p)->(%p)\n", This, pctinfo);
111
112     *pctinfo = 1;
113
114     return S_OK;
115 }
116
117 static HRESULT WINAPI domfrag_GetTypeInfo(
118     IXMLDOMDocumentFragment *iface,
119     UINT iTInfo, LCID lcid,
120     ITypeInfo** ppTInfo )
121 {
122     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
123     HRESULT hr;
124
125     TRACE("(%p)->(%u %u %p)\n", This, iTInfo, lcid, ppTInfo);
126
127     hr = get_typeinfo(IXMLDOMDocumentFragment_tid, ppTInfo);
128
129     return hr;
130 }
131
132 static HRESULT WINAPI domfrag_GetIDsOfNames(
133     IXMLDOMDocumentFragment *iface,
134     REFIID riid, LPOLESTR* rgszNames,
135     UINT cNames, LCID lcid, DISPID* rgDispId )
136 {
137     domfrag *This = impl_from_IXMLDOMDocumentFragment( 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(IXMLDOMDocumentFragment_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 domfrag_Invoke(
158     IXMLDOMDocumentFragment *iface,
159     DISPID dispIdMember, REFIID riid, LCID lcid,
160     WORD wFlags, DISPPARAMS* pDispParams, VARIANT* pVarResult,
161     EXCEPINFO* pExcepInfo, UINT* puArgErr )
162 {
163     domfrag *This = impl_from_IXMLDOMDocumentFragment( 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(IXMLDOMDocumentFragment_tid, &typeinfo);
171     if(SUCCEEDED(hr))
172     {
173         hr = ITypeInfo_Invoke(typeinfo, &This->IXMLDOMDocumentFragment_iface, dispIdMember,
174                 wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr);
175         ITypeInfo_Release(typeinfo);
176     }
177
178     return hr;
179 }
180
181 static HRESULT WINAPI domfrag_get_nodeName(
182     IXMLDOMDocumentFragment *iface,
183     BSTR* p )
184 {
185     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
186
187     static const WCHAR document_fragmentW[] =
188         {'#','d','o','c','u','m','e','n','t','-','f','r','a','g','m','e','n','t',0};
189
190     TRACE("(%p)->(%p)\n", This, p);
191
192     return return_bstr(document_fragmentW, p);
193 }
194
195 static HRESULT WINAPI domfrag_get_nodeValue(
196     IXMLDOMDocumentFragment *iface,
197     VARIANT* value)
198 {
199     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
200
201     FIXME("(%p)->(%p)\n", This, value);
202
203     if(!value)
204         return E_INVALIDARG;
205
206     V_VT(value) = VT_NULL;
207     return S_FALSE;
208 }
209
210 static HRESULT WINAPI domfrag_put_nodeValue(
211     IXMLDOMDocumentFragment *iface,
212     VARIANT value)
213 {
214     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
215     TRACE("(%p)->(v%d)\n", This, V_VT(&value));
216     return E_FAIL;
217 }
218
219 static HRESULT WINAPI domfrag_get_nodeType(
220     IXMLDOMDocumentFragment *iface,
221     DOMNodeType* domNodeType )
222 {
223     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
224
225     TRACE("(%p)->(%p)\n", This, domNodeType);
226
227     *domNodeType = NODE_DOCUMENT_FRAGMENT;
228     return S_OK;
229 }
230
231 static HRESULT WINAPI domfrag_get_parentNode(
232     IXMLDOMDocumentFragment *iface,
233     IXMLDOMNode** parent )
234 {
235     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
236
237     TRACE("(%p)->(%p)\n", This, parent);
238
239     return node_get_parent(&This->node, parent);
240 }
241
242 static HRESULT WINAPI domfrag_get_childNodes(
243     IXMLDOMDocumentFragment *iface,
244     IXMLDOMNodeList** outList)
245 {
246     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
247
248     TRACE("(%p)->(%p)\n", This, outList);
249
250     return node_get_child_nodes(&This->node, outList);
251 }
252
253 static HRESULT WINAPI domfrag_get_firstChild(
254     IXMLDOMDocumentFragment *iface,
255     IXMLDOMNode** domNode)
256 {
257     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
258
259     TRACE("(%p)->(%p)\n", This, domNode);
260
261     return node_get_first_child(&This->node, domNode);
262 }
263
264 static HRESULT WINAPI domfrag_get_lastChild(
265     IXMLDOMDocumentFragment *iface,
266     IXMLDOMNode** domNode)
267 {
268     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
269
270     TRACE("(%p)->(%p)\n", This, domNode);
271
272     return node_get_last_child(&This->node, domNode);
273 }
274
275 static HRESULT WINAPI domfrag_get_previousSibling(
276     IXMLDOMDocumentFragment *iface,
277     IXMLDOMNode** domNode)
278 {
279     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
280
281     TRACE("(%p)->(%p)\n", This, domNode);
282
283     return return_null_node(domNode);
284 }
285
286 static HRESULT WINAPI domfrag_get_nextSibling(
287     IXMLDOMDocumentFragment *iface,
288     IXMLDOMNode** domNode)
289 {
290     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
291
292     TRACE("(%p)->(%p)\n", This, domNode);
293
294     return return_null_node(domNode);
295 }
296
297 static HRESULT WINAPI domfrag_get_attributes(
298     IXMLDOMDocumentFragment *iface,
299     IXMLDOMNamedNodeMap** attributeMap)
300 {
301     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
302
303     TRACE("(%p)->(%p)\n", This, attributeMap);
304
305     return return_null_ptr((void**)attributeMap);
306 }
307
308 static HRESULT WINAPI domfrag_insertBefore(
309     IXMLDOMDocumentFragment *iface,
310     IXMLDOMNode* newNode, VARIANT refChild,
311     IXMLDOMNode** outOldNode)
312 {
313     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
314
315     TRACE("(%p)->(%p x%d %p)\n", This, newNode, V_VT(&refChild), outOldNode);
316
317     /* TODO: test */
318     return node_insert_before(&This->node, newNode, &refChild, outOldNode);
319 }
320
321 static HRESULT WINAPI domfrag_replaceChild(
322     IXMLDOMDocumentFragment *iface,
323     IXMLDOMNode* newNode,
324     IXMLDOMNode* oldNode,
325     IXMLDOMNode** outOldNode)
326 {
327     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
328
329     TRACE("(%p)->(%p %p %p)\n", This, newNode, oldNode, outOldNode);
330
331     /* TODO: test */
332     return node_replace_child(&This->node, newNode, oldNode, outOldNode);
333 }
334
335 static HRESULT WINAPI domfrag_removeChild(
336     IXMLDOMDocumentFragment *iface,
337     IXMLDOMNode* domNode, IXMLDOMNode** oldNode)
338 {
339     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
340     return IXMLDOMNode_removeChild( &This->node.IXMLDOMNode_iface, domNode, oldNode );
341 }
342
343 static HRESULT WINAPI domfrag_appendChild(
344     IXMLDOMDocumentFragment *iface,
345     IXMLDOMNode* newNode, IXMLDOMNode** outNewNode)
346 {
347     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
348     return IXMLDOMNode_appendChild( &This->node.IXMLDOMNode_iface, newNode, outNewNode );
349 }
350
351 static HRESULT WINAPI domfrag_hasChildNodes(
352     IXMLDOMDocumentFragment *iface,
353     VARIANT_BOOL* pbool)
354 {
355     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
356     return IXMLDOMNode_hasChildNodes( &This->node.IXMLDOMNode_iface, pbool );
357 }
358
359 static HRESULT WINAPI domfrag_get_ownerDocument(
360     IXMLDOMDocumentFragment *iface,
361     IXMLDOMDocument** domDocument)
362 {
363     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
364     return IXMLDOMNode_get_ownerDocument( &This->node.IXMLDOMNode_iface, domDocument );
365 }
366
367 static HRESULT WINAPI domfrag_cloneNode(
368     IXMLDOMDocumentFragment *iface,
369     VARIANT_BOOL deep, IXMLDOMNode** outNode)
370 {
371     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
372     TRACE("(%p)->(%d %p)\n", This, deep, outNode);
373     return node_clone( &This->node, deep, outNode );
374 }
375
376 static HRESULT WINAPI domfrag_get_nodeTypeString(
377     IXMLDOMDocumentFragment *iface,
378     BSTR* p)
379 {
380     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
381     static const WCHAR documentfragmentW[] = {'d','o','c','u','m','e','n','t','f','r','a','g','m','e','n','t',0};
382
383     TRACE("(%p)->(%p)\n", This, p);
384
385     return return_bstr(documentfragmentW, p);
386 }
387
388 static HRESULT WINAPI domfrag_get_text(
389     IXMLDOMDocumentFragment *iface,
390     BSTR* p)
391 {
392     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
393     return IXMLDOMNode_get_text( &This->node.IXMLDOMNode_iface, p );
394 }
395
396 static HRESULT WINAPI domfrag_put_text(
397     IXMLDOMDocumentFragment *iface,
398     BSTR p)
399 {
400     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
401     TRACE("(%p)->(%s)\n", This, debugstr_w(p));
402     return node_put_text( &This->node, p );
403 }
404
405 static HRESULT WINAPI domfrag_get_specified(
406     IXMLDOMDocumentFragment *iface,
407     VARIANT_BOOL* isSpecified)
408 {
409     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
410     FIXME("(%p)->(%p) stub!\n", This, isSpecified);
411     *isSpecified = VARIANT_TRUE;
412     return S_OK;
413 }
414
415 static HRESULT WINAPI domfrag_get_definition(
416     IXMLDOMDocumentFragment *iface,
417     IXMLDOMNode** definitionNode)
418 {
419     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
420     FIXME("(%p)->(%p)\n", This, definitionNode);
421     return E_NOTIMPL;
422 }
423
424 static HRESULT WINAPI domfrag_get_nodeTypedValue(
425     IXMLDOMDocumentFragment *iface,
426     VARIANT* var1)
427 {
428     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
429     FIXME("(%p)->(%p)\n", This, var1);
430     return return_null_var(var1);
431 }
432
433 static HRESULT WINAPI domfrag_put_nodeTypedValue(
434     IXMLDOMDocumentFragment *iface,
435     VARIANT typedValue)
436 {
437     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
438     FIXME("(%p)->(%s)\n", This, debugstr_variant(&typedValue));
439     return E_NOTIMPL;
440 }
441
442 static HRESULT WINAPI domfrag_get_dataType(
443     IXMLDOMDocumentFragment *iface,
444     VARIANT* typename)
445 {
446     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
447     TRACE("(%p)->(%p)\n", This, typename);
448     return return_null_var( typename );
449 }
450
451 static HRESULT WINAPI domfrag_put_dataType(
452     IXMLDOMDocumentFragment *iface,
453     BSTR p)
454 {
455     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
456
457     FIXME("(%p)->(%s)\n", This, debugstr_w(p));
458
459     if(!p)
460         return E_INVALIDARG;
461
462     return E_FAIL;
463 }
464
465 static HRESULT WINAPI domfrag_get_xml(
466     IXMLDOMDocumentFragment *iface,
467     BSTR* p)
468 {
469     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
470
471     TRACE("(%p)->(%p)\n", This, p);
472
473     return node_get_xml(&This->node, FALSE, FALSE, p);
474 }
475
476 static HRESULT WINAPI domfrag_transformNode(
477     IXMLDOMDocumentFragment *iface,
478     IXMLDOMNode* domNode, BSTR* p)
479 {
480     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
481     return IXMLDOMNode_transformNode( &This->node.IXMLDOMNode_iface, domNode, p );
482 }
483
484 static HRESULT WINAPI domfrag_selectNodes(
485     IXMLDOMDocumentFragment *iface,
486     BSTR p, IXMLDOMNodeList** outList)
487 {
488     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
489     return IXMLDOMNode_selectNodes( &This->node.IXMLDOMNode_iface, p, outList );
490 }
491
492 static HRESULT WINAPI domfrag_selectSingleNode(
493     IXMLDOMDocumentFragment *iface,
494     BSTR p, IXMLDOMNode** outNode)
495 {
496     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
497     return IXMLDOMNode_selectSingleNode( &This->node.IXMLDOMNode_iface, p, outNode );
498 }
499
500 static HRESULT WINAPI domfrag_get_parsed(
501     IXMLDOMDocumentFragment *iface,
502     VARIANT_BOOL* isParsed)
503 {
504     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
505     FIXME("(%p)->(%p) stub!\n", This, isParsed);
506     *isParsed = VARIANT_TRUE;
507     return S_OK;
508 }
509
510 static HRESULT WINAPI domfrag_get_namespaceURI(
511     IXMLDOMDocumentFragment *iface,
512     BSTR* p)
513 {
514     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
515     TRACE("(%p)->(%p)\n", This, p);
516     return node_get_namespaceURI(&This->node, p);
517 }
518
519 static HRESULT WINAPI domfrag_get_prefix(
520     IXMLDOMDocumentFragment *iface,
521     BSTR* prefix)
522 {
523     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
524     TRACE("(%p)->(%p)\n", This, prefix);
525     return return_null_bstr( prefix );
526 }
527
528 static HRESULT WINAPI domfrag_get_baseName(
529     IXMLDOMDocumentFragment *iface,
530     BSTR* name)
531 {
532     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
533     FIXME("(%p)->(%p): needs test\n", This, name);
534     return return_null_bstr( name );
535 }
536
537 static HRESULT WINAPI domfrag_transformNodeToObject(
538     IXMLDOMDocumentFragment *iface,
539     IXMLDOMNode* domNode, VARIANT var1)
540 {
541     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
542     FIXME("(%p)->(%p %s)\n", This, domNode, debugstr_variant(&var1));
543     return E_NOTIMPL;
544 }
545
546 static const struct IXMLDOMDocumentFragmentVtbl domfrag_vtbl =
547 {
548     domfrag_QueryInterface,
549     domfrag_AddRef,
550     domfrag_Release,
551     domfrag_GetTypeInfoCount,
552     domfrag_GetTypeInfo,
553     domfrag_GetIDsOfNames,
554     domfrag_Invoke,
555     domfrag_get_nodeName,
556     domfrag_get_nodeValue,
557     domfrag_put_nodeValue,
558     domfrag_get_nodeType,
559     domfrag_get_parentNode,
560     domfrag_get_childNodes,
561     domfrag_get_firstChild,
562     domfrag_get_lastChild,
563     domfrag_get_previousSibling,
564     domfrag_get_nextSibling,
565     domfrag_get_attributes,
566     domfrag_insertBefore,
567     domfrag_replaceChild,
568     domfrag_removeChild,
569     domfrag_appendChild,
570     domfrag_hasChildNodes,
571     domfrag_get_ownerDocument,
572     domfrag_cloneNode,
573     domfrag_get_nodeTypeString,
574     domfrag_get_text,
575     domfrag_put_text,
576     domfrag_get_specified,
577     domfrag_get_definition,
578     domfrag_get_nodeTypedValue,
579     domfrag_put_nodeTypedValue,
580     domfrag_get_dataType,
581     domfrag_put_dataType,
582     domfrag_get_xml,
583     domfrag_transformNode,
584     domfrag_selectNodes,
585     domfrag_selectSingleNode,
586     domfrag_get_parsed,
587     domfrag_get_namespaceURI,
588     domfrag_get_prefix,
589     domfrag_get_baseName,
590     domfrag_transformNodeToObject
591 };
592
593 IUnknown* create_doc_fragment( xmlNodePtr fragment )
594 {
595     domfrag *This;
596
597     This = heap_alloc( sizeof *This );
598     if ( !This )
599         return NULL;
600
601     This->IXMLDOMDocumentFragment_iface.lpVtbl = &domfrag_vtbl;
602     This->ref = 1;
603
604     init_xmlnode(&This->node, fragment, (IXMLDOMNode*)&This->IXMLDOMDocumentFragment_iface, NULL);
605
606     return (IUnknown*)&This->IXMLDOMDocumentFragment_iface;
607 }
608
609 #endif