msxml3: Get rid of libxml2 accessors in ::substringData() for Text, Comment and CDATA...
[wine] / dlls / msxml3 / cdata.c
1 /*
2  *    DOM CDATA node 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 "msxml2.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 _domcdata
41 {
42     xmlnode node;
43     const struct IXMLDOMCDATASectionVtbl *lpVtbl;
44     LONG ref;
45 } domcdata;
46
47 static inline domcdata *impl_from_IXMLDOMCDATASection( IXMLDOMCDATASection *iface )
48 {
49     return (domcdata *)((char*)iface - FIELD_OFFSET(domcdata, lpVtbl));
50 }
51
52 static HRESULT WINAPI domcdata_QueryInterface(
53     IXMLDOMCDATASection *iface,
54     REFIID riid,
55     void** ppvObject )
56 {
57     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
58     TRACE("%p %s %p\n", This, debugstr_guid(riid), ppvObject);
59
60     if ( IsEqualGUID( riid, &IID_IXMLDOMCDATASection ) ||
61          IsEqualGUID( riid, &IID_IXMLDOMCharacterData) ||
62          IsEqualGUID( riid, &IID_IDispatch ) ||
63          IsEqualGUID( riid, &IID_IUnknown ) )
64     {
65         *ppvObject = iface;
66     }
67     else if ( IsEqualGUID( riid, &IID_IXMLDOMNode ) )
68     {
69         *ppvObject = IXMLDOMNode_from_impl(&This->node);
70     }
71     else if ( IsEqualGUID( riid, &IID_IXMLDOMText ) ||
72               IsEqualGUID( riid, &IID_IXMLDOMElement ) )
73     {
74         TRACE("Unsupported interface\n");
75         return E_NOINTERFACE;
76     }
77     else
78     {
79         FIXME("Unsupported interface %s\n", debugstr_guid(riid));
80         return E_NOINTERFACE;
81     }
82
83     IXMLDOMText_AddRef((IUnknown*)*ppvObject);
84     return S_OK;
85 }
86
87 static ULONG WINAPI domcdata_AddRef(
88     IXMLDOMCDATASection *iface )
89 {
90     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
91     return InterlockedIncrement( &This->ref );
92 }
93
94 static ULONG WINAPI domcdata_Release(
95     IXMLDOMCDATASection *iface )
96 {
97     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
98     ULONG ref;
99
100     ref = InterlockedDecrement( &This->ref );
101     if ( ref == 0 )
102     {
103         destroy_xmlnode(&This->node);
104         heap_free( This );
105     }
106
107     return ref;
108 }
109
110 static HRESULT WINAPI domcdata_GetTypeInfoCount(
111     IXMLDOMCDATASection *iface,
112     UINT* pctinfo )
113 {
114     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
115
116     TRACE("(%p)->(%p)\n", This, pctinfo);
117
118     *pctinfo = 1;
119
120     return S_OK;
121 }
122
123 static HRESULT WINAPI domcdata_GetTypeInfo(
124     IXMLDOMCDATASection *iface,
125     UINT iTInfo, LCID lcid,
126     ITypeInfo** ppTInfo )
127 {
128     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
129     HRESULT hr;
130
131     TRACE("(%p)->(%u %u %p)\n", This, iTInfo, lcid, ppTInfo);
132
133     hr = get_typeinfo(IXMLDOMCDATASection_tid, ppTInfo);
134
135     return hr;
136 }
137
138 static HRESULT WINAPI domcdata_GetIDsOfNames(
139     IXMLDOMCDATASection *iface,
140     REFIID riid, LPOLESTR* rgszNames,
141     UINT cNames, LCID lcid, DISPID* rgDispId )
142 {
143     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
144     ITypeInfo *typeinfo;
145     HRESULT hr;
146
147     TRACE("(%p)->(%s %p %u %u %p)\n", This, debugstr_guid(riid), rgszNames, cNames,
148           lcid, rgDispId);
149
150     if(!rgszNames || cNames == 0 || !rgDispId)
151         return E_INVALIDARG;
152
153     hr = get_typeinfo(IXMLDOMCDATASection_tid, &typeinfo);
154     if(SUCCEEDED(hr))
155     {
156         hr = ITypeInfo_GetIDsOfNames(typeinfo, rgszNames, cNames, rgDispId);
157         ITypeInfo_Release(typeinfo);
158     }
159
160     return hr;
161 }
162
163 static HRESULT WINAPI domcdata_Invoke(
164     IXMLDOMCDATASection *iface,
165     DISPID dispIdMember, REFIID riid, LCID lcid,
166     WORD wFlags, DISPPARAMS* pDispParams, VARIANT* pVarResult,
167     EXCEPINFO* pExcepInfo, UINT* puArgErr )
168 {
169     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
170     ITypeInfo *typeinfo;
171     HRESULT hr;
172
173     TRACE("(%p)->(%d %s %d %d %p %p %p %p)\n", This, dispIdMember, debugstr_guid(riid),
174           lcid, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr);
175
176     hr = get_typeinfo(IXMLDOMCDATASection_tid, &typeinfo);
177     if(SUCCEEDED(hr))
178     {
179         hr = ITypeInfo_Invoke(typeinfo, &(This->lpVtbl), dispIdMember, wFlags, pDispParams,
180                 pVarResult, pExcepInfo, puArgErr);
181         ITypeInfo_Release(typeinfo);
182     }
183
184     return hr;
185 }
186
187 static HRESULT WINAPI domcdata_get_nodeName(
188     IXMLDOMCDATASection *iface,
189     BSTR* p )
190 {
191     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
192     return IXMLDOMNode_get_nodeName( IXMLDOMNode_from_impl(&This->node), p );
193 }
194
195 static HRESULT WINAPI domcdata_get_nodeValue(
196     IXMLDOMCDATASection *iface,
197     VARIANT* var1 )
198 {
199     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
200     return IXMLDOMNode_get_nodeValue( IXMLDOMNode_from_impl(&This->node), var1 );
201 }
202
203 static HRESULT WINAPI domcdata_put_nodeValue(
204     IXMLDOMCDATASection *iface,
205     VARIANT var1 )
206 {
207     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
208     return IXMLDOMNode_put_nodeValue( IXMLDOMNode_from_impl(&This->node), var1 );
209 }
210
211 static HRESULT WINAPI domcdata_get_nodeType(
212     IXMLDOMCDATASection *iface,
213     DOMNodeType* domNodeType )
214 {
215     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
216     return IXMLDOMNode_get_nodeType( IXMLDOMNode_from_impl(&This->node), domNodeType );
217 }
218
219 static HRESULT WINAPI domcdata_get_parentNode(
220     IXMLDOMCDATASection *iface,
221     IXMLDOMNode** parent )
222 {
223     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
224     return IXMLDOMNode_get_parentNode( IXMLDOMNode_from_impl(&This->node), parent );
225 }
226
227 static HRESULT WINAPI domcdata_get_childNodes(
228     IXMLDOMCDATASection *iface,
229     IXMLDOMNodeList** outList)
230 {
231     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
232     return IXMLDOMNode_get_childNodes( IXMLDOMNode_from_impl(&This->node), outList );
233 }
234
235 static HRESULT WINAPI domcdata_get_firstChild(
236     IXMLDOMCDATASection *iface,
237     IXMLDOMNode** domNode)
238 {
239     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
240     return IXMLDOMNode_get_firstChild( IXMLDOMNode_from_impl(&This->node), domNode );
241 }
242
243 static HRESULT WINAPI domcdata_get_lastChild(
244     IXMLDOMCDATASection *iface,
245     IXMLDOMNode** domNode)
246 {
247     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
248     return IXMLDOMNode_get_lastChild( IXMLDOMNode_from_impl(&This->node), domNode );
249 }
250
251 static HRESULT WINAPI domcdata_get_previousSibling(
252     IXMLDOMCDATASection *iface,
253     IXMLDOMNode** domNode)
254 {
255     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
256     return IXMLDOMNode_get_previousSibling( IXMLDOMNode_from_impl(&This->node), domNode );
257 }
258
259 static HRESULT WINAPI domcdata_get_nextSibling(
260     IXMLDOMCDATASection *iface,
261     IXMLDOMNode** domNode)
262 {
263     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
264     return IXMLDOMNode_get_nextSibling( IXMLDOMNode_from_impl(&This->node), domNode );
265 }
266
267 static HRESULT WINAPI domcdata_get_attributes(
268     IXMLDOMCDATASection *iface,
269     IXMLDOMNamedNodeMap** attributeMap)
270 {
271         domcdata *This = impl_from_IXMLDOMCDATASection( iface );
272     return IXMLDOMNode_get_attributes( IXMLDOMNode_from_impl(&This->node), attributeMap );
273 }
274
275 static HRESULT WINAPI domcdata_insertBefore(
276     IXMLDOMCDATASection *iface,
277     IXMLDOMNode* newNode, VARIANT var1,
278     IXMLDOMNode** outOldNode)
279 {
280     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
281     return IXMLDOMNode_insertBefore( IXMLDOMNode_from_impl(&This->node), newNode, var1, outOldNode );
282 }
283
284 static HRESULT WINAPI domcdata_replaceChild(
285     IXMLDOMCDATASection *iface,
286     IXMLDOMNode* newNode,
287     IXMLDOMNode* oldNode,
288     IXMLDOMNode** outOldNode)
289 {
290     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
291     return IXMLDOMNode_replaceChild( IXMLDOMNode_from_impl(&This->node), newNode, oldNode, outOldNode );
292 }
293
294 static HRESULT WINAPI domcdata_removeChild(
295     IXMLDOMCDATASection *iface,
296     IXMLDOMNode* domNode, IXMLDOMNode** oldNode)
297 {
298     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
299     return IXMLDOMNode_removeChild( IXMLDOMNode_from_impl(&This->node), domNode, oldNode );
300 }
301
302 static HRESULT WINAPI domcdata_appendChild(
303     IXMLDOMCDATASection *iface,
304     IXMLDOMNode* newNode, IXMLDOMNode** outNewNode)
305 {
306     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
307     return IXMLDOMNode_appendChild( IXMLDOMNode_from_impl(&This->node), newNode, outNewNode );
308 }
309
310 static HRESULT WINAPI domcdata_hasChildNodes(
311     IXMLDOMCDATASection *iface,
312     VARIANT_BOOL* pbool)
313 {
314     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
315     return IXMLDOMNode_hasChildNodes( IXMLDOMNode_from_impl(&This->node), pbool );
316 }
317
318 static HRESULT WINAPI domcdata_get_ownerDocument(
319     IXMLDOMCDATASection *iface,
320     IXMLDOMDocument** domDocument)
321 {
322     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
323     return IXMLDOMNode_get_ownerDocument( IXMLDOMNode_from_impl(&This->node), domDocument );
324 }
325
326 static HRESULT WINAPI domcdata_cloneNode(
327     IXMLDOMCDATASection *iface,
328     VARIANT_BOOL pbool, IXMLDOMNode** outNode)
329 {
330     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
331     return IXMLDOMNode_cloneNode( IXMLDOMNode_from_impl(&This->node), pbool, outNode );
332 }
333
334 static HRESULT WINAPI domcdata_get_nodeTypeString(
335     IXMLDOMCDATASection *iface,
336     BSTR* p)
337 {
338     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
339     return IXMLDOMNode_get_nodeTypeString( IXMLDOMNode_from_impl(&This->node), p );
340 }
341
342 static HRESULT WINAPI domcdata_get_text(
343     IXMLDOMCDATASection *iface,
344     BSTR* p)
345 {
346     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
347     return IXMLDOMNode_get_text( IXMLDOMNode_from_impl(&This->node), p );
348 }
349
350 static HRESULT WINAPI domcdata_put_text(
351     IXMLDOMCDATASection *iface,
352     BSTR p)
353 {
354     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
355     return IXMLDOMNode_put_text( IXMLDOMNode_from_impl(&This->node), p );
356 }
357
358 static HRESULT WINAPI domcdata_get_specified(
359     IXMLDOMCDATASection *iface,
360     VARIANT_BOOL* pbool)
361 {
362     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
363     return IXMLDOMNode_get_specified( IXMLDOMNode_from_impl(&This->node), pbool );
364 }
365
366 static HRESULT WINAPI domcdata_get_definition(
367     IXMLDOMCDATASection *iface,
368     IXMLDOMNode** domNode)
369 {
370     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
371     return IXMLDOMNode_get_definition( IXMLDOMNode_from_impl(&This->node), domNode );
372 }
373
374 static HRESULT WINAPI domcdata_get_nodeTypedValue(
375     IXMLDOMCDATASection *iface,
376     VARIANT* var1)
377 {
378     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
379     return IXMLDOMNode_get_nodeTypedValue( IXMLDOMNode_from_impl(&This->node), var1 );
380 }
381
382 static HRESULT WINAPI domcdata_put_nodeTypedValue(
383     IXMLDOMCDATASection *iface,
384     VARIANT var1)
385 {
386     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
387     return IXMLDOMNode_put_nodeTypedValue( IXMLDOMNode_from_impl(&This->node), var1 );
388 }
389
390 static HRESULT WINAPI domcdata_get_dataType(
391     IXMLDOMCDATASection *iface,
392     VARIANT* var1)
393 {
394     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
395     return IXMLDOMNode_get_dataType( IXMLDOMNode_from_impl(&This->node), var1 );
396 }
397
398 static HRESULT WINAPI domcdata_put_dataType(
399     IXMLDOMCDATASection *iface,
400     BSTR p)
401 {
402     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
403     return IXMLDOMNode_put_dataType( IXMLDOMNode_from_impl(&This->node), p );
404 }
405
406 static HRESULT WINAPI domcdata_get_xml(
407     IXMLDOMCDATASection *iface,
408     BSTR* p)
409 {
410     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
411     return IXMLDOMNode_get_xml( IXMLDOMNode_from_impl(&This->node), p );
412 }
413
414 static HRESULT WINAPI domcdata_transformNode(
415     IXMLDOMCDATASection *iface,
416     IXMLDOMNode* domNode, BSTR* p)
417 {
418     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
419     return IXMLDOMNode_transformNode( IXMLDOMNode_from_impl(&This->node), domNode, p );
420 }
421
422 static HRESULT WINAPI domcdata_selectNodes(
423     IXMLDOMCDATASection *iface,
424     BSTR p, IXMLDOMNodeList** outList)
425 {
426     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
427     return IXMLDOMNode_selectNodes( IXMLDOMNode_from_impl(&This->node), p, outList );
428 }
429
430 static HRESULT WINAPI domcdata_selectSingleNode(
431     IXMLDOMCDATASection *iface,
432     BSTR p, IXMLDOMNode** outNode)
433 {
434     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
435     return IXMLDOMNode_selectSingleNode( IXMLDOMNode_from_impl(&This->node), p, outNode );
436 }
437
438 static HRESULT WINAPI domcdata_get_parsed(
439     IXMLDOMCDATASection *iface,
440     VARIANT_BOOL* pbool)
441 {
442     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
443     return IXMLDOMNode_get_parsed( IXMLDOMNode_from_impl(&This->node), pbool );
444 }
445
446 static HRESULT WINAPI domcdata_get_namespaceURI(
447     IXMLDOMCDATASection *iface,
448     BSTR* p)
449 {
450     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
451     return IXMLDOMNode_get_namespaceURI( IXMLDOMNode_from_impl(&This->node), p );
452 }
453
454 static HRESULT WINAPI domcdata_get_prefix(
455     IXMLDOMCDATASection *iface,
456     BSTR* p)
457 {
458     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
459     return IXMLDOMNode_get_prefix( IXMLDOMNode_from_impl(&This->node), p );
460 }
461
462 static HRESULT WINAPI domcdata_get_baseName(
463     IXMLDOMCDATASection *iface,
464     BSTR* p)
465 {
466     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
467     return IXMLDOMNode_get_baseName( IXMLDOMNode_from_impl(&This->node), p );
468 }
469
470 static HRESULT WINAPI domcdata_transformNodeToObject(
471     IXMLDOMCDATASection *iface,
472     IXMLDOMNode* domNode, VARIANT var1)
473 {
474     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
475     return IXMLDOMNode_transformNodeToObject( IXMLDOMNode_from_impl(&This->node), domNode, var1 );
476 }
477
478 static HRESULT WINAPI domcdata_get_data(
479     IXMLDOMCDATASection *iface,
480     BSTR *p)
481 {
482     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
483     HRESULT hr;
484     VARIANT vRet;
485
486     if(!p)
487         return E_INVALIDARG;
488
489     hr = IXMLDOMNode_get_nodeValue( IXMLDOMNode_from_impl(&This->node), &vRet );
490     if(hr == S_OK)
491     {
492         *p = V_BSTR(&vRet);
493     }
494
495     return hr;
496 }
497
498 static HRESULT WINAPI domcdata_put_data(
499     IXMLDOMCDATASection *iface,
500     BSTR data)
501 {
502     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
503     VARIANT val;
504
505     TRACE("%p %s\n", This, debugstr_w(data) );
506
507     V_VT(&val) = VT_BSTR;
508     V_BSTR(&val) = data;
509
510     return IXMLDOMNode_put_nodeValue( IXMLDOMNode_from_impl(&This->node), val );
511 }
512
513 static HRESULT WINAPI domcdata_get_length(
514     IXMLDOMCDATASection *iface,
515     LONG *len)
516 {
517     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
518     HRESULT hr;
519     BSTR data;
520
521     TRACE("%p %p\n", This, len);
522
523     if(!len)
524         return E_INVALIDARG;
525
526     hr = IXMLDOMCDATASection_get_data(iface, &data);
527     if(hr == S_OK)
528     {
529         *len = SysStringLen(data);
530         SysFreeString(data);
531     }
532
533     return S_OK;
534 }
535
536 static HRESULT WINAPI domcdata_substringData(
537     IXMLDOMCDATASection *iface,
538     LONG offset, LONG count, BSTR *p)
539 {
540     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
541     HRESULT hr;
542     BSTR data;
543
544     TRACE("%p %d %d %p\n", This, offset, count, p);
545
546     if(!p)
547         return E_INVALIDARG;
548
549     *p = NULL;
550     if(offset < 0 || count < 0)
551         return E_INVALIDARG;
552
553     if(count == 0)
554         return S_FALSE;
555
556     hr = IXMLDOMCDATASection_get_data(iface, &data);
557     if(hr == S_OK)
558     {
559         LONG len = SysStringLen(data);
560
561         if(offset < len)
562         {
563             if(offset + count > len)
564                 *p = SysAllocString(&data[offset]);
565             else
566                 *p = SysAllocStringLen(&data[offset], count);
567         }
568         else
569             hr = S_FALSE;
570
571         SysFreeString(data);
572     }
573
574     return hr;
575 }
576
577 static HRESULT WINAPI domcdata_appendData(
578     IXMLDOMCDATASection *iface,
579     BSTR p)
580 {
581     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
582     HRESULT hr;
583     BSTR data;
584     LONG p_len;
585
586     TRACE("%p %p\n", This, debugstr_w(p));
587
588     /* Nothing to do if NULL or an Empty string passed in. */
589     if((p_len = SysStringLen(p)) == 0) return S_OK;
590
591     hr = IXMLDOMCDATASection_get_data(iface, &data);
592     if(hr == S_OK)
593     {
594         LONG len = SysStringLen(data);
595         BSTR str = SysAllocStringLen(NULL, p_len + len);
596
597         memcpy(str, data, len*sizeof(WCHAR));
598         memcpy(&str[len], p, p_len*sizeof(WCHAR));
599         str[len+p_len] = 0;
600
601         hr = IXMLDOMCDATASection_put_data(iface, str);
602
603         SysFreeString(str);
604         SysFreeString(data);
605     }
606
607     return hr;
608 }
609
610 static HRESULT WINAPI domcdata_insertData(
611     IXMLDOMCDATASection *iface,
612     LONG offset, BSTR p)
613 {
614     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
615     HRESULT hr;
616     BSTR data;
617     LONG p_len;
618
619     TRACE("%p %d %s\n", This, offset, debugstr_w(p));
620
621     /* If have a NULL or empty string, don't do anything. */
622     if((p_len = SysStringLen(p)) == 0)
623         return S_OK;
624
625     if(offset < 0)
626     {
627         return E_INVALIDARG;
628     }
629
630     hr = IXMLDOMCDATASection_get_data(iface, &data);
631     if(hr == S_OK)
632     {
633         LONG len = SysStringLen(data);
634         BSTR str;
635
636         if(len < offset)
637         {
638             SysFreeString(data);
639             return E_INVALIDARG;
640         }
641
642         str = SysAllocStringLen(NULL, len + p_len);
643         /* start part, supplied string and end part */
644         memcpy(str, data, offset*sizeof(WCHAR));
645         memcpy(&str[offset], p, p_len*sizeof(WCHAR));
646         memcpy(&str[offset+p_len], &data[offset], (len-offset)*sizeof(WCHAR));
647         str[len+p_len] = 0;
648
649         hr = IXMLDOMCDATASection_put_data(iface, str);
650
651         SysFreeString(str);
652         SysFreeString(data);
653     }
654
655     return hr;
656 }
657
658 static HRESULT WINAPI domcdata_deleteData(
659     IXMLDOMCDATASection *iface,
660     LONG offset, LONG count)
661 {
662     HRESULT hr;
663     LONG len = -1;
664     BSTR str;
665
666     TRACE("%p %d %d\n", iface, offset, count);
667
668     hr = IXMLDOMCDATASection_get_length(iface, &len);
669     if(hr != S_OK) return hr;
670
671     if((offset < 0) || (offset > len) || (count < 0))
672         return E_INVALIDARG;
673
674     if(len == 0) return S_OK;
675
676     /* cutting start or end */
677     if((offset == 0) || ((count + offset) >= len))
678     {
679         if(offset == 0)
680             IXMLDOMCDATASection_substringData(iface, count, len - count, &str);
681         else
682             IXMLDOMCDATASection_substringData(iface, 0, offset, &str);
683         hr = IXMLDOMCDATASection_put_data(iface, str);
684     }
685     else
686     /* cutting from the inside */
687     {
688         BSTR str_end;
689
690         IXMLDOMCDATASection_substringData(iface, 0, offset, &str);
691         IXMLDOMCDATASection_substringData(iface, offset + count, len - count, &str_end);
692
693         hr = IXMLDOMCDATASection_put_data(iface, str);
694         if(hr == S_OK)
695             hr = IXMLDOMCDATASection_appendData(iface, str_end);
696
697         SysFreeString(str_end);
698     }
699
700     SysFreeString(str);
701
702     return hr;
703 }
704
705 static HRESULT WINAPI domcdata_replaceData(
706     IXMLDOMCDATASection *iface,
707     LONG offset, LONG count, BSTR p)
708 {
709     domcdata *This = impl_from_IXMLDOMCDATASection( iface );
710     HRESULT hr;
711
712     TRACE("%p %d %d %s\n", This, offset, count, debugstr_w(p));
713
714     hr = IXMLDOMCDATASection_deleteData(iface, offset, count);
715
716     if (hr == S_OK)
717        hr = IXMLDOMCDATASection_insertData(iface, offset, p);
718
719     return hr;
720 }
721
722 static HRESULT WINAPI domcdata_splitText(
723     IXMLDOMCDATASection *iface,
724     LONG offset, IXMLDOMText **txtNode)
725 {
726     FIXME("\n");
727     return E_NOTIMPL;
728 }
729
730
731 static const struct IXMLDOMCDATASectionVtbl domcdata_vtbl =
732 {
733     domcdata_QueryInterface,
734     domcdata_AddRef,
735     domcdata_Release,
736     domcdata_GetTypeInfoCount,
737     domcdata_GetTypeInfo,
738     domcdata_GetIDsOfNames,
739     domcdata_Invoke,
740     domcdata_get_nodeName,
741     domcdata_get_nodeValue,
742     domcdata_put_nodeValue,
743     domcdata_get_nodeType,
744     domcdata_get_parentNode,
745     domcdata_get_childNodes,
746     domcdata_get_firstChild,
747     domcdata_get_lastChild,
748     domcdata_get_previousSibling,
749     domcdata_get_nextSibling,
750     domcdata_get_attributes,
751     domcdata_insertBefore,
752     domcdata_replaceChild,
753     domcdata_removeChild,
754     domcdata_appendChild,
755     domcdata_hasChildNodes,
756     domcdata_get_ownerDocument,
757     domcdata_cloneNode,
758     domcdata_get_nodeTypeString,
759     domcdata_get_text,
760     domcdata_put_text,
761     domcdata_get_specified,
762     domcdata_get_definition,
763     domcdata_get_nodeTypedValue,
764     domcdata_put_nodeTypedValue,
765     domcdata_get_dataType,
766     domcdata_put_dataType,
767     domcdata_get_xml,
768     domcdata_transformNode,
769     domcdata_selectNodes,
770     domcdata_selectSingleNode,
771     domcdata_get_parsed,
772     domcdata_get_namespaceURI,
773     domcdata_get_prefix,
774     domcdata_get_baseName,
775     domcdata_transformNodeToObject,
776     domcdata_get_data,
777     domcdata_put_data,
778     domcdata_get_length,
779     domcdata_substringData,
780     domcdata_appendData,
781     domcdata_insertData,
782     domcdata_deleteData,
783     domcdata_replaceData,
784     domcdata_splitText
785 };
786
787 IUnknown* create_cdata( xmlNodePtr text )
788 {
789     domcdata *This;
790
791     This = heap_alloc( sizeof *This );
792     if ( !This )
793         return NULL;
794
795     This->lpVtbl = &domcdata_vtbl;
796     This->ref = 1;
797
798     init_xmlnode(&This->node, text, (IUnknown*)&This->lpVtbl, NULL);
799
800     return (IUnknown*) &This->lpVtbl;
801 }
802
803 #endif