Release 1.5.29.
[wine] / dlls / msxml3 / text.c
1 /*
2  *    DOM text node implementation
3  *
4  * Copyright 2006 Huw Davies
5  * Copyright 2007-2008 Alistair Leslie-Hughes
6  *
7  * This library is free software; you can redistribute it and/or
8  * modify it under the terms of the GNU Lesser General Public
9  * License as published by the Free Software Foundation; either
10  * version 2.1 of the License, or (at your option) any later version.
11  *
12  * This library is distributed in the hope that it will be useful,
13  * but WITHOUT ANY WARRANTY; without even the implied warranty of
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
15  * Lesser General Public License for more details.
16  *
17  * You should have received a copy of the GNU Lesser General Public
18  * License along with this library; if not, write to the Free Software
19  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
20  */
21
22 #define COBJMACROS
23
24 #include "config.h"
25
26 #include <stdarg.h>
27 #ifdef HAVE_LIBXML2
28 # include <libxml/parser.h>
29 # include <libxml/parserInternals.h>
30 # include <libxml/xmlerror.h>
31 #endif
32
33 #include "windef.h"
34 #include "winbase.h"
35 #include "winuser.h"
36 #include "ole2.h"
37 #include "msxml6.h"
38
39 #include "msxml_private.h"
40
41 #include "wine/debug.h"
42
43 WINE_DEFAULT_DEBUG_CHANNEL(msxml);
44
45 #ifdef HAVE_LIBXML2
46
47 typedef struct _domtext
48 {
49     xmlnode node;
50     IXMLDOMText IXMLDOMText_iface;
51     LONG ref;
52 } domtext;
53
54 static inline domtext *impl_from_IXMLDOMText( IXMLDOMText *iface )
55 {
56     return CONTAINING_RECORD(iface, domtext, IXMLDOMText_iface);
57 }
58
59 static void domtext_reset_noenc(domtext *This)
60 {
61     This->node.node->name = NULL;
62 }
63
64 static HRESULT WINAPI domtext_QueryInterface(
65     IXMLDOMText *iface,
66     REFIID riid,
67     void** ppvObject )
68 {
69     domtext *This = impl_from_IXMLDOMText( iface );
70     TRACE("(%p)->(%s %p)\n", This, debugstr_guid(riid), ppvObject);
71
72     if ( IsEqualGUID( riid, &IID_IXMLDOMText ) ||
73          IsEqualGUID( riid, &IID_IXMLDOMCharacterData) ||
74          IsEqualGUID( riid, &IID_IXMLDOMNode ) ||
75          IsEqualGUID( riid, &IID_IDispatch ) ||
76          IsEqualGUID( riid, &IID_IUnknown ) )
77     {
78         *ppvObject = iface;
79     }
80     else if(node_query_interface(&This->node, riid, ppvObject))
81     {
82         return *ppvObject ? S_OK : E_NOINTERFACE;
83     }
84     else
85     {
86         TRACE("Unsupported interface %s\n", debugstr_guid(riid));
87         *ppvObject = NULL;
88         return E_NOINTERFACE;
89     }
90
91     IXMLDOMText_AddRef(iface);
92     return S_OK;
93 }
94
95 static ULONG WINAPI domtext_AddRef(
96     IXMLDOMText *iface )
97 {
98     domtext *This = impl_from_IXMLDOMText( iface );
99     ULONG ref = InterlockedIncrement( &This->ref );
100     TRACE("(%p)->(%d)\n", This, ref);
101     return ref;
102 }
103
104 static ULONG WINAPI domtext_Release(
105     IXMLDOMText *iface )
106 {
107     domtext *This = impl_from_IXMLDOMText( iface );
108     ULONG ref = InterlockedDecrement( &This->ref );
109
110     TRACE("(%p)->(%d)\n", This, ref);
111     if ( ref == 0 )
112     {
113         destroy_xmlnode(&This->node);
114         heap_free( This );
115     }
116
117     return ref;
118 }
119
120 static HRESULT WINAPI domtext_GetTypeInfoCount(
121     IXMLDOMText *iface,
122     UINT* pctinfo )
123 {
124     domtext *This = impl_from_IXMLDOMText( iface );
125     return IDispatchEx_GetTypeInfoCount(&This->node.dispex.IDispatchEx_iface, pctinfo);
126 }
127
128 static HRESULT WINAPI domtext_GetTypeInfo(
129     IXMLDOMText *iface,
130     UINT iTInfo, LCID lcid,
131     ITypeInfo** ppTInfo )
132 {
133     domtext *This = impl_from_IXMLDOMText( iface );
134     return IDispatchEx_GetTypeInfo(&This->node.dispex.IDispatchEx_iface,
135         iTInfo, lcid, ppTInfo);
136 }
137
138 static HRESULT WINAPI domtext_GetIDsOfNames(
139     IXMLDOMText *iface,
140     REFIID riid, LPOLESTR* rgszNames,
141     UINT cNames, LCID lcid, DISPID* rgDispId )
142 {
143     domtext *This = impl_from_IXMLDOMText( iface );
144     return IDispatchEx_GetIDsOfNames(&This->node.dispex.IDispatchEx_iface,
145         riid, rgszNames, cNames, lcid, rgDispId);
146 }
147
148 static HRESULT WINAPI domtext_Invoke(
149     IXMLDOMText *iface,
150     DISPID dispIdMember, REFIID riid, LCID lcid,
151     WORD wFlags, DISPPARAMS* pDispParams, VARIANT* pVarResult,
152     EXCEPINFO* pExcepInfo, UINT* puArgErr )
153 {
154     domtext *This = impl_from_IXMLDOMText( iface );
155     return IDispatchEx_Invoke(&This->node.dispex.IDispatchEx_iface,
156         dispIdMember, riid, lcid, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr);
157 }
158
159 static HRESULT WINAPI domtext_get_nodeName(
160     IXMLDOMText *iface,
161     BSTR* p )
162 {
163     domtext *This = impl_from_IXMLDOMText( iface );
164
165     static const WCHAR textW[] = {'#','t','e','x','t',0};
166
167     TRACE("(%p)->(%p)\n", This, p);
168
169     return return_bstr(textW, p);
170 }
171
172 static HRESULT WINAPI domtext_get_nodeValue(
173     IXMLDOMText *iface,
174     VARIANT* value )
175 {
176     domtext *This = impl_from_IXMLDOMText( iface );
177
178     TRACE("(%p)->(%p)\n", This, value);
179
180     return node_get_content(&This->node, value);
181 }
182
183 static HRESULT WINAPI domtext_put_nodeValue(
184     IXMLDOMText *iface,
185     VARIANT value)
186 {
187     domtext *This = impl_from_IXMLDOMText( iface );
188
189     TRACE("(%p)->(%s)\n", This, debugstr_variant(&value));
190
191     domtext_reset_noenc(This);
192     return node_put_value(&This->node, &value);
193 }
194
195 static HRESULT WINAPI domtext_get_nodeType(
196     IXMLDOMText *iface,
197     DOMNodeType* domNodeType )
198 {
199     domtext *This = impl_from_IXMLDOMText( iface );
200
201     TRACE("(%p)->(%p)\n", This, domNodeType);
202
203     *domNodeType = NODE_TEXT;
204     return S_OK;
205 }
206
207 static HRESULT WINAPI domtext_get_parentNode(
208     IXMLDOMText *iface,
209     IXMLDOMNode** parent )
210 {
211     domtext *This = impl_from_IXMLDOMText( iface );
212
213     TRACE("(%p)->(%p)\n", This, parent);
214
215     return node_get_parent(&This->node, parent);
216 }
217
218 static HRESULT WINAPI domtext_get_childNodes(
219     IXMLDOMText *iface,
220     IXMLDOMNodeList** outList)
221 {
222     domtext *This = impl_from_IXMLDOMText( iface );
223
224     TRACE("(%p)->(%p)\n", This, outList);
225
226     return node_get_child_nodes(&This->node, outList);
227 }
228
229 static HRESULT WINAPI domtext_get_firstChild(
230     IXMLDOMText *iface,
231     IXMLDOMNode** domNode)
232 {
233     domtext *This = impl_from_IXMLDOMText( iface );
234
235     TRACE("(%p)->(%p)\n", This, domNode);
236
237     return return_null_node(domNode);
238 }
239
240 static HRESULT WINAPI domtext_get_lastChild(
241     IXMLDOMText *iface,
242     IXMLDOMNode** domNode)
243 {
244     domtext *This = impl_from_IXMLDOMText( iface );
245
246     TRACE("(%p)->(%p)\n", This, domNode);
247
248     return return_null_node(domNode);
249 }
250
251 static HRESULT WINAPI domtext_get_previousSibling(
252     IXMLDOMText *iface,
253     IXMLDOMNode** domNode)
254 {
255     domtext *This = impl_from_IXMLDOMText( iface );
256
257     TRACE("(%p)->(%p)\n", This, domNode);
258
259     return node_get_previous_sibling(&This->node, domNode);
260 }
261
262 static HRESULT WINAPI domtext_get_nextSibling(
263     IXMLDOMText *iface,
264     IXMLDOMNode** domNode)
265 {
266     domtext *This = impl_from_IXMLDOMText( iface );
267
268     TRACE("(%p)->(%p)\n", This, domNode);
269
270     return node_get_next_sibling(&This->node, domNode);
271 }
272
273 static HRESULT WINAPI domtext_get_attributes(
274     IXMLDOMText *iface,
275     IXMLDOMNamedNodeMap** attributeMap)
276 {
277     domtext *This = impl_from_IXMLDOMText( iface );
278
279     TRACE("(%p)->(%p)\n", This, attributeMap);
280
281     return return_null_ptr((void**)attributeMap);
282 }
283
284 static HRESULT WINAPI domtext_insertBefore(
285     IXMLDOMText *iface,
286     IXMLDOMNode* newNode, VARIANT refChild,
287     IXMLDOMNode** outOldNode)
288 {
289     domtext *This = impl_from_IXMLDOMText( iface );
290
291     FIXME("(%p)->(%p %s %p) needs test\n", This, newNode, debugstr_variant(&refChild), outOldNode);
292
293     return node_insert_before(&This->node, newNode, &refChild, outOldNode);
294 }
295
296 static HRESULT WINAPI domtext_replaceChild(
297     IXMLDOMText *iface,
298     IXMLDOMNode* newNode,
299     IXMLDOMNode* oldNode,
300     IXMLDOMNode** outOldNode)
301 {
302     domtext *This = impl_from_IXMLDOMText( iface );
303
304     FIXME("(%p)->(%p %p %p) needs test\n", This, newNode, oldNode, outOldNode);
305
306     return node_replace_child(&This->node, newNode, oldNode, outOldNode);
307 }
308
309 static HRESULT WINAPI domtext_removeChild(
310     IXMLDOMText *iface,
311     IXMLDOMNode *child, IXMLDOMNode **oldChild)
312 {
313     domtext *This = impl_from_IXMLDOMText( iface );
314     TRACE("(%p)->(%p %p)\n", This, child, oldChild);
315     return node_remove_child(&This->node, child, oldChild);
316 }
317
318 static HRESULT WINAPI domtext_appendChild(
319     IXMLDOMText *iface,
320     IXMLDOMNode *child, IXMLDOMNode **outChild)
321 {
322     domtext *This = impl_from_IXMLDOMText( iface );
323     TRACE("(%p)->(%p %p)\n", This, child, outChild);
324     return node_append_child(&This->node, child, outChild);
325 }
326
327 static HRESULT WINAPI domtext_hasChildNodes(
328     IXMLDOMText *iface,
329     VARIANT_BOOL *ret)
330 {
331     domtext *This = impl_from_IXMLDOMText( iface );
332     TRACE("(%p)->(%p)\n", This, ret);
333     return node_has_childnodes(&This->node, ret);
334 }
335
336 static HRESULT WINAPI domtext_get_ownerDocument(
337     IXMLDOMText *iface,
338     IXMLDOMDocument **doc)
339 {
340     domtext *This = impl_from_IXMLDOMText( iface );
341     TRACE("(%p)->(%p)\n", This, doc);
342     return node_get_owner_doc(&This->node, doc);
343 }
344
345 static HRESULT WINAPI domtext_cloneNode(
346     IXMLDOMText *iface,
347     VARIANT_BOOL deep, IXMLDOMNode** outNode)
348 {
349     domtext *This = impl_from_IXMLDOMText( iface );
350     TRACE("(%p)->(%d %p)\n", This, deep, outNode);
351     return node_clone( &This->node, deep, outNode );
352 }
353
354 static HRESULT WINAPI domtext_get_nodeTypeString(
355     IXMLDOMText *iface,
356     BSTR* p)
357 {
358     domtext *This = impl_from_IXMLDOMText( iface );
359     static const WCHAR textW[] = {'t','e','x','t',0};
360
361     TRACE("(%p)->(%p)\n", This, p);
362
363     return return_bstr(textW, p);
364 }
365
366 static HRESULT WINAPI domtext_get_text(
367     IXMLDOMText *iface,
368     BSTR* p)
369 {
370     domtext *This = impl_from_IXMLDOMText( iface );
371     TRACE("(%p)->(%p)\n", This, p);
372     return node_get_text(&This->node, p);
373 }
374
375 static HRESULT WINAPI domtext_put_text(
376     IXMLDOMText *iface,
377     BSTR p)
378 {
379     domtext *This = impl_from_IXMLDOMText( iface );
380     TRACE("(%p)->(%s)\n", This, debugstr_w(p));
381     domtext_reset_noenc(This);
382     return node_put_text( &This->node, p );
383 }
384
385 static HRESULT WINAPI domtext_get_specified(
386     IXMLDOMText *iface,
387     VARIANT_BOOL* isSpecified)
388 {
389     domtext *This = impl_from_IXMLDOMText( iface );
390     FIXME("(%p)->(%p) stub!\n", This, isSpecified);
391     *isSpecified = VARIANT_TRUE;
392     return S_OK;
393 }
394
395 static HRESULT WINAPI domtext_get_definition(
396     IXMLDOMText *iface,
397     IXMLDOMNode** definitionNode)
398 {
399     domtext *This = impl_from_IXMLDOMText( iface );
400     FIXME("(%p)->(%p)\n", This, definitionNode);
401     return E_NOTIMPL;
402 }
403
404 static HRESULT WINAPI domtext_get_nodeTypedValue(
405     IXMLDOMText *iface,
406     VARIANT* var1)
407 {
408     domtext *This = impl_from_IXMLDOMText( iface );
409     IXMLDOMNode* parent = NULL;
410     HRESULT hr;
411
412     TRACE("(%p)->(%p)\n", This, var1);
413
414     if (!var1)
415         return E_INVALIDARG;
416
417     hr = IXMLDOMText_get_parentNode(iface, &parent);
418
419     if (hr == S_OK)
420     {
421         hr = IXMLDOMNode_get_nodeTypedValue(parent, var1);
422         IXMLDOMNode_Release(parent);
423     }
424     else
425     {
426         V_VT(var1) = VT_NULL;
427         V_BSTR(var1) = NULL;
428         hr = S_FALSE;
429     }
430
431     return hr;
432 }
433
434 static HRESULT WINAPI domtext_put_nodeTypedValue(
435     IXMLDOMText *iface,
436     VARIANT value)
437 {
438     domtext *This = impl_from_IXMLDOMText( iface );
439     IXMLDOMNode* parent = NULL;
440     HRESULT hr;
441
442     TRACE("(%p)->(%s)\n", This, debugstr_variant(&value));
443
444     hr = IXMLDOMText_get_parentNode(iface, &parent);
445
446     if (hr == S_OK)
447     {
448         hr = IXMLDOMNode_put_nodeTypedValue(parent, value);
449         IXMLDOMNode_Release(parent);
450     }
451     else
452     {
453         hr = S_FALSE;
454     }
455
456     return hr;
457 }
458
459 static HRESULT WINAPI domtext_get_dataType(
460     IXMLDOMText *iface,
461     VARIANT* dtName)
462 {
463     domtext *This = impl_from_IXMLDOMText( iface );
464     IXMLDOMNode* parent = NULL;
465     HRESULT hr;
466
467     TRACE("(%p)->(%p)\n", This, dtName);
468
469     if (!dtName)
470         return E_INVALIDARG;
471
472     hr = IXMLDOMText_get_parentNode(iface, &parent);
473
474     if (hr == S_OK)
475     {
476         hr = IXMLDOMNode_get_dataType(parent, dtName);
477         IXMLDOMNode_Release(parent);
478     }
479     else
480     {
481         V_VT(dtName) = VT_NULL;
482         V_BSTR(dtName) = NULL;
483         hr = S_FALSE;
484     }
485
486     return hr;
487 }
488
489 static HRESULT WINAPI domtext_put_dataType(
490     IXMLDOMText *iface,
491     BSTR dtName)
492 {
493     domtext *This = impl_from_IXMLDOMText( iface );
494     IXMLDOMNode* parent = NULL;
495     HRESULT hr;
496
497     TRACE("(%p)->(%p)\n", This, dtName);
498
499     if (!dtName)
500         return E_INVALIDARG;
501
502     hr = IXMLDOMText_get_parentNode(iface, &parent);
503
504     if (hr == S_OK)
505     {
506         hr = IXMLDOMNode_put_dataType(parent, dtName);
507         IXMLDOMNode_Release(parent);
508     }
509     else
510     {
511         hr = S_FALSE;
512     }
513
514     return hr;
515 }
516
517 static HRESULT WINAPI domtext_get_xml(
518     IXMLDOMText *iface,
519     BSTR* p)
520 {
521     domtext *This = impl_from_IXMLDOMText( iface );
522
523     TRACE("(%p)->(%p)\n", This, p);
524
525     return node_get_xml(&This->node, FALSE, p);
526 }
527
528 static HRESULT WINAPI domtext_transformNode(
529     IXMLDOMText *iface,
530     IXMLDOMNode *node, BSTR *p)
531 {
532     domtext *This = impl_from_IXMLDOMText( iface );
533     TRACE("(%p)->(%p %p)\n", This, node, p);
534     return node_transform_node(&This->node, node, p);
535 }
536
537 static HRESULT WINAPI domtext_selectNodes(
538     IXMLDOMText *iface,
539     BSTR p, IXMLDOMNodeList** outList)
540 {
541     domtext *This = impl_from_IXMLDOMText( iface );
542     TRACE("(%p)->(%s %p)\n", This, debugstr_w(p), outList);
543     return node_select_nodes(&This->node, p, outList);
544 }
545
546 static HRESULT WINAPI domtext_selectSingleNode(
547     IXMLDOMText *iface,
548     BSTR p, IXMLDOMNode** outNode)
549 {
550     domtext *This = impl_from_IXMLDOMText( iface );
551     TRACE("(%p)->(%s %p)\n", This, debugstr_w(p), outNode);
552     return node_select_singlenode(&This->node, p, outNode);
553 }
554
555 static HRESULT WINAPI domtext_get_parsed(
556     IXMLDOMText *iface,
557     VARIANT_BOOL* isParsed)
558 {
559     domtext *This = impl_from_IXMLDOMText( iface );
560     FIXME("(%p)->(%p) stub!\n", This, isParsed);
561     *isParsed = VARIANT_TRUE;
562     return S_OK;
563 }
564
565 static HRESULT WINAPI domtext_get_namespaceURI(
566     IXMLDOMText *iface,
567     BSTR* p)
568 {
569     domtext *This = impl_from_IXMLDOMText( iface );
570     TRACE("(%p)->(%p)\n", This, p);
571     return node_get_namespaceURI(&This->node, p);
572 }
573
574 static HRESULT WINAPI domtext_get_prefix(
575     IXMLDOMText *iface,
576     BSTR* prefix)
577 {
578     domtext *This = impl_from_IXMLDOMText( iface );
579     TRACE("(%p)->(%p)\n", This, prefix);
580     return return_null_bstr( prefix );
581 }
582
583 static HRESULT WINAPI domtext_get_baseName(
584     IXMLDOMText *iface,
585     BSTR* name)
586 {
587     domtext *This = impl_from_IXMLDOMText( iface );
588     TRACE("(%p)->(%p)\n", This, name);
589     return return_null_bstr( name );
590 }
591
592 static HRESULT WINAPI domtext_transformNodeToObject(
593     IXMLDOMText *iface,
594     IXMLDOMNode* domNode, VARIANT var1)
595 {
596     domtext *This = impl_from_IXMLDOMText( iface );
597     FIXME("(%p)->(%p %s)\n", This, domNode, debugstr_variant(&var1));
598     return E_NOTIMPL;
599 }
600
601 static HRESULT WINAPI domtext_get_data(
602     IXMLDOMText *iface,
603     BSTR *p)
604 {
605     domtext *This = impl_from_IXMLDOMText( iface );
606
607     if(!p)
608         return E_INVALIDARG;
609
610     *p = bstr_from_xmlChar(This->node.node->content);
611     return S_OK;
612 }
613
614 static HRESULT WINAPI domtext_put_data(
615     IXMLDOMText *iface,
616     BSTR data)
617 {
618     domtext *This = impl_from_IXMLDOMText( iface );
619     static WCHAR rnW[] = {'\r','\n',0};
620
621     TRACE("(%p)->(%s)\n", This, debugstr_w(data));
622
623     if (data && !strcmpW(rnW, data))
624         This->node.node->name = xmlStringTextNoenc;
625     else
626         domtext_reset_noenc(This);
627     return node_set_content(&This->node, data);
628 }
629
630 static HRESULT WINAPI domtext_get_length(
631     IXMLDOMText *iface,
632     LONG *len)
633 {
634     domtext *This = impl_from_IXMLDOMText( iface );
635     HRESULT hr;
636     BSTR data;
637
638     TRACE("(%p)->(%p)\n", This, len);
639
640     if(!len)
641         return E_INVALIDARG;
642
643     hr = IXMLDOMText_get_data(iface, &data);
644     if(hr == S_OK)
645     {
646         *len = SysStringLen(data);
647         SysFreeString(data);
648     }
649
650     return hr;
651 }
652
653 static HRESULT WINAPI domtext_substringData(
654     IXMLDOMText *iface,
655     LONG offset, LONG count, BSTR *p)
656 {
657     domtext *This = impl_from_IXMLDOMText( iface );
658     HRESULT hr;
659     BSTR data;
660
661     TRACE("(%p)->(%d %d %p)\n", This, offset, count, p);
662
663     if(!p)
664         return E_INVALIDARG;
665
666     *p = NULL;
667     if(offset < 0 || count < 0)
668         return E_INVALIDARG;
669
670     if(count == 0)
671         return S_FALSE;
672
673     hr = IXMLDOMText_get_data(iface, &data);
674     if(hr == S_OK)
675     {
676         LONG len = SysStringLen(data);
677
678         if(offset < len)
679         {
680             if(offset + count > len)
681                 *p = SysAllocString(&data[offset]);
682             else
683                 *p = SysAllocStringLen(&data[offset], count);
684         }
685         else
686             hr = S_FALSE;
687
688         SysFreeString(data);
689     }
690
691     return hr;
692 }
693
694 static HRESULT WINAPI domtext_appendData(
695     IXMLDOMText *iface,
696     BSTR p)
697 {
698     domtext *This = impl_from_IXMLDOMText( iface );
699     HRESULT hr;
700     BSTR data;
701     LONG p_len;
702
703     TRACE("(%p)->(%s)\n", This, debugstr_w(p));
704
705     /* Nothing to do if NULL or an Empty string passed in. */
706     if((p_len = SysStringLen(p)) == 0) return S_OK;
707
708     hr = IXMLDOMText_get_data(iface, &data);
709     if(hr == S_OK)
710     {
711         LONG len = SysStringLen(data);
712         BSTR str = SysAllocStringLen(NULL, p_len + len);
713
714         memcpy(str, data, len*sizeof(WCHAR));
715         memcpy(&str[len], p, p_len*sizeof(WCHAR));
716         str[len+p_len] = 0;
717
718         hr = IXMLDOMText_put_data(iface, str);
719
720         SysFreeString(str);
721         SysFreeString(data);
722     }
723
724     return hr;
725 }
726
727 static HRESULT WINAPI domtext_insertData(
728     IXMLDOMText *iface,
729     LONG offset, BSTR p)
730 {
731     domtext *This = impl_from_IXMLDOMText( iface );
732     HRESULT hr;
733     BSTR data;
734     LONG p_len;
735
736     TRACE("(%p)->(%d %s)\n", This, offset, debugstr_w(p));
737
738     /* If have a NULL or empty string, don't do anything. */
739     if((p_len = SysStringLen(p)) == 0)
740         return S_OK;
741
742     if(offset < 0)
743     {
744         return E_INVALIDARG;
745     }
746
747     hr = IXMLDOMText_get_data(iface, &data);
748     if(hr == S_OK)
749     {
750         LONG len = SysStringLen(data);
751         BSTR str;
752
753         if(len < offset)
754         {
755             SysFreeString(data);
756             return E_INVALIDARG;
757         }
758
759         str = SysAllocStringLen(NULL, len + p_len);
760         /* start part, supplied string and end part */
761         memcpy(str, data, offset*sizeof(WCHAR));
762         memcpy(&str[offset], p, p_len*sizeof(WCHAR));
763         memcpy(&str[offset+p_len], &data[offset], (len-offset)*sizeof(WCHAR));
764         str[len+p_len] = 0;
765
766         hr = IXMLDOMText_put_data(iface, str);
767
768         SysFreeString(str);
769         SysFreeString(data);
770     }
771
772     return hr;
773 }
774
775 static HRESULT WINAPI domtext_deleteData(
776     IXMLDOMText *iface,
777     LONG offset, LONG count)
778 {
779     HRESULT hr;
780     LONG len = -1;
781     BSTR str;
782
783     TRACE("(%p)->(%d %d)\n", iface, offset, count);
784
785     hr = IXMLDOMText_get_length(iface, &len);
786     if(hr != S_OK) return hr;
787
788     if((offset < 0) || (offset > len) || (count < 0))
789         return E_INVALIDARG;
790
791     if(len == 0) return S_OK;
792
793     /* cutting start or end */
794     if((offset == 0) || ((count + offset) >= len))
795     {
796         if(offset == 0)
797             IXMLDOMText_substringData(iface, count, len - count, &str);
798         else
799             IXMLDOMText_substringData(iface, 0, offset, &str);
800         hr = IXMLDOMText_put_data(iface, str);
801     }
802     else
803     /* cutting from the inside */
804     {
805         BSTR str_end;
806
807         IXMLDOMText_substringData(iface, 0, offset, &str);
808         IXMLDOMText_substringData(iface, offset + count, len - count, &str_end);
809
810         hr = IXMLDOMText_put_data(iface, str);
811         if(hr == S_OK)
812             hr = IXMLDOMText_appendData(iface, str_end);
813
814         SysFreeString(str_end);
815     }
816
817     SysFreeString(str);
818
819     return hr;
820 }
821
822 static HRESULT WINAPI domtext_replaceData(
823     IXMLDOMText *iface,
824     LONG offset, LONG count, BSTR p)
825 {
826     domtext *This = impl_from_IXMLDOMText( iface );
827     HRESULT hr;
828
829     TRACE("(%p)->(%d %d %s)\n", This, offset, count, debugstr_w(p));
830
831     hr = IXMLDOMText_deleteData(iface, offset, count);
832
833     if (hr == S_OK)
834        hr = IXMLDOMText_insertData(iface, offset, p);
835
836     return hr;
837 }
838
839 static HRESULT WINAPI domtext_splitText(
840     IXMLDOMText *iface,
841     LONG offset, IXMLDOMText **txtNode)
842 {
843     domtext *This = impl_from_IXMLDOMText( iface );
844     LONG length = 0;
845
846     TRACE("(%p)->(%d %p)\n", This, offset, txtNode);
847
848     if (!txtNode || offset < 0) return E_INVALIDARG;
849
850     *txtNode = NULL;
851
852     IXMLDOMText_get_length(iface, &length);
853
854     if (offset > length) return E_INVALIDARG;
855     if (offset == length) return S_FALSE;
856
857     FIXME("adjacent text nodes are not supported\n");
858
859     return E_NOTIMPL;
860 }
861
862 static const struct IXMLDOMTextVtbl domtext_vtbl =
863 {
864     domtext_QueryInterface,
865     domtext_AddRef,
866     domtext_Release,
867     domtext_GetTypeInfoCount,
868     domtext_GetTypeInfo,
869     domtext_GetIDsOfNames,
870     domtext_Invoke,
871     domtext_get_nodeName,
872     domtext_get_nodeValue,
873     domtext_put_nodeValue,
874     domtext_get_nodeType,
875     domtext_get_parentNode,
876     domtext_get_childNodes,
877     domtext_get_firstChild,
878     domtext_get_lastChild,
879     domtext_get_previousSibling,
880     domtext_get_nextSibling,
881     domtext_get_attributes,
882     domtext_insertBefore,
883     domtext_replaceChild,
884     domtext_removeChild,
885     domtext_appendChild,
886     domtext_hasChildNodes,
887     domtext_get_ownerDocument,
888     domtext_cloneNode,
889     domtext_get_nodeTypeString,
890     domtext_get_text,
891     domtext_put_text,
892     domtext_get_specified,
893     domtext_get_definition,
894     domtext_get_nodeTypedValue,
895     domtext_put_nodeTypedValue,
896     domtext_get_dataType,
897     domtext_put_dataType,
898     domtext_get_xml,
899     domtext_transformNode,
900     domtext_selectNodes,
901     domtext_selectSingleNode,
902     domtext_get_parsed,
903     domtext_get_namespaceURI,
904     domtext_get_prefix,
905     domtext_get_baseName,
906     domtext_transformNodeToObject,
907     domtext_get_data,
908     domtext_put_data,
909     domtext_get_length,
910     domtext_substringData,
911     domtext_appendData,
912     domtext_insertData,
913     domtext_deleteData,
914     domtext_replaceData,
915     domtext_splitText
916 };
917
918 static const tid_t domtext_iface_tids[] = {
919     IXMLDOMText_tid,
920     0
921 };
922
923 static dispex_static_data_t domtext_dispex = {
924     NULL,
925     IXMLDOMText_tid,
926     NULL,
927     domtext_iface_tids
928 };
929
930 IUnknown* create_text( xmlNodePtr text )
931 {
932     domtext *This;
933
934     This = heap_alloc( sizeof *This );
935     if ( !This )
936         return NULL;
937
938     This->IXMLDOMText_iface.lpVtbl = &domtext_vtbl;
939     This->ref = 1;
940
941     init_xmlnode(&This->node, text, (IXMLDOMNode*)&This->IXMLDOMText_iface, &domtext_dispex);
942
943     return (IUnknown*)&This->IXMLDOMText_iface;
944 }
945
946 #endif