winhelp: Constify the internal .hlp file parsing.
[wine.git] / dlls / msxml3 / domdoc.c
bloba2d72433e3863759f978335f2dd4bab60c148e01
1 /*
2 * DOM Document implementation
4 * Copyright 2005 Mike McCormack
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.
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.
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
21 #define COBJMACROS
23 #include "config.h"
25 #include <stdarg.h>
26 #include <assert.h>
27 #include "windef.h"
28 #include "winbase.h"
29 #include "winuser.h"
30 #include "winnls.h"
31 #include "ole2.h"
32 #include "msxml2.h"
33 #include "wininet.h"
34 #include "urlmon.h"
35 #include "winreg.h"
36 #include "shlwapi.h"
37 #include "ocidl.h"
38 #include "objsafe.h"
40 #include "wine/debug.h"
42 #include "msxml_private.h"
44 WINE_DEFAULT_DEBUG_CHANNEL(msxml);
46 #ifdef HAVE_LIBXML2
48 static const WCHAR SZ_PROPERTY_SELECTION_LANGUAGE[] = {'S','e','l','e','c','t','i','o','n','L','a','n','g','u','a','g','e',0};
49 static const WCHAR SZ_VALUE_XPATH[] = {'X','P','a','t','h',0};
50 static const WCHAR SZ_VALUE_XSLPATTERN[] = {'X','S','L','P','a','t','t','e','r','n',0};
52 typedef struct bsc_t bsc_t;
54 typedef struct _domdoc
56 const struct IXMLDOMDocument2Vtbl *lpVtbl;
57 const struct IPersistStreamVtbl *lpvtblIPersistStream;
58 const struct IObjectWithSiteVtbl *lpvtblIObjectWithSite;
59 const struct IObjectSafetyVtbl *lpvtblIObjectSafety;
60 LONG ref;
61 VARIANT_BOOL async;
62 VARIANT_BOOL validating;
63 VARIANT_BOOL resolving;
64 VARIANT_BOOL preserving;
65 BOOL bUseXPath;
66 IUnknown *node_unk;
67 IXMLDOMNode *node;
68 IXMLDOMSchemaCollection *schema;
69 bsc_t *bsc;
70 HRESULT error;
72 /* IPersistStream */
73 IStream *stream;
75 /* IObjectWithSite*/
76 IUnknown *site;
78 /* IObjectSafety */
79 DWORD safeopt;
80 } domdoc;
82 struct bsc_t {
83 const struct IBindStatusCallbackVtbl *lpVtbl;
85 LONG ref;
87 domdoc *doc;
88 IBinding *binding;
89 IStream *memstream;
92 static inline bsc_t *impl_from_IBindStatusCallback( IBindStatusCallback *iface )
94 return (bsc_t *)((char*)iface - FIELD_OFFSET(bsc_t, lpVtbl));
97 static xmlDocPtr doparse( char *ptr, int len )
99 #ifdef HAVE_XMLREADMEMORY
101 * use xmlReadMemory if possible so we can suppress
102 * writing errors to stderr
104 return xmlReadMemory( ptr, len, NULL, NULL,
105 XML_PARSE_NOERROR | XML_PARSE_NOWARNING | XML_PARSE_NOBLANKS );
106 #else
107 return xmlParseMemory( ptr, len );
108 #endif
111 static HRESULT WINAPI bsc_QueryInterface(
112 IBindStatusCallback *iface,
113 REFIID riid,
114 LPVOID *ppobj )
116 if (IsEqualGUID(riid, &IID_IUnknown) ||
117 IsEqualGUID(riid, &IID_IBindStatusCallback))
119 IBindStatusCallback_AddRef( iface );
120 *ppobj = iface;
121 return S_OK;
124 FIXME("interface %s not implemented\n", debugstr_guid(riid));
125 return E_NOINTERFACE;
128 static ULONG WINAPI bsc_AddRef(
129 IBindStatusCallback *iface )
131 bsc_t *This = impl_from_IBindStatusCallback(iface);
132 LONG ref = InterlockedIncrement(&This->ref);
134 TRACE("(%p) ref=%d\n", This, ref);
136 return ref;
139 static ULONG WINAPI bsc_Release(
140 IBindStatusCallback *iface )
142 bsc_t *This = impl_from_IBindStatusCallback(iface);
143 LONG ref = InterlockedDecrement(&This->ref);
145 TRACE("(%p) ref=%d\n", This, ref);
147 if(!ref) {
148 if(This->binding)
149 IBinding_Release(This->binding);
150 if(This->doc && This->doc->bsc == This)
151 This->doc->bsc = NULL;
152 if(This->memstream)
153 IStream_Release(This->memstream);
154 HeapFree(GetProcessHeap(), 0, This);
157 return ref;
160 static HRESULT WINAPI bsc_OnStartBinding(
161 IBindStatusCallback* iface,
162 DWORD dwReserved,
163 IBinding* pib)
165 bsc_t *This = impl_from_IBindStatusCallback(iface);
166 HRESULT hr;
168 TRACE("(%p)->(%x %p)\n", This, dwReserved, pib);
170 This->binding = pib;
171 IBindStatusCallback_AddRef(pib);
173 hr = CreateStreamOnHGlobal(NULL, TRUE, &This->memstream);
174 if(FAILED(hr))
175 return hr;
177 return S_OK;
180 static HRESULT WINAPI bsc_GetPriority(
181 IBindStatusCallback* iface,
182 LONG* pnPriority)
184 return S_OK;
187 static HRESULT WINAPI bsc_OnLowResource(
188 IBindStatusCallback* iface,
189 DWORD reserved)
191 return S_OK;
194 static HRESULT WINAPI bsc_OnProgress(
195 IBindStatusCallback* iface,
196 ULONG ulProgress,
197 ULONG ulProgressMax,
198 ULONG ulStatusCode,
199 LPCWSTR szStatusText)
201 return S_OK;
204 static HRESULT WINAPI bsc_OnStopBinding(
205 IBindStatusCallback* iface,
206 HRESULT hresult,
207 LPCWSTR szError)
209 bsc_t *This = impl_from_IBindStatusCallback(iface);
210 HRESULT hr;
212 TRACE("(%p)->(%08x %s)\n", This, hresult, debugstr_w(szError));
214 if(This->binding) {
215 IBinding_Release(This->binding);
216 This->binding = NULL;
219 if(This->doc && SUCCEEDED(hresult)) {
220 HGLOBAL hglobal;
221 hr = GetHGlobalFromStream(This->memstream, &hglobal);
222 if(SUCCEEDED(hr))
224 DWORD len = GlobalSize(hglobal);
225 char *ptr = GlobalLock(hglobal);
226 xmlDocPtr xmldoc;
227 if(len != 0) {
228 xmldoc = doparse( ptr, len );
229 if(xmldoc) {
230 xmldoc->_private = 0;
231 attach_xmlnode(This->doc->node, (xmlNodePtr) xmldoc);
234 GlobalUnlock(hglobal);
238 return S_OK;
241 static HRESULT WINAPI bsc_GetBindInfo(
242 IBindStatusCallback* iface,
243 DWORD* grfBINDF,
244 BINDINFO* pbindinfo)
246 *grfBINDF = BINDF_GETNEWESTVERSION|BINDF_PULLDATA|BINDF_RESYNCHRONIZE|BINDF_PRAGMA_NO_CACHE;
248 return S_OK;
251 static HRESULT WINAPI bsc_OnDataAvailable(
252 IBindStatusCallback* iface,
253 DWORD grfBSCF,
254 DWORD dwSize,
255 FORMATETC* pformatetc,
256 STGMEDIUM* pstgmed)
258 bsc_t *This = impl_from_IBindStatusCallback(iface);
259 BYTE buf[4096];
260 DWORD read, written;
261 HRESULT hr;
263 TRACE("(%p)->(%x %d %p %p)\n", This, grfBSCF, dwSize, pformatetc, pstgmed);
267 hr = IStream_Read(pstgmed->pstm, buf, sizeof(buf), &read);
268 if(FAILED(hr))
269 break;
271 hr = IStream_Write(This->memstream, buf, read, &written);
272 } while(SUCCEEDED(hr) && written != 0 && read != 0);
274 return S_OK;
277 static HRESULT WINAPI bsc_OnObjectAvailable(
278 IBindStatusCallback* iface,
279 REFIID riid,
280 IUnknown* punk)
282 return S_OK;
285 static const struct IBindStatusCallbackVtbl bsc_vtbl =
287 bsc_QueryInterface,
288 bsc_AddRef,
289 bsc_Release,
290 bsc_OnStartBinding,
291 bsc_GetPriority,
292 bsc_OnLowResource,
293 bsc_OnProgress,
294 bsc_OnStopBinding,
295 bsc_GetBindInfo,
296 bsc_OnDataAvailable,
297 bsc_OnObjectAvailable
300 static bsc_t *create_bsc(domdoc *doc)
302 bsc_t *bsc = HeapAlloc(GetProcessHeap(), 0, sizeof(bsc));
304 bsc->lpVtbl = &bsc_vtbl;
305 bsc->ref = 1;
306 bsc->doc = doc;
307 bsc->binding = NULL;
309 return bsc;
312 LONG xmldoc_add_ref(xmlDocPtr doc)
314 LONG ref = InterlockedIncrement((LONG*)&doc->_private);
315 TRACE("%d\n", ref);
316 return ref;
319 LONG xmldoc_release(xmlDocPtr doc)
321 LONG ref = InterlockedDecrement((LONG*)&doc->_private);
322 TRACE("%d\n", ref);
323 if(ref == 0)
325 TRACE("freeing docptr %p\n", doc);
326 xmlFreeDoc(doc);
329 return ref;
332 static inline domdoc *impl_from_IXMLDOMDocument2( IXMLDOMDocument2 *iface )
334 return (domdoc *)((char*)iface - FIELD_OFFSET(domdoc, lpVtbl));
337 static inline xmlDocPtr get_doc( domdoc *This )
339 return (xmlDocPtr) xmlNodePtr_from_domnode( This->node, XML_DOCUMENT_NODE );
342 static inline domdoc *impl_from_IPersistStream(IPersistStream *iface)
344 return (domdoc *)((char*)iface - FIELD_OFFSET(domdoc, lpvtblIPersistStream));
347 static inline domdoc *impl_from_IObjectWithSite(IObjectWithSite *iface)
349 return (domdoc *)((char*)iface - FIELD_OFFSET(domdoc, lpvtblIObjectWithSite));
352 static inline domdoc *impl_from_IObjectSafety(IObjectSafety *iface)
354 return (domdoc *)((char*)iface - FIELD_OFFSET(domdoc, lpvtblIObjectSafety));
358 /************************************************************************
359 * xmldoc implementation of IPersistStream.
361 static HRESULT WINAPI xmldoc_IPersistStream_QueryInterface(
362 IPersistStream *iface, REFIID riid, LPVOID *ppvObj)
364 domdoc *this = impl_from_IPersistStream(iface);
365 return IXMLDocument_QueryInterface((IXMLDocument *)this, riid, ppvObj);
368 static ULONG WINAPI xmldoc_IPersistStream_AddRef(
369 IPersistStream *iface)
371 domdoc *this = impl_from_IPersistStream(iface);
372 return IXMLDocument_AddRef((IXMLDocument *)this);
375 static ULONG WINAPI xmldoc_IPersistStream_Release(
376 IPersistStream *iface)
378 domdoc *this = impl_from_IPersistStream(iface);
379 return IXMLDocument_Release((IXMLDocument *)this);
382 static HRESULT WINAPI xmldoc_IPersistStream_GetClassID(
383 IPersistStream *iface, CLSID *classid)
385 TRACE("(%p,%p): stub!\n", iface, classid);
387 if(!classid)
388 return E_POINTER;
390 *classid = CLSID_DOMDocument2;
392 return S_OK;
395 static HRESULT WINAPI xmldoc_IPersistStream_IsDirty(
396 IPersistStream *iface)
398 domdoc *This = impl_from_IPersistStream(iface);
400 FIXME("(%p->%p): stub!\n", iface, This);
402 return S_FALSE;
405 static HRESULT WINAPI xmldoc_IPersistStream_Load(
406 IPersistStream *iface, LPSTREAM pStm)
408 domdoc *This = impl_from_IPersistStream(iface);
409 HRESULT hr;
410 HGLOBAL hglobal;
411 DWORD read, written, len;
412 BYTE buf[4096];
413 char *ptr;
414 xmlDocPtr xmldoc = NULL;
416 TRACE("(%p, %p)\n", iface, pStm);
418 if (!pStm)
419 return E_INVALIDARG;
421 hr = CreateStreamOnHGlobal(NULL, TRUE, &This->stream);
422 if (FAILED(hr))
423 return hr;
427 IStream_Read(pStm, buf, sizeof(buf), &read);
428 hr = IStream_Write(This->stream, buf, read, &written);
429 } while(SUCCEEDED(hr) && written != 0 && read != 0);
431 if (FAILED(hr))
433 ERR("Failed to copy stream\n");
434 return hr;
437 hr = GetHGlobalFromStream(This->stream, &hglobal);
438 if (FAILED(hr))
439 return hr;
441 len = GlobalSize(hglobal);
442 ptr = GlobalLock(hglobal);
443 if (len != 0)
444 xmldoc = parse_xml(ptr, len);
445 GlobalUnlock(hglobal);
447 if (!xmldoc)
449 ERR("Failed to parse xml\n");
450 return E_FAIL;
453 attach_xmlnode( This->node, (xmlNodePtr)xmldoc );
455 return S_OK;
458 static HRESULT WINAPI xmldoc_IPersistStream_Save(
459 IPersistStream *iface, LPSTREAM pStm, BOOL fClearDirty)
461 FIXME("(%p, %p, %d): stub!\n", iface, pStm, fClearDirty);
462 return E_NOTIMPL;
465 static HRESULT WINAPI xmldoc_IPersistStream_GetSizeMax(
466 IPersistStream *iface, ULARGE_INTEGER *pcbSize)
468 TRACE("(%p, %p): stub!\n", iface, pcbSize);
469 return E_NOTIMPL;
472 static const IPersistStreamVtbl xmldoc_IPersistStream_VTable =
474 xmldoc_IPersistStream_QueryInterface,
475 xmldoc_IPersistStream_AddRef,
476 xmldoc_IPersistStream_Release,
477 xmldoc_IPersistStream_GetClassID,
478 xmldoc_IPersistStream_IsDirty,
479 xmldoc_IPersistStream_Load,
480 xmldoc_IPersistStream_Save,
481 xmldoc_IPersistStream_GetSizeMax,
484 static HRESULT WINAPI domdoc_QueryInterface( IXMLDOMDocument2 *iface, REFIID riid, void** ppvObject )
486 domdoc *This = impl_from_IXMLDOMDocument2( iface );
488 TRACE("%p %s %p\n", This, debugstr_guid( riid ), ppvObject );
490 *ppvObject = NULL;
492 if ( IsEqualGUID( riid, &IID_IUnknown ) ||
493 IsEqualGUID( riid, &IID_IDispatch ) ||
494 IsEqualGUID( riid, &IID_IXMLDOMDocument ) ||
495 IsEqualGUID( riid, &IID_IXMLDOMDocument2 ) )
497 *ppvObject = iface;
499 else if ( IsEqualGUID( riid, &IID_IXMLDOMNode ) )
501 return IUnknown_QueryInterface(This->node_unk, riid, ppvObject);
503 else if (IsEqualGUID(&IID_IPersistStream, riid))
505 *ppvObject = (IPersistStream*)&(This->lpvtblIPersistStream);
507 else if (IsEqualGUID(&IID_IObjectWithSite, riid))
509 *ppvObject = (IObjectWithSite*)&(This->lpvtblIObjectWithSite);
511 else if(IsEqualGUID(&IID_IRunnableObject, riid))
513 TRACE("IID_IRunnableObject not supported returning NULL\n");
514 return E_NOINTERFACE;
516 else
518 FIXME("interface %s not implemented\n", debugstr_guid(riid));
519 return E_NOINTERFACE;
522 IXMLDOMDocument_AddRef( iface );
524 return S_OK;
528 static ULONG WINAPI domdoc_AddRef(
529 IXMLDOMDocument2 *iface )
531 domdoc *This = impl_from_IXMLDOMDocument2( iface );
532 TRACE("%p\n", This );
533 return InterlockedIncrement( &This->ref );
537 static ULONG WINAPI domdoc_Release(
538 IXMLDOMDocument2 *iface )
540 domdoc *This = impl_from_IXMLDOMDocument2( iface );
541 LONG ref;
543 TRACE("%p\n", This );
545 ref = InterlockedDecrement( &This->ref );
546 if ( ref == 0 )
548 if(This->bsc) {
549 if(This->bsc->binding)
550 IBinding_Abort(This->bsc->binding);
551 This->bsc->doc = NULL;
552 IBindStatusCallback_Release((IBindStatusCallback*)&This->bsc->lpVtbl);
555 if (This->site)
556 IUnknown_Release( This->site );
557 IUnknown_Release( This->node_unk );
558 if(This->schema) IXMLDOMSchemaCollection_Release( This->schema );
559 if (This->stream) IStream_Release(This->stream);
560 HeapFree( GetProcessHeap(), 0, This );
563 return ref;
566 static HRESULT WINAPI domdoc_GetTypeInfoCount( IXMLDOMDocument2 *iface, UINT* pctinfo )
568 domdoc *This = impl_from_IXMLDOMDocument2( iface );
570 TRACE("(%p)->(%p)\n", This, pctinfo);
572 *pctinfo = 1;
574 return S_OK;
577 static HRESULT WINAPI domdoc_GetTypeInfo(
578 IXMLDOMDocument2 *iface,
579 UINT iTInfo, LCID lcid, ITypeInfo** ppTInfo )
581 domdoc *This = impl_from_IXMLDOMDocument2( iface );
582 HRESULT hr;
584 TRACE("(%p)->(%u %u %p)\n", This, iTInfo, lcid, ppTInfo);
586 hr = get_typeinfo(IXMLDOMDocument2_tid, ppTInfo);
588 return hr;
591 static HRESULT WINAPI domdoc_GetIDsOfNames(
592 IXMLDOMDocument2 *iface,
593 REFIID riid,
594 LPOLESTR* rgszNames,
595 UINT cNames,
596 LCID lcid,
597 DISPID* rgDispId)
599 domdoc *This = impl_from_IXMLDOMDocument2( iface );
600 ITypeInfo *typeinfo;
601 HRESULT hr;
603 TRACE("(%p)->(%s %p %u %u %p)\n", This, debugstr_guid(riid), rgszNames, cNames,
604 lcid, rgDispId);
606 if(!rgszNames || cNames == 0 || !rgDispId)
607 return E_INVALIDARG;
609 hr = get_typeinfo(IXMLDOMDocument2_tid, &typeinfo);
610 if(SUCCEEDED(hr))
612 hr = ITypeInfo_GetIDsOfNames(typeinfo, rgszNames, cNames, rgDispId);
613 ITypeInfo_Release(typeinfo);
616 return hr;
620 static HRESULT WINAPI domdoc_Invoke(
621 IXMLDOMDocument2 *iface,
622 DISPID dispIdMember,
623 REFIID riid,
624 LCID lcid,
625 WORD wFlags,
626 DISPPARAMS* pDispParams,
627 VARIANT* pVarResult,
628 EXCEPINFO* pExcepInfo,
629 UINT* puArgErr)
631 domdoc *This = impl_from_IXMLDOMDocument2( iface );
632 ITypeInfo *typeinfo;
633 HRESULT hr;
635 TRACE("(%p)->(%d %s %d %d %p %p %p %p)\n", This, dispIdMember, debugstr_guid(riid),
636 lcid, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr);
638 hr = get_typeinfo(IXMLDOMDocument2_tid, &typeinfo);
639 if(SUCCEEDED(hr))
641 hr = ITypeInfo_Invoke(typeinfo, &(This->lpVtbl), dispIdMember, wFlags, pDispParams,
642 pVarResult, pExcepInfo, puArgErr);
643 ITypeInfo_Release(typeinfo);
646 return hr;
650 static HRESULT WINAPI domdoc_get_nodeName(
651 IXMLDOMDocument2 *iface,
652 BSTR* name )
654 domdoc *This = impl_from_IXMLDOMDocument2( iface );
655 return IXMLDOMNode_get_nodeName( This->node, name );
659 static HRESULT WINAPI domdoc_get_nodeValue(
660 IXMLDOMDocument2 *iface,
661 VARIANT* value )
663 domdoc *This = impl_from_IXMLDOMDocument2( iface );
664 return IXMLDOMNode_get_nodeValue( This->node, value );
668 static HRESULT WINAPI domdoc_put_nodeValue(
669 IXMLDOMDocument2 *iface,
670 VARIANT value)
672 domdoc *This = impl_from_IXMLDOMDocument2( iface );
673 return IXMLDOMNode_put_nodeValue( This->node, value );
677 static HRESULT WINAPI domdoc_get_nodeType(
678 IXMLDOMDocument2 *iface,
679 DOMNodeType* type )
681 domdoc *This = impl_from_IXMLDOMDocument2( iface );
682 return IXMLDOMNode_get_nodeType( This->node, type );
686 static HRESULT WINAPI domdoc_get_parentNode(
687 IXMLDOMDocument2 *iface,
688 IXMLDOMNode** parent )
690 domdoc *This = impl_from_IXMLDOMDocument2( iface );
691 return IXMLDOMNode_get_parentNode( This->node, parent );
695 static HRESULT WINAPI domdoc_get_childNodes(
696 IXMLDOMDocument2 *iface,
697 IXMLDOMNodeList** childList )
699 domdoc *This = impl_from_IXMLDOMDocument2( iface );
700 return IXMLDOMNode_get_childNodes( This->node, childList );
704 static HRESULT WINAPI domdoc_get_firstChild(
705 IXMLDOMDocument2 *iface,
706 IXMLDOMNode** firstChild )
708 domdoc *This = impl_from_IXMLDOMDocument2( iface );
709 return IXMLDOMNode_get_firstChild( This->node, firstChild );
713 static HRESULT WINAPI domdoc_get_lastChild(
714 IXMLDOMDocument2 *iface,
715 IXMLDOMNode** lastChild )
717 domdoc *This = impl_from_IXMLDOMDocument2( iface );
718 return IXMLDOMNode_get_lastChild( This->node, lastChild );
722 static HRESULT WINAPI domdoc_get_previousSibling(
723 IXMLDOMDocument2 *iface,
724 IXMLDOMNode** previousSibling )
726 domdoc *This = impl_from_IXMLDOMDocument2( iface );
727 return IXMLDOMNode_get_previousSibling( This->node, previousSibling );
731 static HRESULT WINAPI domdoc_get_nextSibling(
732 IXMLDOMDocument2 *iface,
733 IXMLDOMNode** nextSibling )
735 domdoc *This = impl_from_IXMLDOMDocument2( iface );
736 return IXMLDOMNode_get_nextSibling( This->node, nextSibling );
740 static HRESULT WINAPI domdoc_get_attributes(
741 IXMLDOMDocument2 *iface,
742 IXMLDOMNamedNodeMap** attributeMap )
744 domdoc *This = impl_from_IXMLDOMDocument2( iface );
745 return IXMLDOMNode_get_attributes( This->node, attributeMap );
749 static HRESULT WINAPI domdoc_insertBefore(
750 IXMLDOMDocument2 *iface,
751 IXMLDOMNode* newChild,
752 VARIANT refChild,
753 IXMLDOMNode** outNewChild )
755 domdoc *This = impl_from_IXMLDOMDocument2( iface );
756 return IXMLDOMNode_insertBefore( This->node, newChild, refChild, outNewChild );
760 static HRESULT WINAPI domdoc_replaceChild(
761 IXMLDOMDocument2 *iface,
762 IXMLDOMNode* newChild,
763 IXMLDOMNode* oldChild,
764 IXMLDOMNode** outOldChild)
766 domdoc *This = impl_from_IXMLDOMDocument2( iface );
767 return IXMLDOMNode_replaceChild( This->node, newChild, oldChild, outOldChild );
771 static HRESULT WINAPI domdoc_removeChild(
772 IXMLDOMDocument2 *iface,
773 IXMLDOMNode* childNode,
774 IXMLDOMNode** oldChild)
776 domdoc *This = impl_from_IXMLDOMDocument2( iface );
777 return IXMLDOMNode_removeChild( This->node, childNode, oldChild );
781 static HRESULT WINAPI domdoc_appendChild(
782 IXMLDOMDocument2 *iface,
783 IXMLDOMNode* newChild,
784 IXMLDOMNode** outNewChild)
786 domdoc *This = impl_from_IXMLDOMDocument2( iface );
787 return IXMLDOMNode_appendChild( This->node, newChild, outNewChild );
791 static HRESULT WINAPI domdoc_hasChildNodes(
792 IXMLDOMDocument2 *iface,
793 VARIANT_BOOL* hasChild)
795 domdoc *This = impl_from_IXMLDOMDocument2( iface );
796 return IXMLDOMNode_hasChildNodes( This->node, hasChild );
800 static HRESULT WINAPI domdoc_get_ownerDocument(
801 IXMLDOMDocument2 *iface,
802 IXMLDOMDocument** DOMDocument)
804 domdoc *This = impl_from_IXMLDOMDocument2( iface );
805 return IXMLDOMNode_get_ownerDocument( This->node, DOMDocument );
809 static HRESULT WINAPI domdoc_cloneNode(
810 IXMLDOMDocument2 *iface,
811 VARIANT_BOOL deep,
812 IXMLDOMNode** cloneRoot)
814 domdoc *This = impl_from_IXMLDOMDocument2( iface );
815 return IXMLDOMNode_cloneNode( This->node, deep, cloneRoot );
819 static HRESULT WINAPI domdoc_get_nodeTypeString(
820 IXMLDOMDocument2 *iface,
821 BSTR* nodeType )
823 domdoc *This = impl_from_IXMLDOMDocument2( iface );
824 return IXMLDOMNode_get_nodeTypeString( This->node, nodeType );
828 static HRESULT WINAPI domdoc_get_text(
829 IXMLDOMDocument2 *iface,
830 BSTR* text )
832 domdoc *This = impl_from_IXMLDOMDocument2( iface );
833 return IXMLDOMNode_get_text( This->node, text );
837 static HRESULT WINAPI domdoc_put_text(
838 IXMLDOMDocument2 *iface,
839 BSTR text )
841 domdoc *This = impl_from_IXMLDOMDocument2( iface );
842 return IXMLDOMNode_put_text( This->node, text );
846 static HRESULT WINAPI domdoc_get_specified(
847 IXMLDOMDocument2 *iface,
848 VARIANT_BOOL* isSpecified )
850 domdoc *This = impl_from_IXMLDOMDocument2( iface );
851 return IXMLDOMNode_get_specified( This->node, isSpecified );
855 static HRESULT WINAPI domdoc_get_definition(
856 IXMLDOMDocument2 *iface,
857 IXMLDOMNode** definitionNode )
859 domdoc *This = impl_from_IXMLDOMDocument2( iface );
860 return IXMLDOMNode_get_definition( This->node, definitionNode );
864 static HRESULT WINAPI domdoc_get_nodeTypedValue(
865 IXMLDOMDocument2 *iface,
866 VARIANT* typedValue )
868 domdoc *This = impl_from_IXMLDOMDocument2( iface );
869 return IXMLDOMNode_get_nodeTypedValue( This->node, typedValue );
872 static HRESULT WINAPI domdoc_put_nodeTypedValue(
873 IXMLDOMDocument2 *iface,
874 VARIANT typedValue )
876 domdoc *This = impl_from_IXMLDOMDocument2( iface );
877 return IXMLDOMNode_put_nodeTypedValue( This->node, typedValue );
881 static HRESULT WINAPI domdoc_get_dataType(
882 IXMLDOMDocument2 *iface,
883 VARIANT* dataTypeName )
885 domdoc *This = impl_from_IXMLDOMDocument2( iface );
886 return IXMLDOMNode_get_dataType( This->node, dataTypeName );
890 static HRESULT WINAPI domdoc_put_dataType(
891 IXMLDOMDocument2 *iface,
892 BSTR dataTypeName )
894 domdoc *This = impl_from_IXMLDOMDocument2( iface );
895 return IXMLDOMNode_put_dataType( This->node, dataTypeName );
899 static HRESULT WINAPI domdoc_get_xml(
900 IXMLDOMDocument2 *iface,
901 BSTR* xmlString )
903 domdoc *This = impl_from_IXMLDOMDocument2( iface );
904 return IXMLDOMNode_get_xml( This->node, xmlString );
908 static HRESULT WINAPI domdoc_transformNode(
909 IXMLDOMDocument2 *iface,
910 IXMLDOMNode* styleSheet,
911 BSTR* xmlString )
913 domdoc *This = impl_from_IXMLDOMDocument2( iface );
914 return IXMLDOMNode_transformNode( This->node, styleSheet, xmlString );
918 static HRESULT WINAPI domdoc_selectNodes(
919 IXMLDOMDocument2 *iface,
920 BSTR queryString,
921 IXMLDOMNodeList** resultList )
923 domdoc *This = impl_from_IXMLDOMDocument2( iface );
924 return IXMLDOMNode_selectNodes( This->node, queryString, resultList );
928 static HRESULT WINAPI domdoc_selectSingleNode(
929 IXMLDOMDocument2 *iface,
930 BSTR queryString,
931 IXMLDOMNode** resultNode )
933 domdoc *This = impl_from_IXMLDOMDocument2( iface );
934 return IXMLDOMNode_selectSingleNode( This->node, queryString, resultNode );
938 static HRESULT WINAPI domdoc_get_parsed(
939 IXMLDOMDocument2 *iface,
940 VARIANT_BOOL* isParsed )
942 domdoc *This = impl_from_IXMLDOMDocument2( iface );
943 return IXMLDOMNode_get_parsed( This->node, isParsed );
947 static HRESULT WINAPI domdoc_get_namespaceURI(
948 IXMLDOMDocument2 *iface,
949 BSTR* namespaceURI )
951 domdoc *This = impl_from_IXMLDOMDocument2( iface );
952 return IXMLDOMNode_get_namespaceURI( This->node, namespaceURI );
956 static HRESULT WINAPI domdoc_get_prefix(
957 IXMLDOMDocument2 *iface,
958 BSTR* prefixString )
960 domdoc *This = impl_from_IXMLDOMDocument2( iface );
961 return IXMLDOMNode_get_prefix( This->node, prefixString );
965 static HRESULT WINAPI domdoc_get_baseName(
966 IXMLDOMDocument2 *iface,
967 BSTR* nameString )
969 domdoc *This = impl_from_IXMLDOMDocument2( iface );
970 return IXMLDOMNode_get_baseName( This->node, nameString );
974 static HRESULT WINAPI domdoc_transformNodeToObject(
975 IXMLDOMDocument2 *iface,
976 IXMLDOMNode* stylesheet,
977 VARIANT outputObject)
979 domdoc *This = impl_from_IXMLDOMDocument2( iface );
980 return IXMLDOMNode_transformNodeToObject( This->node, stylesheet, outputObject );
984 static HRESULT WINAPI domdoc_get_doctype(
985 IXMLDOMDocument2 *iface,
986 IXMLDOMDocumentType** documentType )
988 FIXME("\n");
989 return E_NOTIMPL;
993 static HRESULT WINAPI domdoc_get_implementation(
994 IXMLDOMDocument2 *iface,
995 IXMLDOMImplementation** impl )
997 if(!impl)
998 return E_INVALIDARG;
1000 *impl = (IXMLDOMImplementation*)create_doc_Implementation();
1002 return S_OK;
1005 static HRESULT WINAPI domdoc_get_documentElement(
1006 IXMLDOMDocument2 *iface,
1007 IXMLDOMElement** DOMElement )
1009 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1010 xmlDocPtr xmldoc = NULL;
1011 xmlNodePtr root = NULL;
1012 IXMLDOMNode *element_node;
1013 HRESULT hr;
1015 TRACE("%p %p\n", This, This->node);
1017 if(!DOMElement)
1018 return E_INVALIDARG;
1020 *DOMElement = NULL;
1022 xmldoc = get_doc( This );
1024 root = xmlDocGetRootElement( xmldoc );
1025 if ( !root )
1026 return S_FALSE;
1028 element_node = create_node( root );
1029 if(!element_node) return S_FALSE;
1031 hr = IXMLDOMNode_QueryInterface(element_node, &IID_IXMLDOMElement, (LPVOID*)DOMElement);
1032 IXMLDOMNode_Release(element_node);
1034 return hr;
1038 static HRESULT WINAPI domdoc_put_documentElement(
1039 IXMLDOMDocument2 *iface,
1040 IXMLDOMElement* DOMElement )
1042 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1043 IXMLDOMNode *elementNode;
1044 xmlnode *xmlNode;
1045 HRESULT hr;
1047 TRACE("(%p)->(%p)\n", This, DOMElement);
1049 hr = IXMLDOMElement_QueryInterface( DOMElement, &IID_IXMLDOMNode, (void**)&elementNode );
1050 if(FAILED(hr))
1051 return hr;
1053 xmlNode = impl_from_IXMLDOMNode( elementNode );
1054 xmlDocSetRootElement( get_doc(This), xmlNode->node);
1055 IXMLDOMNode_Release( elementNode );
1057 return S_OK;
1061 static HRESULT WINAPI domdoc_createElement(
1062 IXMLDOMDocument2 *iface,
1063 BSTR tagname,
1064 IXMLDOMElement** element )
1066 xmlNodePtr xmlnode;
1067 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1068 xmlChar *xml_name;
1069 IUnknown *elem_unk;
1070 HRESULT hr;
1072 TRACE("%p->(%s,%p)\n", iface, debugstr_w(tagname), element);
1074 xml_name = xmlChar_from_wchar((WCHAR*)tagname);
1075 xmlnode = xmlNewDocNode(get_doc(This), NULL, xml_name, NULL);
1077 TRACE("created xmlptr %p\n", xmlnode);
1078 elem_unk = create_element(xmlnode, NULL);
1079 HeapFree(GetProcessHeap(), 0, xml_name);
1081 hr = IUnknown_QueryInterface(elem_unk, &IID_IXMLDOMElement, (void **)element);
1082 IUnknown_Release(elem_unk);
1083 TRACE("returning %p\n", *element);
1084 return hr;
1088 static HRESULT WINAPI domdoc_createDocumentFragment(
1089 IXMLDOMDocument2 *iface,
1090 IXMLDOMDocumentFragment** docFrag )
1092 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1093 xmlNodePtr xmlnode;
1095 TRACE("%p\n", iface);
1097 if(!docFrag)
1098 return E_INVALIDARG;
1100 *docFrag = NULL;
1102 xmlnode = xmlNewDocFragment(get_doc( This ) );
1104 if(!xmlnode)
1105 return E_FAIL;
1107 xmlnode->doc = get_doc( This );
1109 *docFrag = (IXMLDOMDocumentFragment*)create_doc_fragment(xmlnode);
1111 return S_OK;
1115 static HRESULT WINAPI domdoc_createTextNode(
1116 IXMLDOMDocument2 *iface,
1117 BSTR data,
1118 IXMLDOMText** text )
1120 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1121 xmlNodePtr xmlnode;
1122 xmlChar *xml_content;
1124 TRACE("%p->(%s %p)\n", iface, debugstr_w(data), text);
1126 if(!text)
1127 return E_INVALIDARG;
1129 *text = NULL;
1131 xml_content = xmlChar_from_wchar((WCHAR*)data);
1132 xmlnode = xmlNewText(xml_content);
1133 HeapFree(GetProcessHeap(), 0, xml_content);
1135 if(!xmlnode)
1136 return E_FAIL;
1138 xmlnode->doc = get_doc( This );
1140 *text = (IXMLDOMText*)create_text(xmlnode);
1142 return S_OK;
1146 static HRESULT WINAPI domdoc_createComment(
1147 IXMLDOMDocument2 *iface,
1148 BSTR data,
1149 IXMLDOMComment** comment )
1151 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1152 xmlNodePtr xmlnode;
1153 xmlChar *xml_content;
1155 TRACE("%p->(%s %p)\n", iface, debugstr_w(data), comment);
1157 if(!comment)
1158 return E_INVALIDARG;
1160 *comment = NULL;
1162 xml_content = xmlChar_from_wchar((WCHAR*)data);
1163 xmlnode = xmlNewComment(xml_content);
1164 HeapFree(GetProcessHeap(), 0, xml_content);
1166 if(!xmlnode)
1167 return E_FAIL;
1169 xmlnode->doc = get_doc( This );
1171 *comment = (IXMLDOMComment*)create_comment(xmlnode);
1173 return S_OK;
1177 static HRESULT WINAPI domdoc_createCDATASection(
1178 IXMLDOMDocument2 *iface,
1179 BSTR data,
1180 IXMLDOMCDATASection** cdata )
1182 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1183 xmlNodePtr xmlnode;
1184 xmlChar *xml_content;
1186 TRACE("%p->(%s %p)\n", iface, debugstr_w(data), cdata);
1188 if(!cdata)
1189 return E_INVALIDARG;
1191 *cdata = NULL;
1193 xml_content = xmlChar_from_wchar((WCHAR*)data);
1194 xmlnode = xmlNewCDataBlock(get_doc( This ), xml_content, strlen( (char*)xml_content) );
1195 HeapFree(GetProcessHeap(), 0, xml_content);
1197 if(!xmlnode)
1198 return E_FAIL;
1200 xmlnode->doc = get_doc( This );
1202 *cdata = (IXMLDOMCDATASection*)create_cdata(xmlnode);
1204 return S_OK;
1208 static HRESULT WINAPI domdoc_createProcessingInstruction(
1209 IXMLDOMDocument2 *iface,
1210 BSTR target,
1211 BSTR data,
1212 IXMLDOMProcessingInstruction** pi )
1214 #ifdef HAVE_XMLNEWDOCPI
1215 xmlNodePtr xmlnode;
1216 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1217 xmlChar *xml_target, *xml_content;
1219 TRACE("%p->(%s %s %p)\n", iface, debugstr_w(target), debugstr_w(data), pi);
1221 if(!pi)
1222 return E_INVALIDARG;
1224 if(!target || lstrlenW(target) == 0)
1225 return E_FAIL;
1227 xml_target = xmlChar_from_wchar((WCHAR*)target);
1228 xml_content = xmlChar_from_wchar((WCHAR*)data);
1230 xmlnode = xmlNewDocPI(get_doc(This), xml_target, xml_content);
1231 TRACE("created xmlptr %p\n", xmlnode);
1232 *pi = (IXMLDOMProcessingInstruction*)create_pi(xmlnode);
1234 HeapFree(GetProcessHeap(), 0, xml_content);
1235 HeapFree(GetProcessHeap(), 0, xml_target);
1237 return S_OK;
1238 #else
1239 FIXME("Libxml 2.6.15 or greater required.\n");
1240 return E_NOTIMPL;
1241 #endif
1245 static HRESULT WINAPI domdoc_createAttribute(
1246 IXMLDOMDocument2 *iface,
1247 BSTR name,
1248 IXMLDOMAttribute** attribute )
1250 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1251 xmlNodePtr xmlnode;
1252 xmlChar *xml_name;
1254 TRACE("%p->(%s %p)\n", iface, debugstr_w(name), attribute);
1256 if(!attribute)
1257 return E_INVALIDARG;
1259 *attribute = NULL;
1261 xml_name = xmlChar_from_wchar((WCHAR*)name);
1262 xmlnode = (xmlNode *)xmlNewProp(NULL, xml_name, NULL);
1263 HeapFree(GetProcessHeap(), 0, xml_name);
1265 if(!xmlnode)
1266 return E_FAIL;
1268 xmlnode->doc = get_doc( This );
1270 *attribute = (IXMLDOMAttribute*)create_attribute(xmlnode);
1272 return S_OK;
1276 static HRESULT WINAPI domdoc_createEntityReference(
1277 IXMLDOMDocument2 *iface,
1278 BSTR name,
1279 IXMLDOMEntityReference** entityRef )
1281 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1282 xmlNodePtr xmlnode;
1283 xmlChar *xml_name;
1285 TRACE("%p\n", iface);
1287 if(!entityRef)
1288 return E_INVALIDARG;
1290 *entityRef = NULL;
1292 xml_name = xmlChar_from_wchar((WCHAR*)name);
1293 xmlnode = xmlNewReference(get_doc( This ), xml_name );
1294 HeapFree(GetProcessHeap(), 0, xml_name);
1296 if(!xmlnode)
1297 return E_FAIL;
1299 xmlnode->doc = get_doc( This );
1301 *entityRef = (IXMLDOMEntityReference*)create_doc_entity_ref(xmlnode);
1303 return S_OK;
1307 static HRESULT WINAPI domdoc_getElementsByTagName(
1308 IXMLDOMDocument2 *iface,
1309 BSTR tagName,
1310 IXMLDOMNodeList** resultList )
1312 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1313 LPWSTR szPattern;
1314 HRESULT hr;
1315 TRACE("(%p)->(%s, %p)\n", This, debugstr_w(tagName), resultList);
1317 szPattern = HeapAlloc(GetProcessHeap(), 0, sizeof(WCHAR)*(2+lstrlenW(tagName)+1));
1318 szPattern[0] = szPattern[1] = '/';
1319 lstrcpyW(szPattern + 2, tagName);
1321 hr = queryresult_create((xmlNodePtr)get_doc(This), szPattern, resultList);
1322 HeapFree(GetProcessHeap(), 0, szPattern);
1324 return hr;
1327 static DOMNodeType get_node_type(VARIANT Type)
1329 if(V_VT(&Type) == VT_I4)
1330 return V_I4(&Type);
1332 FIXME("Unsupported variant type %x\n", V_VT(&Type));
1333 return 0;
1336 static HRESULT WINAPI domdoc_createNode(
1337 IXMLDOMDocument2 *iface,
1338 VARIANT Type,
1339 BSTR name,
1340 BSTR namespaceURI,
1341 IXMLDOMNode** node )
1343 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1344 DOMNodeType node_type;
1345 xmlNodePtr xmlnode = NULL;
1346 xmlChar *xml_name;
1348 TRACE("(%p)->(type,%s,%s,%p)\n", This, debugstr_w(name), debugstr_w(namespaceURI), node);
1350 node_type = get_node_type(Type);
1351 TRACE("node_type %d\n", node_type);
1353 xml_name = xmlChar_from_wchar((WCHAR*)name);
1355 switch(node_type)
1357 case NODE_ELEMENT:
1358 xmlnode = xmlNewDocNode(get_doc(This), NULL, xml_name, NULL);
1359 *node = create_node(xmlnode);
1360 TRACE("created %p\n", xmlnode);
1361 break;
1362 case NODE_ATTRIBUTE:
1363 xmlnode = (xmlNode *)xmlNewProp(NULL, xml_name, NULL);
1364 if(xmlnode)
1366 xmlnode->doc = get_doc( This );
1368 *node = (IXMLDOMNode*)create_attribute(xmlnode);
1371 TRACE("created %p\n", xmlnode);
1372 break;
1374 default:
1375 FIXME("unhandled node type %d\n", node_type);
1376 break;
1379 HeapFree(GetProcessHeap(), 0, xml_name);
1381 if(xmlnode && *node)
1382 return S_OK;
1384 return E_FAIL;
1387 static HRESULT WINAPI domdoc_nodeFromID(
1388 IXMLDOMDocument2 *iface,
1389 BSTR idString,
1390 IXMLDOMNode** node )
1392 FIXME("\n");
1393 return E_NOTIMPL;
1396 static HRESULT doread( domdoc *This, LPWSTR filename )
1398 HRESULT hr;
1399 IBindCtx *pbc;
1400 WCHAR url[INTERNET_MAX_URL_LENGTH];
1402 TRACE("%s\n", debugstr_w( filename ));
1404 if(!PathIsURLW(filename))
1406 WCHAR fullpath[MAX_PATH];
1407 DWORD needed = sizeof(url)/sizeof(WCHAR);
1409 if(!PathSearchAndQualifyW(filename, fullpath, sizeof(fullpath)/sizeof(WCHAR)))
1411 WARN("can't find path\n");
1412 return E_FAIL;
1415 if(FAILED(UrlCreateFromPathW(fullpath, url, &needed, 0)))
1417 ERR("can't create url from path\n");
1418 return E_FAIL;
1420 filename = url;
1423 hr = CreateBindCtx(0, &pbc);
1424 if(SUCCEEDED(hr))
1426 if(This->bsc) {
1427 if(This->bsc->binding)
1428 IBinding_Abort(This->bsc->binding);
1429 This->bsc->doc = NULL;
1430 IBindStatusCallback_Release((IBindStatusCallback*)&This->bsc->lpVtbl);
1433 This->bsc = create_bsc(This);
1435 hr = RegisterBindStatusCallback(pbc, (IBindStatusCallback*)&This->bsc->lpVtbl, NULL, 0);
1436 if(SUCCEEDED(hr))
1438 IMoniker *moniker;
1439 hr = CreateURLMoniker(NULL, filename, &moniker);
1440 if(SUCCEEDED(hr))
1442 IStream *stream;
1443 hr = IMoniker_BindToStorage(moniker, pbc, NULL, &IID_IStream, (LPVOID*)&stream);
1444 IMoniker_Release(moniker);
1445 if(stream)
1446 IStream_Release(stream);
1449 IBindCtx_Release(pbc);
1452 return hr;
1455 static HRESULT WINAPI domdoc_load(
1456 IXMLDOMDocument2 *iface,
1457 VARIANT xmlSource,
1458 VARIANT_BOOL* isSuccessful )
1460 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1461 LPWSTR filename = NULL;
1462 HRESULT hr = S_FALSE;
1463 IXMLDOMDocument2 *pNewDoc = NULL;
1464 IStream *pStream = NULL;
1465 xmlDocPtr xmldoc;
1467 TRACE("type %d\n", V_VT(&xmlSource) );
1469 *isSuccessful = VARIANT_FALSE;
1471 assert( This->node );
1473 attach_xmlnode(This->node, NULL);
1475 switch( V_VT(&xmlSource) )
1477 case VT_BSTR:
1478 filename = V_BSTR(&xmlSource);
1479 break;
1480 case VT_UNKNOWN:
1481 hr = IUnknown_QueryInterface(V_UNKNOWN(&xmlSource), &IID_IXMLDOMDocument2, (void**)&pNewDoc);
1482 if(hr == S_OK)
1484 if(pNewDoc)
1486 domdoc *newDoc = impl_from_IXMLDOMDocument2( pNewDoc );
1487 xmldoc = xmlCopyDoc(get_doc(newDoc), 1);
1488 attach_xmlnode(This->node, (xmlNodePtr) xmldoc);
1490 *isSuccessful = VARIANT_TRUE;
1492 return S_OK;
1495 hr = IUnknown_QueryInterface(V_UNKNOWN(&xmlSource), &IID_IStream, (void**)&pStream);
1496 if(hr == S_OK)
1498 IPersistStream *pDocStream;
1499 hr = IUnknown_QueryInterface(iface, &IID_IPersistStream, (void**)&pDocStream);
1500 if(hr == S_OK)
1502 hr = xmldoc_IPersistStream_Load(pDocStream, pStream);
1503 IStream_Release(pStream);
1504 if(hr == S_OK)
1506 *isSuccessful = VARIANT_TRUE;
1508 TRACE("Using ID_IStream to load Document\n");
1509 return S_OK;
1511 else
1513 ERR("xmldoc_IPersistStream_Load failed (%d)\n", hr);
1516 else
1518 ERR("QueryInterface IID_IPersistStream failed (%d)\n", hr);
1521 else
1523 /* ISequentialStream */
1524 FIXME("Unknown type not supported (%d) (%p)(%p)\n", hr, pNewDoc, V_UNKNOWN(&xmlSource)->lpVtbl);
1526 break;
1527 default:
1528 FIXME("VT type not supported (%d)\n", V_VT(&xmlSource));
1531 TRACE("filename (%s)\n", debugstr_w(filename));
1533 if ( filename )
1535 hr = doread( This, filename );
1537 if ( FAILED(hr) )
1538 This->error = E_FAIL;
1539 else
1541 hr = This->error = S_OK;
1542 *isSuccessful = VARIANT_TRUE;
1546 if(!filename || FAILED(hr)) {
1547 xmldoc = xmlNewDoc(NULL);
1548 xmldoc->_private = 0;
1549 attach_xmlnode(This->node, (xmlNodePtr) xmldoc);
1550 hr = S_FALSE;
1553 TRACE("ret (%d)\n", hr);
1555 return hr;
1559 static HRESULT WINAPI domdoc_get_readyState(
1560 IXMLDOMDocument2 *iface,
1561 long* value )
1563 FIXME("\n");
1564 return E_NOTIMPL;
1568 static HRESULT WINAPI domdoc_get_parseError(
1569 IXMLDOMDocument2 *iface,
1570 IXMLDOMParseError** errorObj )
1572 BSTR error_string = NULL;
1573 static const WCHAR err[] = {'e','r','r','o','r',0};
1574 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1576 FIXME("(%p)->(%p): creating a dummy parseError\n", iface, errorObj);
1578 if(This->error)
1579 error_string = SysAllocString(err);
1581 *errorObj = create_parseError(This->error, NULL, error_string, NULL, 0, 0, 0);
1582 if(!*errorObj) return E_OUTOFMEMORY;
1583 return S_OK;
1587 static HRESULT WINAPI domdoc_get_url(
1588 IXMLDOMDocument2 *iface,
1589 BSTR* urlString )
1591 FIXME("\n");
1592 return E_NOTIMPL;
1596 static HRESULT WINAPI domdoc_get_async(
1597 IXMLDOMDocument2 *iface,
1598 VARIANT_BOOL* isAsync )
1600 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1602 TRACE("%p <- %d\n", isAsync, This->async);
1603 *isAsync = This->async;
1604 return S_OK;
1608 static HRESULT WINAPI domdoc_put_async(
1609 IXMLDOMDocument2 *iface,
1610 VARIANT_BOOL isAsync )
1612 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1614 TRACE("%d\n", isAsync);
1615 This->async = isAsync;
1616 return S_OK;
1620 static HRESULT WINAPI domdoc_abort(
1621 IXMLDOMDocument2 *iface )
1623 FIXME("\n");
1624 return E_NOTIMPL;
1628 static BOOL bstr_to_utf8( BSTR bstr, char **pstr, int *plen )
1630 UINT len, blen = SysStringLen( bstr );
1631 LPSTR str;
1633 len = WideCharToMultiByte( CP_UTF8, 0, bstr, blen, NULL, 0, NULL, NULL );
1634 str = HeapAlloc( GetProcessHeap(), 0, len );
1635 if ( !str )
1636 return FALSE;
1637 WideCharToMultiByte( CP_UTF8, 0, bstr, blen, str, len, NULL, NULL );
1638 *plen = len;
1639 *pstr = str;
1640 return TRUE;
1643 static HRESULT WINAPI domdoc_loadXML(
1644 IXMLDOMDocument2 *iface,
1645 BSTR bstrXML,
1646 VARIANT_BOOL* isSuccessful )
1648 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1649 xmlDocPtr xmldoc = NULL;
1650 char *str;
1651 int len;
1652 HRESULT hr = S_FALSE;
1654 TRACE("%p %s %p\n", This, debugstr_w( bstrXML ), isSuccessful );
1656 assert ( This->node );
1658 attach_xmlnode( This->node, NULL );
1660 if ( isSuccessful )
1662 *isSuccessful = VARIANT_FALSE;
1664 if ( bstrXML && bstr_to_utf8( bstrXML, &str, &len ) )
1666 xmldoc = doparse( str, len );
1667 HeapFree( GetProcessHeap(), 0, str );
1668 if ( !xmldoc )
1669 This->error = E_FAIL;
1670 else
1672 hr = This->error = S_OK;
1673 *isSuccessful = VARIANT_TRUE;
1677 if(!xmldoc)
1678 xmldoc = xmlNewDoc(NULL);
1680 xmldoc->_private = 0;
1681 attach_xmlnode( This->node, (xmlNodePtr) xmldoc );
1683 return hr;
1687 static HRESULT WINAPI domdoc_save(
1688 IXMLDOMDocument2 *iface,
1689 VARIANT destination )
1691 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1692 HANDLE handle;
1693 xmlChar *mem, *p;
1694 int size;
1695 HRESULT ret = S_OK;
1696 DWORD written;
1698 TRACE("(%p)->(var(vt %x, %s))\n", This, V_VT(&destination),
1699 V_VT(&destination) == VT_BSTR ? debugstr_w(V_BSTR(&destination)) : NULL);
1701 if(V_VT(&destination) != VT_BSTR && V_VT(&destination) != VT_UNKNOWN)
1703 FIXME("Unhandled vt %d\n", V_VT(&destination));
1704 return S_FALSE;
1707 if(V_VT(&destination) == VT_UNKNOWN)
1709 IUnknown *pUnk = V_UNKNOWN(&destination);
1710 IXMLDOMDocument *pDocument;
1712 ret = IXMLDOMDocument_QueryInterface(pUnk, &IID_IXMLDOMDocument2, (void**)&pDocument);
1713 if(ret == S_OK)
1715 BSTR bXML;
1716 VARIANT_BOOL bSuccessful;
1718 ret = IXMLDOMDocument_get_xml(iface, &bXML);
1719 if(ret == S_OK)
1721 ret = IXMLDOMDocument_loadXML(pDocument, bXML, &bSuccessful);
1723 SysFreeString(bXML);
1726 IXMLDOMDocument_Release(pDocument);
1729 return ret;
1732 handle = CreateFileW( V_BSTR(&destination), GENERIC_WRITE, 0,
1733 NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL );
1734 if( handle == INVALID_HANDLE_VALUE )
1736 WARN("failed to create file\n");
1737 return S_FALSE;
1740 xmlDocDumpMemory(get_doc(This), &mem, &size);
1743 * libxml2 always adds XML declaration on top of the file and one for each processing instruction node in DOM tree.
1744 * MSXML adds XML declaration only for processing instruction nodes.
1745 * We skip the first XML declaration generated by libxml2 to get exactly what we need.
1747 p = mem;
1748 if(size > 2 && p[0] == '<' && p[1] == '?') {
1749 while(p < mem+size && (p[0] != '?' || p[1] != '>'))
1750 p++;
1751 p += 2;
1752 while(p < mem+size && isspace(*p))
1753 p++;
1754 size -= p-mem;
1757 if(!WriteFile(handle, p, (DWORD)size, &written, NULL) || written != (DWORD)size)
1759 WARN("write error\n");
1760 ret = S_FALSE;
1763 xmlFree(mem);
1764 CloseHandle(handle);
1765 return ret;
1768 static HRESULT WINAPI domdoc_get_validateOnParse(
1769 IXMLDOMDocument2 *iface,
1770 VARIANT_BOOL* isValidating )
1772 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1774 TRACE("%p <- %d\n", isValidating, This->validating);
1775 *isValidating = This->validating;
1776 return S_OK;
1780 static HRESULT WINAPI domdoc_put_validateOnParse(
1781 IXMLDOMDocument2 *iface,
1782 VARIANT_BOOL isValidating )
1784 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1786 TRACE("%d\n", isValidating);
1787 This->validating = isValidating;
1788 return S_OK;
1792 static HRESULT WINAPI domdoc_get_resolveExternals(
1793 IXMLDOMDocument2 *iface,
1794 VARIANT_BOOL* isResolving )
1796 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1798 TRACE("%p <- %d\n", isResolving, This->resolving);
1799 *isResolving = This->resolving;
1800 return S_OK;
1804 static HRESULT WINAPI domdoc_put_resolveExternals(
1805 IXMLDOMDocument2 *iface,
1806 VARIANT_BOOL isResolving )
1808 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1810 TRACE("%d\n", isResolving);
1811 This->resolving = isResolving;
1812 return S_OK;
1816 static HRESULT WINAPI domdoc_get_preserveWhiteSpace(
1817 IXMLDOMDocument2 *iface,
1818 VARIANT_BOOL* isPreserving )
1820 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1822 TRACE("%p <- %d\n", isPreserving, This->preserving);
1823 *isPreserving = This->preserving;
1824 return S_OK;
1828 static HRESULT WINAPI domdoc_put_preserveWhiteSpace(
1829 IXMLDOMDocument2 *iface,
1830 VARIANT_BOOL isPreserving )
1832 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1834 TRACE("%d\n", isPreserving);
1835 This->preserving = isPreserving;
1836 return S_OK;
1840 static HRESULT WINAPI domdoc_put_onReadyStateChange(
1841 IXMLDOMDocument2 *iface,
1842 VARIANT readyStateChangeSink )
1844 FIXME("\n");
1845 return E_NOTIMPL;
1849 static HRESULT WINAPI domdoc_put_onDataAvailable(
1850 IXMLDOMDocument2 *iface,
1851 VARIANT onDataAvailableSink )
1853 FIXME("\n");
1854 return E_NOTIMPL;
1857 static HRESULT WINAPI domdoc_put_onTransformNode(
1858 IXMLDOMDocument2 *iface,
1859 VARIANT onTransformNodeSink )
1861 FIXME("\n");
1862 return E_NOTIMPL;
1865 static HRESULT WINAPI domdoc_get_namespaces(
1866 IXMLDOMDocument2* iface,
1867 IXMLDOMSchemaCollection** schemaCollection )
1869 FIXME("\n");
1870 return E_NOTIMPL;
1873 static HRESULT WINAPI domdoc_get_schemas(
1874 IXMLDOMDocument2* iface,
1875 VARIANT* var1 )
1877 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1878 HRESULT hr = S_FALSE;
1879 IXMLDOMSchemaCollection *cur_schema = This->schema;
1881 TRACE("(%p)->(%p)\n", This, var1);
1883 VariantInit(var1); /* Test shows we don't call VariantClear here */
1884 V_VT(var1) = VT_NULL;
1886 if(cur_schema)
1888 hr = IXMLDOMSchemaCollection_QueryInterface(cur_schema, &IID_IDispatch, (void**)&V_DISPATCH(var1));
1889 if(SUCCEEDED(hr))
1890 V_VT(var1) = VT_DISPATCH;
1892 return hr;
1895 static HRESULT WINAPI domdoc_putref_schemas(
1896 IXMLDOMDocument2* iface,
1897 VARIANT var1)
1899 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1900 HRESULT hr = E_FAIL;
1901 IXMLDOMSchemaCollection *new_schema = NULL;
1903 FIXME("(%p): semi-stub\n", This);
1904 switch(V_VT(&var1))
1906 case VT_UNKNOWN:
1907 hr = IUnknown_QueryInterface(V_UNKNOWN(&var1), &IID_IXMLDOMSchemaCollection, (void**)&new_schema);
1908 break;
1910 case VT_DISPATCH:
1911 hr = IDispatch_QueryInterface(V_DISPATCH(&var1), &IID_IXMLDOMSchemaCollection, (void**)&new_schema);
1912 break;
1914 case VT_NULL:
1915 case VT_EMPTY:
1916 hr = S_OK;
1917 break;
1919 default:
1920 WARN("Can't get schema from vt %x\n", V_VT(&var1));
1923 if(SUCCEEDED(hr))
1925 IXMLDOMSchemaCollection *old_schema = InterlockedExchangePointer((void**)&This->schema, new_schema);
1926 if(old_schema) IXMLDOMSchemaCollection_Release(old_schema);
1929 return hr;
1932 static HRESULT WINAPI domdoc_validate(
1933 IXMLDOMDocument2* iface,
1934 IXMLDOMParseError** err)
1936 FIXME("\n");
1937 return E_NOTIMPL;
1940 static HRESULT WINAPI domdoc_setProperty(
1941 IXMLDOMDocument2* iface,
1942 BSTR p,
1943 VARIANT var)
1945 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1947 if (lstrcmpiW(p, SZ_PROPERTY_SELECTION_LANGUAGE) == 0)
1949 VARIANT varStr;
1950 HRESULT hr;
1951 BSTR bstr;
1953 V_VT(&varStr) = VT_EMPTY;
1954 if (V_VT(&var) != VT_BSTR)
1956 if (FAILED(hr = VariantChangeType(&varStr, &var, 0, VT_BSTR)))
1957 return hr;
1958 bstr = V_BSTR(&varStr);
1960 else
1961 bstr = V_BSTR(&var);
1963 hr = S_OK;
1964 if (lstrcmpiW(bstr, SZ_VALUE_XPATH) == 0)
1965 This->bUseXPath = TRUE;
1966 else if (lstrcmpiW(bstr, SZ_VALUE_XSLPATTERN) == 0)
1967 This->bUseXPath = FALSE;
1968 else
1969 hr = E_FAIL;
1971 VariantClear(&varStr);
1972 return hr;
1975 FIXME("Unknown property %s\n", wine_dbgstr_w(p));
1976 return E_FAIL;
1979 static HRESULT WINAPI domdoc_getProperty(
1980 IXMLDOMDocument2* iface,
1981 BSTR p,
1982 VARIANT* var)
1984 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1986 if (var == NULL)
1987 return E_INVALIDARG;
1988 if (lstrcmpiW(p, SZ_PROPERTY_SELECTION_LANGUAGE) == 0)
1990 V_VT(var) = VT_BSTR;
1991 if (This->bUseXPath)
1992 V_BSTR(var) = SysAllocString(SZ_VALUE_XPATH);
1993 else
1994 V_BSTR(var) = SysAllocString(SZ_VALUE_XSLPATTERN);
1995 return S_OK;
1998 FIXME("Unknown property %s\n", wine_dbgstr_w(p));
1999 return E_FAIL;
2002 static const struct IXMLDOMDocument2Vtbl domdoc_vtbl =
2004 domdoc_QueryInterface,
2005 domdoc_AddRef,
2006 domdoc_Release,
2007 domdoc_GetTypeInfoCount,
2008 domdoc_GetTypeInfo,
2009 domdoc_GetIDsOfNames,
2010 domdoc_Invoke,
2011 domdoc_get_nodeName,
2012 domdoc_get_nodeValue,
2013 domdoc_put_nodeValue,
2014 domdoc_get_nodeType,
2015 domdoc_get_parentNode,
2016 domdoc_get_childNodes,
2017 domdoc_get_firstChild,
2018 domdoc_get_lastChild,
2019 domdoc_get_previousSibling,
2020 domdoc_get_nextSibling,
2021 domdoc_get_attributes,
2022 domdoc_insertBefore,
2023 domdoc_replaceChild,
2024 domdoc_removeChild,
2025 domdoc_appendChild,
2026 domdoc_hasChildNodes,
2027 domdoc_get_ownerDocument,
2028 domdoc_cloneNode,
2029 domdoc_get_nodeTypeString,
2030 domdoc_get_text,
2031 domdoc_put_text,
2032 domdoc_get_specified,
2033 domdoc_get_definition,
2034 domdoc_get_nodeTypedValue,
2035 domdoc_put_nodeTypedValue,
2036 domdoc_get_dataType,
2037 domdoc_put_dataType,
2038 domdoc_get_xml,
2039 domdoc_transformNode,
2040 domdoc_selectNodes,
2041 domdoc_selectSingleNode,
2042 domdoc_get_parsed,
2043 domdoc_get_namespaceURI,
2044 domdoc_get_prefix,
2045 domdoc_get_baseName,
2046 domdoc_transformNodeToObject,
2047 domdoc_get_doctype,
2048 domdoc_get_implementation,
2049 domdoc_get_documentElement,
2050 domdoc_put_documentElement,
2051 domdoc_createElement,
2052 domdoc_createDocumentFragment,
2053 domdoc_createTextNode,
2054 domdoc_createComment,
2055 domdoc_createCDATASection,
2056 domdoc_createProcessingInstruction,
2057 domdoc_createAttribute,
2058 domdoc_createEntityReference,
2059 domdoc_getElementsByTagName,
2060 domdoc_createNode,
2061 domdoc_nodeFromID,
2062 domdoc_load,
2063 domdoc_get_readyState,
2064 domdoc_get_parseError,
2065 domdoc_get_url,
2066 domdoc_get_async,
2067 domdoc_put_async,
2068 domdoc_abort,
2069 domdoc_loadXML,
2070 domdoc_save,
2071 domdoc_get_validateOnParse,
2072 domdoc_put_validateOnParse,
2073 domdoc_get_resolveExternals,
2074 domdoc_put_resolveExternals,
2075 domdoc_get_preserveWhiteSpace,
2076 domdoc_put_preserveWhiteSpace,
2077 domdoc_put_onReadyStateChange,
2078 domdoc_put_onDataAvailable,
2079 domdoc_put_onTransformNode,
2080 domdoc_get_namespaces,
2081 domdoc_get_schemas,
2082 domdoc_putref_schemas,
2083 domdoc_validate,
2084 domdoc_setProperty,
2085 domdoc_getProperty
2088 /* xmldoc implementation of IObjectWithSite */
2089 static HRESULT WINAPI
2090 xmldoc_ObjectWithSite_QueryInterface( IObjectWithSite* iface, REFIID riid, void** ppvObject )
2092 domdoc *This = impl_from_IObjectWithSite(iface);
2093 return IXMLDocument_QueryInterface( (IXMLDocument *)This, riid, ppvObject );
2096 static ULONG WINAPI
2097 xmldoc_ObjectWithSite_AddRef( IObjectWithSite* iface )
2099 domdoc *This = impl_from_IObjectWithSite(iface);
2100 return IXMLDocument_AddRef((IXMLDocument *)This);
2103 static ULONG WINAPI
2104 xmldoc_ObjectWithSite_Release( IObjectWithSite* iface )
2106 domdoc *This = impl_from_IObjectWithSite(iface);
2107 return IXMLDocument_Release((IXMLDocument *)This);
2110 static HRESULT WINAPI
2111 xmldoc_GetSite( IObjectWithSite *iface, REFIID iid, void ** ppvSite )
2113 domdoc *This = impl_from_IObjectWithSite(iface);
2115 TRACE("%p %s %p\n", This, debugstr_guid( iid ), ppvSite );
2117 if ( !This->site )
2118 return E_FAIL;
2120 return IUnknown_QueryInterface( This->site, iid, ppvSite );
2123 static HRESULT WINAPI
2124 xmldoc_SetSite( IObjectWithSite *iface, IUnknown *punk )
2126 domdoc *This = impl_from_IObjectWithSite(iface);
2128 TRACE("%p %p\n", iface, punk);
2130 if(!punk)
2132 if(This->site)
2134 IUnknown_Release( This->site );
2135 This->site = NULL;
2138 return S_OK;
2141 if ( punk )
2142 IUnknown_AddRef( punk );
2144 if(This->site)
2145 IUnknown_Release( This->site );
2147 This->site = punk;
2149 return S_OK;
2152 static const IObjectWithSiteVtbl domdocObjectSite =
2154 xmldoc_ObjectWithSite_QueryInterface,
2155 xmldoc_ObjectWithSite_AddRef,
2156 xmldoc_ObjectWithSite_Release,
2157 xmldoc_SetSite,
2158 xmldoc_GetSite,
2161 static HRESULT WINAPI xmldoc_Safety_QueryInterface(IObjectSafety *iface, REFIID riid, void **ppv)
2163 domdoc *This = impl_from_IObjectSafety(iface);
2164 return IXMLDocument_QueryInterface( (IXMLDocument *)This, riid, ppv );
2167 static ULONG WINAPI xmldoc_Safety_AddRef(IObjectSafety *iface)
2169 domdoc *This = impl_from_IObjectSafety(iface);
2170 return IXMLDocument_AddRef((IXMLDocument *)This);
2173 static ULONG WINAPI xmldoc_Safety_Release(IObjectSafety *iface)
2175 domdoc *This = impl_from_IObjectSafety(iface);
2176 return IXMLDocument_Release((IXMLDocument *)This);
2179 #define SUPPORTED_OPTIONS (INTERFACESAFE_FOR_UNTRUSTED_CALLER|INTERFACESAFE_FOR_UNTRUSTED_DATA|INTERFACE_USES_SECURITY_MANAGER)
2181 static HRESULT WINAPI xmldoc_Safety_GetInterfaceSafetyOptions(IObjectSafety *iface, REFIID riid,
2182 DWORD *pdwSupportedOptions, DWORD *pdwEnabledOptions)
2184 domdoc *This = impl_from_IObjectSafety(iface);
2186 TRACE("(%p)->(%s %p %p)\n", This, debugstr_guid(riid), pdwSupportedOptions, pdwEnabledOptions);
2188 if(!pdwSupportedOptions || !pdwEnabledOptions)
2189 return E_POINTER;
2191 *pdwSupportedOptions = SUPPORTED_OPTIONS;
2192 *pdwEnabledOptions = This->safeopt;
2194 return S_OK;
2197 static HRESULT WINAPI xmldoc_Safety_SetInterfaceSafetyOptions(IObjectSafety *iface, REFIID riid,
2198 DWORD dwOptionSetMask, DWORD dwEnabledOptions)
2200 domdoc *This = impl_from_IObjectSafety(iface);
2202 TRACE("(%p)->(%s %x %x)\n", This, debugstr_guid(riid), dwOptionSetMask, dwEnabledOptions);
2204 if(dwOptionSetMask & ~SUPPORTED_OPTIONS)
2205 return E_FAIL;
2207 This->safeopt = dwEnabledOptions & dwEnabledOptions;
2208 return S_OK;
2211 static const IObjectSafetyVtbl domdocObjectSafetyVtbl = {
2212 xmldoc_Safety_QueryInterface,
2213 xmldoc_Safety_AddRef,
2214 xmldoc_Safety_Release,
2215 xmldoc_Safety_GetInterfaceSafetyOptions,
2216 xmldoc_Safety_SetInterfaceSafetyOptions
2219 HRESULT DOMDocument_create_from_xmldoc(xmlDocPtr xmldoc, IXMLDOMDocument2 **document)
2221 domdoc *doc;
2222 HRESULT hr;
2224 doc = HeapAlloc( GetProcessHeap(), 0, sizeof (*doc) );
2225 if( !doc )
2226 return E_OUTOFMEMORY;
2228 doc->lpVtbl = &domdoc_vtbl;
2229 doc->lpvtblIPersistStream = &xmldoc_IPersistStream_VTable;
2230 doc->lpvtblIObjectWithSite = &domdocObjectSite;
2231 doc->lpvtblIObjectSafety = &domdocObjectSafetyVtbl;
2232 doc->ref = 1;
2233 doc->async = 0;
2234 doc->validating = 0;
2235 doc->resolving = 0;
2236 doc->preserving = 0;
2237 doc->bUseXPath = FALSE;
2238 doc->error = S_OK;
2239 doc->schema = NULL;
2240 doc->stream = NULL;
2241 doc->site = NULL;
2242 doc->safeopt = 0;
2243 doc->bsc = NULL;
2245 doc->node_unk = create_basic_node( (xmlNodePtr)xmldoc, (IUnknown*)&doc->lpVtbl );
2246 if(!doc->node_unk)
2248 HeapFree(GetProcessHeap(), 0, doc);
2249 return E_FAIL;
2252 hr = IUnknown_QueryInterface(doc->node_unk, &IID_IXMLDOMNode, (LPVOID*)&doc->node);
2253 if(FAILED(hr))
2255 IUnknown_Release(doc->node_unk);
2256 HeapFree( GetProcessHeap(), 0, doc );
2257 return E_FAIL;
2259 /* The ref on doc->node is actually looped back into this object, so release it */
2260 IXMLDOMNode_Release(doc->node);
2262 *document = (IXMLDOMDocument2*)&doc->lpVtbl;
2264 TRACE("returning iface %p\n", *document);
2265 return S_OK;
2268 HRESULT DOMDocument_create(IUnknown *pUnkOuter, LPVOID *ppObj)
2270 xmlDocPtr xmldoc;
2271 HRESULT hr;
2273 TRACE("(%p,%p)\n", pUnkOuter, ppObj);
2275 xmldoc = xmlNewDoc(NULL);
2276 if(!xmldoc)
2277 return E_OUTOFMEMORY;
2279 xmldoc->_private = 0;
2281 hr = DOMDocument_create_from_xmldoc(xmldoc, (IXMLDOMDocument2**)ppObj);
2282 if(FAILED(hr))
2283 xmlFreeDoc(xmldoc);
2285 return hr;
2288 #else
2290 HRESULT DOMDocument_create(IUnknown *pUnkOuter, LPVOID *ppObj)
2292 MESSAGE("This program tried to use a DOMDocument object, but\n"
2293 "libxml2 support was not present at compile time.\n");
2294 return E_NOTIMPL;
2297 #endif