msxml3: Remove appendChild() forward.
[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)->(%s)\n", This, debugstr_variant(&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 %s %p)\n", This, newNode, debugstr_variant(&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 *child, IXMLDOMNode **oldChild)
338 {
339     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
340     TRACE("(%p)->(%p %p)\n", This, child, oldChild);
341     return node_remove_child(&This->node, child, oldChild);
342 }
343
344 static HRESULT WINAPI domfrag_appendChild(
345     IXMLDOMDocumentFragment *iface,
346     IXMLDOMNode *child, IXMLDOMNode **outChild)
347 {
348     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
349     TRACE("(%p)->(%p %p)\n", This, child, outChild);
350     return node_append_child(&This->node, child, outChild);
351 }
352
353 static HRESULT WINAPI domfrag_hasChildNodes(
354     IXMLDOMDocumentFragment *iface,
355     VARIANT_BOOL* pbool)
356 {
357     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
358     return IXMLDOMNode_hasChildNodes( &This->node.IXMLDOMNode_iface, pbool );
359 }
360
361 static HRESULT WINAPI domfrag_get_ownerDocument(
362     IXMLDOMDocumentFragment *iface,
363     IXMLDOMDocument** domDocument)
364 {
365     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
366     return IXMLDOMNode_get_ownerDocument( &This->node.IXMLDOMNode_iface, domDocument );
367 }
368
369 static HRESULT WINAPI domfrag_cloneNode(
370     IXMLDOMDocumentFragment *iface,
371     VARIANT_BOOL deep, IXMLDOMNode** outNode)
372 {
373     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
374     TRACE("(%p)->(%d %p)\n", This, deep, outNode);
375     return node_clone( &This->node, deep, outNode );
376 }
377
378 static HRESULT WINAPI domfrag_get_nodeTypeString(
379     IXMLDOMDocumentFragment *iface,
380     BSTR* p)
381 {
382     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
383     static const WCHAR documentfragmentW[] = {'d','o','c','u','m','e','n','t','f','r','a','g','m','e','n','t',0};
384
385     TRACE("(%p)->(%p)\n", This, p);
386
387     return return_bstr(documentfragmentW, p);
388 }
389
390 static HRESULT WINAPI domfrag_get_text(
391     IXMLDOMDocumentFragment *iface,
392     BSTR* p)
393 {
394     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
395     return IXMLDOMNode_get_text( &This->node.IXMLDOMNode_iface, p );
396 }
397
398 static HRESULT WINAPI domfrag_put_text(
399     IXMLDOMDocumentFragment *iface,
400     BSTR p)
401 {
402     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
403     TRACE("(%p)->(%s)\n", This, debugstr_w(p));
404     return node_put_text( &This->node, p );
405 }
406
407 static HRESULT WINAPI domfrag_get_specified(
408     IXMLDOMDocumentFragment *iface,
409     VARIANT_BOOL* isSpecified)
410 {
411     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
412     FIXME("(%p)->(%p) stub!\n", This, isSpecified);
413     *isSpecified = VARIANT_TRUE;
414     return S_OK;
415 }
416
417 static HRESULT WINAPI domfrag_get_definition(
418     IXMLDOMDocumentFragment *iface,
419     IXMLDOMNode** definitionNode)
420 {
421     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
422     FIXME("(%p)->(%p)\n", This, definitionNode);
423     return E_NOTIMPL;
424 }
425
426 static HRESULT WINAPI domfrag_get_nodeTypedValue(
427     IXMLDOMDocumentFragment *iface,
428     VARIANT* var1)
429 {
430     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
431     FIXME("(%p)->(%p)\n", This, var1);
432     return return_null_var(var1);
433 }
434
435 static HRESULT WINAPI domfrag_put_nodeTypedValue(
436     IXMLDOMDocumentFragment *iface,
437     VARIANT typedValue)
438 {
439     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
440     FIXME("(%p)->(%s)\n", This, debugstr_variant(&typedValue));
441     return E_NOTIMPL;
442 }
443
444 static HRESULT WINAPI domfrag_get_dataType(
445     IXMLDOMDocumentFragment *iface,
446     VARIANT* typename)
447 {
448     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
449     TRACE("(%p)->(%p)\n", This, typename);
450     return return_null_var( typename );
451 }
452
453 static HRESULT WINAPI domfrag_put_dataType(
454     IXMLDOMDocumentFragment *iface,
455     BSTR p)
456 {
457     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
458
459     FIXME("(%p)->(%s)\n", This, debugstr_w(p));
460
461     if(!p)
462         return E_INVALIDARG;
463
464     return E_FAIL;
465 }
466
467 static HRESULT WINAPI domfrag_get_xml(
468     IXMLDOMDocumentFragment *iface,
469     BSTR* p)
470 {
471     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
472
473     TRACE("(%p)->(%p)\n", This, p);
474
475     return node_get_xml(&This->node, FALSE, FALSE, p);
476 }
477
478 static HRESULT WINAPI domfrag_transformNode(
479     IXMLDOMDocumentFragment *iface,
480     IXMLDOMNode* domNode, BSTR* p)
481 {
482     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
483     return IXMLDOMNode_transformNode( &This->node.IXMLDOMNode_iface, domNode, p );
484 }
485
486 static HRESULT WINAPI domfrag_selectNodes(
487     IXMLDOMDocumentFragment *iface,
488     BSTR p, IXMLDOMNodeList** outList)
489 {
490     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
491     return IXMLDOMNode_selectNodes( &This->node.IXMLDOMNode_iface, p, outList );
492 }
493
494 static HRESULT WINAPI domfrag_selectSingleNode(
495     IXMLDOMDocumentFragment *iface,
496     BSTR p, IXMLDOMNode** outNode)
497 {
498     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
499     return IXMLDOMNode_selectSingleNode( &This->node.IXMLDOMNode_iface, p, outNode );
500 }
501
502 static HRESULT WINAPI domfrag_get_parsed(
503     IXMLDOMDocumentFragment *iface,
504     VARIANT_BOOL* isParsed)
505 {
506     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
507     FIXME("(%p)->(%p) stub!\n", This, isParsed);
508     *isParsed = VARIANT_TRUE;
509     return S_OK;
510 }
511
512 static HRESULT WINAPI domfrag_get_namespaceURI(
513     IXMLDOMDocumentFragment *iface,
514     BSTR* p)
515 {
516     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
517     TRACE("(%p)->(%p)\n", This, p);
518     return node_get_namespaceURI(&This->node, p);
519 }
520
521 static HRESULT WINAPI domfrag_get_prefix(
522     IXMLDOMDocumentFragment *iface,
523     BSTR* prefix)
524 {
525     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
526     TRACE("(%p)->(%p)\n", This, prefix);
527     return return_null_bstr( prefix );
528 }
529
530 static HRESULT WINAPI domfrag_get_baseName(
531     IXMLDOMDocumentFragment *iface,
532     BSTR* name)
533 {
534     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
535     FIXME("(%p)->(%p): needs test\n", This, name);
536     return return_null_bstr( name );
537 }
538
539 static HRESULT WINAPI domfrag_transformNodeToObject(
540     IXMLDOMDocumentFragment *iface,
541     IXMLDOMNode* domNode, VARIANT var1)
542 {
543     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
544     FIXME("(%p)->(%p %s)\n", This, domNode, debugstr_variant(&var1));
545     return E_NOTIMPL;
546 }
547
548 static const struct IXMLDOMDocumentFragmentVtbl domfrag_vtbl =
549 {
550     domfrag_QueryInterface,
551     domfrag_AddRef,
552     domfrag_Release,
553     domfrag_GetTypeInfoCount,
554     domfrag_GetTypeInfo,
555     domfrag_GetIDsOfNames,
556     domfrag_Invoke,
557     domfrag_get_nodeName,
558     domfrag_get_nodeValue,
559     domfrag_put_nodeValue,
560     domfrag_get_nodeType,
561     domfrag_get_parentNode,
562     domfrag_get_childNodes,
563     domfrag_get_firstChild,
564     domfrag_get_lastChild,
565     domfrag_get_previousSibling,
566     domfrag_get_nextSibling,
567     domfrag_get_attributes,
568     domfrag_insertBefore,
569     domfrag_replaceChild,
570     domfrag_removeChild,
571     domfrag_appendChild,
572     domfrag_hasChildNodes,
573     domfrag_get_ownerDocument,
574     domfrag_cloneNode,
575     domfrag_get_nodeTypeString,
576     domfrag_get_text,
577     domfrag_put_text,
578     domfrag_get_specified,
579     domfrag_get_definition,
580     domfrag_get_nodeTypedValue,
581     domfrag_put_nodeTypedValue,
582     domfrag_get_dataType,
583     domfrag_put_dataType,
584     domfrag_get_xml,
585     domfrag_transformNode,
586     domfrag_selectNodes,
587     domfrag_selectSingleNode,
588     domfrag_get_parsed,
589     domfrag_get_namespaceURI,
590     domfrag_get_prefix,
591     domfrag_get_baseName,
592     domfrag_transformNodeToObject
593 };
594
595 IUnknown* create_doc_fragment( xmlNodePtr fragment )
596 {
597     domfrag *This;
598
599     This = heap_alloc( sizeof *This );
600     if ( !This )
601         return NULL;
602
603     This->IXMLDOMDocumentFragment_iface.lpVtbl = &domfrag_vtbl;
604     This->ref = 1;
605
606     init_xmlnode(&This->node, fragment, (IXMLDOMNode*)&This->IXMLDOMDocumentFragment_iface, NULL);
607
608     return (IUnknown*)&This->IXMLDOMDocumentFragment_iface;
609 }
610
611 #endif