push 1a31a9cb1b14d0c41b748a6144c6aef5fac5e083
[wine/hacks.git] / dlls / msxml3 / domdoc.c
blobfed1b8b6149b9a63b2efd06bca48ef866c7061e0
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"
38 #include "wine/debug.h"
40 #include "msxml_private.h"
42 WINE_DEFAULT_DEBUG_CHANNEL(msxml);
44 #ifdef HAVE_LIBXML2
46 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};
47 static const WCHAR SZ_VALUE_XPATH[] = {'X','P','a','t','h',0};
48 static const WCHAR SZ_VALUE_XSLPATTERN[] = {'X','S','L','P','a','t','t','e','r','n',0};
50 typedef struct {
51 const struct IBindStatusCallbackVtbl *lpVtbl;
52 } bsc;
54 static HRESULT WINAPI bsc_QueryInterface(
55 IBindStatusCallback *iface,
56 REFIID riid,
57 LPVOID *ppobj )
59 if (IsEqualGUID(riid, &IID_IUnknown) ||
60 IsEqualGUID(riid, &IID_IBindStatusCallback))
62 IBindStatusCallback_AddRef( iface );
63 *ppobj = iface;
64 return S_OK;
67 FIXME("interface %s not implemented\n", debugstr_guid(riid));
68 return E_NOINTERFACE;
71 static ULONG WINAPI bsc_AddRef(
72 IBindStatusCallback *iface )
74 return 2;
77 static ULONG WINAPI bsc_Release(
78 IBindStatusCallback *iface )
80 return 1;
83 static HRESULT WINAPI bsc_OnStartBinding(
84 IBindStatusCallback* iface,
85 DWORD dwReserved,
86 IBinding* pib)
88 return S_OK;
91 static HRESULT WINAPI bsc_GetPriority(
92 IBindStatusCallback* iface,
93 LONG* pnPriority)
95 return S_OK;
98 static HRESULT WINAPI bsc_OnLowResource(
99 IBindStatusCallback* iface,
100 DWORD reserved)
102 return S_OK;
105 static HRESULT WINAPI bsc_OnProgress(
106 IBindStatusCallback* iface,
107 ULONG ulProgress,
108 ULONG ulProgressMax,
109 ULONG ulStatusCode,
110 LPCWSTR szStatusText)
112 return S_OK;
115 static HRESULT WINAPI bsc_OnStopBinding(
116 IBindStatusCallback* iface,
117 HRESULT hresult,
118 LPCWSTR szError)
120 return S_OK;
123 static HRESULT WINAPI bsc_GetBindInfo(
124 IBindStatusCallback* iface,
125 DWORD* grfBINDF,
126 BINDINFO* pbindinfo)
128 *grfBINDF = BINDF_RESYNCHRONIZE;
130 return S_OK;
133 static HRESULT WINAPI bsc_OnDataAvailable(
134 IBindStatusCallback* iface,
135 DWORD grfBSCF,
136 DWORD dwSize,
137 FORMATETC* pformatetc,
138 STGMEDIUM* pstgmed)
140 return S_OK;
143 static HRESULT WINAPI bsc_OnObjectAvailable(
144 IBindStatusCallback* iface,
145 REFIID riid,
146 IUnknown* punk)
148 return S_OK;
151 static const struct IBindStatusCallbackVtbl bsc_vtbl =
153 bsc_QueryInterface,
154 bsc_AddRef,
155 bsc_Release,
156 bsc_OnStartBinding,
157 bsc_GetPriority,
158 bsc_OnLowResource,
159 bsc_OnProgress,
160 bsc_OnStopBinding,
161 bsc_GetBindInfo,
162 bsc_OnDataAvailable,
163 bsc_OnObjectAvailable
166 static bsc domdoc_bsc = { &bsc_vtbl };
168 typedef struct _domdoc
170 const struct IXMLDOMDocument2Vtbl *lpVtbl;
171 const struct IPersistStreamVtbl *lpvtblIPersistStream;
172 LONG ref;
173 VARIANT_BOOL async;
174 VARIANT_BOOL validating;
175 VARIANT_BOOL resolving;
176 VARIANT_BOOL preserving;
177 BOOL bUseXPath;
178 IUnknown *node_unk;
179 IXMLDOMNode *node;
180 IXMLDOMSchemaCollection *schema;
181 HRESULT error;
183 /* IPersistStream */
184 IStream *stream;
185 } domdoc;
187 LONG xmldoc_add_ref(xmlDocPtr doc)
189 LONG ref = InterlockedIncrement((LONG*)&doc->_private);
190 TRACE("%d\n", ref);
191 return ref;
194 LONG xmldoc_release(xmlDocPtr doc)
196 LONG ref = InterlockedDecrement((LONG*)&doc->_private);
197 TRACE("%d\n", ref);
198 if(ref == 0)
200 TRACE("freeing docptr %p\n", doc);
201 xmlFreeDoc(doc);
204 return ref;
207 static inline domdoc *impl_from_IXMLDOMDocument2( IXMLDOMDocument2 *iface )
209 return (domdoc *)((char*)iface - FIELD_OFFSET(domdoc, lpVtbl));
212 static inline xmlDocPtr get_doc( domdoc *This )
214 return (xmlDocPtr) xmlNodePtr_from_domnode( This->node, XML_DOCUMENT_NODE );
217 static inline domdoc *impl_from_IPersistStream(IPersistStream *iface)
219 return (domdoc *)((char*)iface - FIELD_OFFSET(domdoc, lpvtblIPersistStream));
222 /************************************************************************
223 * xmldoc implementation of IPersistStream.
225 static HRESULT WINAPI xmldoc_IPersistStream_QueryInterface(
226 IPersistStream *iface, REFIID riid, LPVOID *ppvObj)
228 domdoc *this = impl_from_IPersistStream(iface);
229 return IXMLDocument_QueryInterface((IXMLDocument *)this, riid, ppvObj);
232 static ULONG WINAPI xmldoc_IPersistStream_AddRef(
233 IPersistStream *iface)
235 domdoc *this = impl_from_IPersistStream(iface);
236 return IXMLDocument_AddRef((IXMLDocument *)this);
239 static ULONG WINAPI xmldoc_IPersistStream_Release(
240 IPersistStream *iface)
242 domdoc *this = impl_from_IPersistStream(iface);
243 return IXMLDocument_Release((IXMLDocument *)this);
246 static HRESULT WINAPI xmldoc_IPersistStream_GetClassID(
247 IPersistStream *iface, CLSID *classid)
249 TRACE("(%p,%p): stub!\n", iface, classid);
251 if(!classid)
252 return E_POINTER;
254 *classid = CLSID_DOMDocument2;
256 return S_OK;
259 static HRESULT WINAPI xmldoc_IPersistStream_IsDirty(
260 IPersistStream *iface)
262 domdoc *This = impl_from_IPersistStream(iface);
264 FIXME("(%p->%p): stub!\n", iface, This);
266 return S_FALSE;
269 static HRESULT WINAPI xmldoc_IPersistStream_Load(
270 IPersistStream *iface, LPSTREAM pStm)
272 domdoc *This = impl_from_IPersistStream(iface);
273 HRESULT hr;
274 HGLOBAL hglobal;
275 DWORD read, written, len;
276 BYTE buf[4096];
277 char *ptr;
278 xmlDocPtr xmldoc = NULL;
280 TRACE("(%p, %p)\n", iface, pStm);
282 if (!pStm)
283 return E_INVALIDARG;
285 hr = CreateStreamOnHGlobal(NULL, TRUE, &This->stream);
286 if (FAILED(hr))
287 return hr;
291 IStream_Read(pStm, buf, sizeof(buf), &read);
292 hr = IStream_Write(This->stream, buf, read, &written);
293 } while(SUCCEEDED(hr) && written != 0 && read != 0);
295 if (FAILED(hr))
297 ERR("Failed to copy stream\n");
298 return hr;
301 hr = GetHGlobalFromStream(This->stream, &hglobal);
302 if (FAILED(hr))
303 return hr;
305 len = GlobalSize(hglobal);
306 ptr = GlobalLock(hglobal);
307 if (len != 0)
308 xmldoc = parse_xml(ptr, len);
309 GlobalUnlock(hglobal);
311 if (!xmldoc)
313 ERR("Failed to parse xml\n");
314 return E_FAIL;
317 attach_xmlnode( This->node, (xmlNodePtr)xmldoc );
319 return S_OK;
322 static HRESULT WINAPI xmldoc_IPersistStream_Save(
323 IPersistStream *iface, LPSTREAM pStm, BOOL fClearDirty)
325 FIXME("(%p, %p, %d): stub!\n", iface, pStm, fClearDirty);
326 return E_NOTIMPL;
329 static HRESULT WINAPI xmldoc_IPersistStream_GetSizeMax(
330 IPersistStream *iface, ULARGE_INTEGER *pcbSize)
332 TRACE("(%p, %p): stub!\n", iface, pcbSize);
333 return E_NOTIMPL;
336 static const IPersistStreamVtbl xmldoc_IPersistStream_VTable =
338 xmldoc_IPersistStream_QueryInterface,
339 xmldoc_IPersistStream_AddRef,
340 xmldoc_IPersistStream_Release,
341 xmldoc_IPersistStream_GetClassID,
342 xmldoc_IPersistStream_IsDirty,
343 xmldoc_IPersistStream_Load,
344 xmldoc_IPersistStream_Save,
345 xmldoc_IPersistStream_GetSizeMax,
348 static HRESULT WINAPI domdoc_QueryInterface( IXMLDOMDocument2 *iface, REFIID riid, void** ppvObject )
350 domdoc *This = impl_from_IXMLDOMDocument2( iface );
352 TRACE("%p %s %p\n", This, debugstr_guid( riid ), ppvObject );
354 *ppvObject = NULL;
356 if ( IsEqualGUID( riid, &IID_IUnknown ) ||
357 IsEqualGUID( riid, &IID_IDispatch ) ||
358 IsEqualGUID( riid, &IID_IXMLDOMDocument ) ||
359 IsEqualGUID( riid, &IID_IXMLDOMDocument2 ) )
361 *ppvObject = iface;
363 else if ( IsEqualGUID( riid, &IID_IXMLDOMNode ) )
365 return IUnknown_QueryInterface(This->node_unk, riid, ppvObject);
367 else if (IsEqualGUID(&IID_IPersistStream, riid))
369 *ppvObject = (IPersistStream*)&(This->lpvtblIPersistStream);
371 else if(IsEqualGUID(&IID_IRunnableObject, riid))
373 TRACE("IID_IRunnableObject not supported returning NULL\n");
374 return E_NOINTERFACE;
376 else
378 FIXME("interface %s not implemented\n", debugstr_guid(riid));
379 return E_NOINTERFACE;
382 IXMLDOMDocument_AddRef( iface );
384 return S_OK;
388 static ULONG WINAPI domdoc_AddRef(
389 IXMLDOMDocument2 *iface )
391 domdoc *This = impl_from_IXMLDOMDocument2( iface );
392 TRACE("%p\n", This );
393 return InterlockedIncrement( &This->ref );
397 static ULONG WINAPI domdoc_Release(
398 IXMLDOMDocument2 *iface )
400 domdoc *This = impl_from_IXMLDOMDocument2( iface );
401 LONG ref;
403 TRACE("%p\n", This );
405 ref = InterlockedDecrement( &This->ref );
406 if ( ref == 0 )
408 IUnknown_Release( This->node_unk );
409 if(This->schema) IXMLDOMSchemaCollection_Release( This->schema );
410 if (This->stream) IStream_Release(This->stream);
411 HeapFree( GetProcessHeap(), 0, This );
414 return ref;
417 static HRESULT WINAPI domdoc_GetTypeInfoCount( IXMLDOMDocument2 *iface, UINT* pctinfo )
419 domdoc *This = impl_from_IXMLDOMDocument2( iface );
421 TRACE("(%p)->(%p)\n", This, pctinfo);
423 *pctinfo = 1;
425 return S_OK;
428 static HRESULT WINAPI domdoc_GetTypeInfo(
429 IXMLDOMDocument2 *iface,
430 UINT iTInfo, LCID lcid, ITypeInfo** ppTInfo )
432 domdoc *This = impl_from_IXMLDOMDocument2( iface );
433 HRESULT hr;
435 TRACE("(%p)->(%u %u %p)\n", This, iTInfo, lcid, ppTInfo);
437 hr = get_typeinfo(IXMLDOMDocument2_tid, ppTInfo);
439 return hr;
442 static HRESULT WINAPI domdoc_GetIDsOfNames(
443 IXMLDOMDocument2 *iface,
444 REFIID riid,
445 LPOLESTR* rgszNames,
446 UINT cNames,
447 LCID lcid,
448 DISPID* rgDispId)
450 domdoc *This = impl_from_IXMLDOMDocument2( iface );
451 ITypeInfo *typeinfo;
452 HRESULT hr;
454 TRACE("(%p)->(%s %p %u %u %p)\n", This, debugstr_guid(riid), rgszNames, cNames,
455 lcid, rgDispId);
457 if(!rgszNames || cNames == 0 || !rgDispId)
458 return E_INVALIDARG;
460 hr = get_typeinfo(IXMLDOMDocument2_tid, &typeinfo);
461 if(SUCCEEDED(hr))
463 hr = ITypeInfo_GetIDsOfNames(typeinfo, rgszNames, cNames, rgDispId);
464 ITypeInfo_Release(typeinfo);
467 return hr;
471 static HRESULT WINAPI domdoc_Invoke(
472 IXMLDOMDocument2 *iface,
473 DISPID dispIdMember,
474 REFIID riid,
475 LCID lcid,
476 WORD wFlags,
477 DISPPARAMS* pDispParams,
478 VARIANT* pVarResult,
479 EXCEPINFO* pExcepInfo,
480 UINT* puArgErr)
482 domdoc *This = impl_from_IXMLDOMDocument2( iface );
483 ITypeInfo *typeinfo;
484 HRESULT hr;
486 TRACE("(%p)->(%d %s %d %d %p %p %p %p)\n", This, dispIdMember, debugstr_guid(riid),
487 lcid, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr);
489 hr = get_typeinfo(IXMLDOMDocument2_tid, &typeinfo);
490 if(SUCCEEDED(hr))
492 hr = ITypeInfo_Invoke(typeinfo, &(This->lpVtbl), dispIdMember, wFlags, pDispParams,
493 pVarResult, pExcepInfo, puArgErr);
494 ITypeInfo_Release(typeinfo);
497 return hr;
501 static HRESULT WINAPI domdoc_get_nodeName(
502 IXMLDOMDocument2 *iface,
503 BSTR* name )
505 domdoc *This = impl_from_IXMLDOMDocument2( iface );
506 return IXMLDOMNode_get_nodeName( This->node, name );
510 static HRESULT WINAPI domdoc_get_nodeValue(
511 IXMLDOMDocument2 *iface,
512 VARIANT* value )
514 domdoc *This = impl_from_IXMLDOMDocument2( iface );
515 return IXMLDOMNode_get_nodeValue( This->node, value );
519 static HRESULT WINAPI domdoc_put_nodeValue(
520 IXMLDOMDocument2 *iface,
521 VARIANT value)
523 domdoc *This = impl_from_IXMLDOMDocument2( iface );
524 return IXMLDOMNode_put_nodeValue( This->node, value );
528 static HRESULT WINAPI domdoc_get_nodeType(
529 IXMLDOMDocument2 *iface,
530 DOMNodeType* type )
532 domdoc *This = impl_from_IXMLDOMDocument2( iface );
533 return IXMLDOMNode_get_nodeType( This->node, type );
537 static HRESULT WINAPI domdoc_get_parentNode(
538 IXMLDOMDocument2 *iface,
539 IXMLDOMNode** parent )
541 domdoc *This = impl_from_IXMLDOMDocument2( iface );
542 return IXMLDOMNode_get_parentNode( This->node, parent );
546 static HRESULT WINAPI domdoc_get_childNodes(
547 IXMLDOMDocument2 *iface,
548 IXMLDOMNodeList** childList )
550 domdoc *This = impl_from_IXMLDOMDocument2( iface );
551 return IXMLDOMNode_get_childNodes( This->node, childList );
555 static HRESULT WINAPI domdoc_get_firstChild(
556 IXMLDOMDocument2 *iface,
557 IXMLDOMNode** firstChild )
559 domdoc *This = impl_from_IXMLDOMDocument2( iface );
560 return IXMLDOMNode_get_firstChild( This->node, firstChild );
564 static HRESULT WINAPI domdoc_get_lastChild(
565 IXMLDOMDocument2 *iface,
566 IXMLDOMNode** lastChild )
568 domdoc *This = impl_from_IXMLDOMDocument2( iface );
569 return IXMLDOMNode_get_lastChild( This->node, lastChild );
573 static HRESULT WINAPI domdoc_get_previousSibling(
574 IXMLDOMDocument2 *iface,
575 IXMLDOMNode** previousSibling )
577 domdoc *This = impl_from_IXMLDOMDocument2( iface );
578 return IXMLDOMNode_get_previousSibling( This->node, previousSibling );
582 static HRESULT WINAPI domdoc_get_nextSibling(
583 IXMLDOMDocument2 *iface,
584 IXMLDOMNode** nextSibling )
586 domdoc *This = impl_from_IXMLDOMDocument2( iface );
587 return IXMLDOMNode_get_nextSibling( This->node, nextSibling );
591 static HRESULT WINAPI domdoc_get_attributes(
592 IXMLDOMDocument2 *iface,
593 IXMLDOMNamedNodeMap** attributeMap )
595 domdoc *This = impl_from_IXMLDOMDocument2( iface );
596 return IXMLDOMNode_get_attributes( This->node, attributeMap );
600 static HRESULT WINAPI domdoc_insertBefore(
601 IXMLDOMDocument2 *iface,
602 IXMLDOMNode* newChild,
603 VARIANT refChild,
604 IXMLDOMNode** outNewChild )
606 domdoc *This = impl_from_IXMLDOMDocument2( iface );
607 return IXMLDOMNode_insertBefore( This->node, newChild, refChild, outNewChild );
611 static HRESULT WINAPI domdoc_replaceChild(
612 IXMLDOMDocument2 *iface,
613 IXMLDOMNode* newChild,
614 IXMLDOMNode* oldChild,
615 IXMLDOMNode** outOldChild)
617 domdoc *This = impl_from_IXMLDOMDocument2( iface );
618 return IXMLDOMNode_replaceChild( This->node, newChild, oldChild, outOldChild );
622 static HRESULT WINAPI domdoc_removeChild(
623 IXMLDOMDocument2 *iface,
624 IXMLDOMNode* childNode,
625 IXMLDOMNode** oldChild)
627 domdoc *This = impl_from_IXMLDOMDocument2( iface );
628 return IXMLDOMNode_removeChild( This->node, childNode, oldChild );
632 static HRESULT WINAPI domdoc_appendChild(
633 IXMLDOMDocument2 *iface,
634 IXMLDOMNode* newChild,
635 IXMLDOMNode** outNewChild)
637 domdoc *This = impl_from_IXMLDOMDocument2( iface );
638 return IXMLDOMNode_appendChild( This->node, newChild, outNewChild );
642 static HRESULT WINAPI domdoc_hasChildNodes(
643 IXMLDOMDocument2 *iface,
644 VARIANT_BOOL* hasChild)
646 domdoc *This = impl_from_IXMLDOMDocument2( iface );
647 return IXMLDOMNode_hasChildNodes( This->node, hasChild );
651 static HRESULT WINAPI domdoc_get_ownerDocument(
652 IXMLDOMDocument2 *iface,
653 IXMLDOMDocument** DOMDocument)
655 domdoc *This = impl_from_IXMLDOMDocument2( iface );
656 return IXMLDOMNode_get_ownerDocument( This->node, DOMDocument );
660 static HRESULT WINAPI domdoc_cloneNode(
661 IXMLDOMDocument2 *iface,
662 VARIANT_BOOL deep,
663 IXMLDOMNode** cloneRoot)
665 domdoc *This = impl_from_IXMLDOMDocument2( iface );
666 return IXMLDOMNode_cloneNode( This->node, deep, cloneRoot );
670 static HRESULT WINAPI domdoc_get_nodeTypeString(
671 IXMLDOMDocument2 *iface,
672 BSTR* nodeType )
674 domdoc *This = impl_from_IXMLDOMDocument2( iface );
675 return IXMLDOMNode_get_nodeTypeString( This->node, nodeType );
679 static HRESULT WINAPI domdoc_get_text(
680 IXMLDOMDocument2 *iface,
681 BSTR* text )
683 domdoc *This = impl_from_IXMLDOMDocument2( iface );
684 return IXMLDOMNode_get_text( This->node, text );
688 static HRESULT WINAPI domdoc_put_text(
689 IXMLDOMDocument2 *iface,
690 BSTR text )
692 domdoc *This = impl_from_IXMLDOMDocument2( iface );
693 return IXMLDOMNode_put_text( This->node, text );
697 static HRESULT WINAPI domdoc_get_specified(
698 IXMLDOMDocument2 *iface,
699 VARIANT_BOOL* isSpecified )
701 domdoc *This = impl_from_IXMLDOMDocument2( iface );
702 return IXMLDOMNode_get_specified( This->node, isSpecified );
706 static HRESULT WINAPI domdoc_get_definition(
707 IXMLDOMDocument2 *iface,
708 IXMLDOMNode** definitionNode )
710 domdoc *This = impl_from_IXMLDOMDocument2( iface );
711 return IXMLDOMNode_get_definition( This->node, definitionNode );
715 static HRESULT WINAPI domdoc_get_nodeTypedValue(
716 IXMLDOMDocument2 *iface,
717 VARIANT* typedValue )
719 domdoc *This = impl_from_IXMLDOMDocument2( iface );
720 return IXMLDOMNode_get_nodeTypedValue( This->node, typedValue );
723 static HRESULT WINAPI domdoc_put_nodeTypedValue(
724 IXMLDOMDocument2 *iface,
725 VARIANT typedValue )
727 domdoc *This = impl_from_IXMLDOMDocument2( iface );
728 return IXMLDOMNode_put_nodeTypedValue( This->node, typedValue );
732 static HRESULT WINAPI domdoc_get_dataType(
733 IXMLDOMDocument2 *iface,
734 VARIANT* dataTypeName )
736 domdoc *This = impl_from_IXMLDOMDocument2( iface );
737 return IXMLDOMNode_get_dataType( This->node, dataTypeName );
741 static HRESULT WINAPI domdoc_put_dataType(
742 IXMLDOMDocument2 *iface,
743 BSTR dataTypeName )
745 domdoc *This = impl_from_IXMLDOMDocument2( iface );
746 return IXMLDOMNode_put_dataType( This->node, dataTypeName );
750 static HRESULT WINAPI domdoc_get_xml(
751 IXMLDOMDocument2 *iface,
752 BSTR* xmlString )
754 domdoc *This = impl_from_IXMLDOMDocument2( iface );
755 return IXMLDOMNode_get_xml( This->node, xmlString );
759 static HRESULT WINAPI domdoc_transformNode(
760 IXMLDOMDocument2 *iface,
761 IXMLDOMNode* styleSheet,
762 BSTR* xmlString )
764 domdoc *This = impl_from_IXMLDOMDocument2( iface );
765 return IXMLDOMNode_transformNode( This->node, styleSheet, xmlString );
769 static HRESULT WINAPI domdoc_selectNodes(
770 IXMLDOMDocument2 *iface,
771 BSTR queryString,
772 IXMLDOMNodeList** resultList )
774 domdoc *This = impl_from_IXMLDOMDocument2( iface );
775 return IXMLDOMNode_selectNodes( This->node, queryString, resultList );
779 static HRESULT WINAPI domdoc_selectSingleNode(
780 IXMLDOMDocument2 *iface,
781 BSTR queryString,
782 IXMLDOMNode** resultNode )
784 domdoc *This = impl_from_IXMLDOMDocument2( iface );
785 return IXMLDOMNode_selectSingleNode( This->node, queryString, resultNode );
789 static HRESULT WINAPI domdoc_get_parsed(
790 IXMLDOMDocument2 *iface,
791 VARIANT_BOOL* isParsed )
793 domdoc *This = impl_from_IXMLDOMDocument2( iface );
794 return IXMLDOMNode_get_parsed( This->node, isParsed );
798 static HRESULT WINAPI domdoc_get_namespaceURI(
799 IXMLDOMDocument2 *iface,
800 BSTR* namespaceURI )
802 domdoc *This = impl_from_IXMLDOMDocument2( iface );
803 return IXMLDOMNode_get_namespaceURI( This->node, namespaceURI );
807 static HRESULT WINAPI domdoc_get_prefix(
808 IXMLDOMDocument2 *iface,
809 BSTR* prefixString )
811 domdoc *This = impl_from_IXMLDOMDocument2( iface );
812 return IXMLDOMNode_get_prefix( This->node, prefixString );
816 static HRESULT WINAPI domdoc_get_baseName(
817 IXMLDOMDocument2 *iface,
818 BSTR* nameString )
820 domdoc *This = impl_from_IXMLDOMDocument2( iface );
821 return IXMLDOMNode_get_baseName( This->node, nameString );
825 static HRESULT WINAPI domdoc_transformNodeToObject(
826 IXMLDOMDocument2 *iface,
827 IXMLDOMNode* stylesheet,
828 VARIANT outputObject)
830 domdoc *This = impl_from_IXMLDOMDocument2( iface );
831 return IXMLDOMNode_transformNodeToObject( This->node, stylesheet, outputObject );
835 static HRESULT WINAPI domdoc_get_doctype(
836 IXMLDOMDocument2 *iface,
837 IXMLDOMDocumentType** documentType )
839 FIXME("\n");
840 return E_NOTIMPL;
844 static HRESULT WINAPI domdoc_get_implementation(
845 IXMLDOMDocument2 *iface,
846 IXMLDOMImplementation** impl )
848 if(!impl)
849 return E_INVALIDARG;
851 *impl = (IXMLDOMImplementation*)create_doc_Implementation();
853 return S_OK;
856 static HRESULT WINAPI domdoc_get_documentElement(
857 IXMLDOMDocument2 *iface,
858 IXMLDOMElement** DOMElement )
860 domdoc *This = impl_from_IXMLDOMDocument2( iface );
861 xmlDocPtr xmldoc = NULL;
862 xmlNodePtr root = NULL;
863 IXMLDOMNode *element_node;
864 HRESULT hr;
866 TRACE("%p %p\n", This, This->node);
868 if(!DOMElement)
869 return E_INVALIDARG;
871 *DOMElement = NULL;
873 xmldoc = get_doc( This );
875 root = xmlDocGetRootElement( xmldoc );
876 if ( !root )
877 return S_FALSE;
879 element_node = create_node( root );
880 if(!element_node) return S_FALSE;
882 hr = IXMLDOMNode_QueryInterface(element_node, &IID_IXMLDOMElement, (LPVOID*)DOMElement);
883 IXMLDOMNode_Release(element_node);
885 return hr;
889 static HRESULT WINAPI domdoc_documentElement(
890 IXMLDOMDocument2 *iface,
891 IXMLDOMElement* DOMElement )
893 FIXME("\n");
894 return E_NOTIMPL;
898 static HRESULT WINAPI domdoc_createElement(
899 IXMLDOMDocument2 *iface,
900 BSTR tagname,
901 IXMLDOMElement** element )
903 xmlNodePtr xmlnode;
904 domdoc *This = impl_from_IXMLDOMDocument2( iface );
905 xmlChar *xml_name;
906 IUnknown *elem_unk;
907 HRESULT hr;
909 TRACE("%p->(%s,%p)\n", iface, debugstr_w(tagname), element);
911 xml_name = xmlChar_from_wchar((WCHAR*)tagname);
912 xmlnode = xmlNewDocNode(get_doc(This), NULL, xml_name, NULL);
914 TRACE("created xmlptr %p\n", xmlnode);
915 elem_unk = create_element(xmlnode, NULL);
916 HeapFree(GetProcessHeap(), 0, xml_name);
918 hr = IUnknown_QueryInterface(elem_unk, &IID_IXMLDOMElement, (void **)element);
919 IUnknown_Release(elem_unk);
920 TRACE("returning %p\n", *element);
921 return hr;
925 static HRESULT WINAPI domdoc_createDocumentFragment(
926 IXMLDOMDocument2 *iface,
927 IXMLDOMDocumentFragment** docFrag )
929 domdoc *This = impl_from_IXMLDOMDocument2( iface );
930 xmlNodePtr xmlnode;
932 TRACE("%p\n", iface);
934 if(!docFrag)
935 return E_INVALIDARG;
937 *docFrag = NULL;
939 xmlnode = xmlNewDocFragment(get_doc( This ) );
941 if(!xmlnode)
942 return E_FAIL;
944 xmlnode->doc = get_doc( This );
946 *docFrag = (IXMLDOMDocumentFragment*)create_doc_fragment(xmlnode);
948 return S_OK;
952 static HRESULT WINAPI domdoc_createTextNode(
953 IXMLDOMDocument2 *iface,
954 BSTR data,
955 IXMLDOMText** text )
957 domdoc *This = impl_from_IXMLDOMDocument2( iface );
958 xmlNodePtr xmlnode;
959 xmlChar *xml_content;
961 TRACE("%p->(%s %p)\n", iface, debugstr_w(data), text);
963 if(!text)
964 return E_INVALIDARG;
966 *text = NULL;
968 xml_content = xmlChar_from_wchar((WCHAR*)data);
969 xmlnode = xmlNewText(xml_content);
970 HeapFree(GetProcessHeap(), 0, xml_content);
972 if(!xmlnode)
973 return E_FAIL;
975 xmlnode->doc = get_doc( This );
977 *text = (IXMLDOMText*)create_text(xmlnode);
979 return S_OK;
983 static HRESULT WINAPI domdoc_createComment(
984 IXMLDOMDocument2 *iface,
985 BSTR data,
986 IXMLDOMComment** comment )
988 domdoc *This = impl_from_IXMLDOMDocument2( iface );
989 xmlNodePtr xmlnode;
990 xmlChar *xml_content;
992 TRACE("%p->(%s %p)\n", iface, debugstr_w(data), comment);
994 if(!comment)
995 return E_INVALIDARG;
997 *comment = NULL;
999 xml_content = xmlChar_from_wchar((WCHAR*)data);
1000 xmlnode = xmlNewComment(xml_content);
1001 HeapFree(GetProcessHeap(), 0, xml_content);
1003 if(!xmlnode)
1004 return E_FAIL;
1006 xmlnode->doc = get_doc( This );
1008 *comment = (IXMLDOMComment*)create_comment(xmlnode);
1010 return S_OK;
1014 static HRESULT WINAPI domdoc_createCDATASection(
1015 IXMLDOMDocument2 *iface,
1016 BSTR data,
1017 IXMLDOMCDATASection** cdata )
1019 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1020 xmlNodePtr xmlnode;
1021 xmlChar *xml_content;
1023 TRACE("%p->(%s %p)\n", iface, debugstr_w(data), comment);
1025 if(!cdata)
1026 return E_INVALIDARG;
1028 *cdata = NULL;
1030 xml_content = xmlChar_from_wchar((WCHAR*)data);
1031 xmlnode = xmlNewCDataBlock(get_doc( This ), xml_content, strlen( (char*)xml_content) );
1032 HeapFree(GetProcessHeap(), 0, xml_content);
1034 if(!xmlnode)
1035 return E_FAIL;
1037 xmlnode->doc = get_doc( This );
1039 *cdata = (IXMLDOMCDATASection*)create_cdata(xmlnode);
1041 return S_OK;
1045 static HRESULT WINAPI domdoc_createProcessingInstruction(
1046 IXMLDOMDocument2 *iface,
1047 BSTR target,
1048 BSTR data,
1049 IXMLDOMProcessingInstruction** pi )
1051 #ifdef HAVE_XMLNEWDOCPI
1052 xmlNodePtr xmlnode;
1053 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1054 xmlChar *xml_target, *xml_content;
1056 TRACE("%p->(%s %s %p)\n", iface, debugstr_w(target), debugstr_w(data), pi);
1058 if(!pi)
1059 return E_INVALIDARG;
1061 if(!target || lstrlenW(target) == 0)
1062 return E_FAIL;
1064 xml_target = xmlChar_from_wchar((WCHAR*)target);
1065 xml_content = xmlChar_from_wchar((WCHAR*)data);
1067 xmlnode = xmlNewDocPI(get_doc(This), xml_target, xml_content);
1068 TRACE("created xmlptr %p\n", xmlnode);
1069 *pi = (IXMLDOMProcessingInstruction*)create_pi(xmlnode);
1071 HeapFree(GetProcessHeap(), 0, xml_content);
1072 HeapFree(GetProcessHeap(), 0, xml_target);
1074 return S_OK;
1075 #else
1076 FIXME("Libxml 2.6.15 or greater required.\n");
1077 return E_NOTIMPL;
1078 #endif
1082 static HRESULT WINAPI domdoc_createAttribute(
1083 IXMLDOMDocument2 *iface,
1084 BSTR name,
1085 IXMLDOMAttribute** attribute )
1087 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1088 xmlNodePtr xmlnode;
1089 xmlChar *xml_name;
1091 TRACE("%p->(%s %p)\n", iface, debugstr_w(name), attribute);
1093 if(!attribute)
1094 return E_INVALIDARG;
1096 *attribute = NULL;
1098 xml_name = xmlChar_from_wchar((WCHAR*)name);
1099 xmlnode = (xmlNode *)xmlNewProp(NULL, xml_name, NULL);
1100 HeapFree(GetProcessHeap(), 0, xml_name);
1102 if(!xmlnode)
1103 return E_FAIL;
1105 xmlnode->doc = get_doc( This );
1107 *attribute = (IXMLDOMAttribute*)create_attribute(xmlnode);
1109 return S_OK;
1113 static HRESULT WINAPI domdoc_createEntityReference(
1114 IXMLDOMDocument2 *iface,
1115 BSTR name,
1116 IXMLDOMEntityReference** entityRef )
1118 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1119 xmlNodePtr xmlnode;
1120 xmlChar *xml_name;
1122 TRACE("%p\n", iface);
1124 if(!entityRef)
1125 return E_INVALIDARG;
1127 *entityRef = NULL;
1129 xml_name = xmlChar_from_wchar((WCHAR*)name);
1130 xmlnode = xmlNewReference(get_doc( This ), xml_name );
1131 HeapFree(GetProcessHeap(), 0, xml_name);
1133 if(!xmlnode)
1134 return E_FAIL;
1136 xmlnode->doc = get_doc( This );
1138 *entityRef = (IXMLDOMEntityReference*)create_doc_entity_ref(xmlnode);
1140 return S_OK;
1144 static HRESULT WINAPI domdoc_getElementsByTagName(
1145 IXMLDOMDocument2 *iface,
1146 BSTR tagName,
1147 IXMLDOMNodeList** resultList )
1149 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1150 LPWSTR szPattern;
1151 HRESULT hr;
1152 TRACE("(%p)->(%s, %p)\n", This, debugstr_w(tagName), resultList);
1154 szPattern = HeapAlloc(GetProcessHeap(), 0, sizeof(WCHAR)*(2+lstrlenW(tagName)+1));
1155 szPattern[0] = szPattern[1] = '/';
1156 lstrcpyW(szPattern + 2, tagName);
1158 hr = queryresult_create((xmlNodePtr)get_doc(This), szPattern, resultList);
1159 HeapFree(GetProcessHeap(), 0, szPattern);
1161 return hr;
1164 static DOMNodeType get_node_type(VARIANT Type)
1166 if(V_VT(&Type) == VT_I4)
1167 return V_I4(&Type);
1169 FIXME("Unsupported variant type %x\n", V_VT(&Type));
1170 return 0;
1173 static HRESULT WINAPI domdoc_createNode(
1174 IXMLDOMDocument2 *iface,
1175 VARIANT Type,
1176 BSTR name,
1177 BSTR namespaceURI,
1178 IXMLDOMNode** node )
1180 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1181 DOMNodeType node_type;
1182 xmlNodePtr xmlnode = NULL;
1183 xmlChar *xml_name;
1185 TRACE("(%p)->(type,%s,%s,%p)\n", This, debugstr_w(name), debugstr_w(namespaceURI), node);
1187 node_type = get_node_type(Type);
1188 TRACE("node_type %d\n", node_type);
1190 xml_name = xmlChar_from_wchar((WCHAR*)name);
1192 switch(node_type)
1194 case NODE_ELEMENT:
1195 xmlnode = xmlNewDocNode(get_doc(This), NULL, xml_name, NULL);
1196 *node = create_node(xmlnode);
1197 TRACE("created %p\n", xmlnode);
1198 break;
1200 default:
1201 FIXME("unhandled node type %d\n", node_type);
1202 break;
1205 HeapFree(GetProcessHeap(), 0, xml_name);
1207 if(xmlnode && *node)
1208 return S_OK;
1210 return E_FAIL;
1213 static HRESULT WINAPI domdoc_nodeFromID(
1214 IXMLDOMDocument2 *iface,
1215 BSTR idString,
1216 IXMLDOMNode** node )
1218 FIXME("\n");
1219 return E_NOTIMPL;
1222 static xmlDocPtr doparse( char *ptr, int len )
1224 #ifdef HAVE_XMLREADMEMORY
1226 * use xmlReadMemory if possible so we can suppress
1227 * writing errors to stderr
1229 return xmlReadMemory( ptr, len, NULL, NULL,
1230 XML_PARSE_NOERROR | XML_PARSE_NOWARNING | XML_PARSE_NOBLANKS );
1231 #else
1232 return xmlParseMemory( ptr, len );
1233 #endif
1236 static xmlDocPtr doread( LPWSTR filename )
1238 xmlDocPtr xmldoc = NULL;
1239 HRESULT hr;
1240 IBindCtx *pbc;
1241 IStream *stream, *memstream;
1242 WCHAR url[INTERNET_MAX_URL_LENGTH];
1243 BYTE buf[4096];
1244 DWORD read, written;
1246 TRACE("%s\n", debugstr_w( filename ));
1248 if(!PathIsURLW(filename))
1250 WCHAR fullpath[MAX_PATH];
1251 DWORD needed = sizeof(url)/sizeof(WCHAR);
1253 if(!PathSearchAndQualifyW(filename, fullpath, sizeof(fullpath)/sizeof(WCHAR)))
1255 WARN("can't find path\n");
1256 return NULL;
1259 if(FAILED(UrlCreateFromPathW(fullpath, url, &needed, 0)))
1261 ERR("can't create url from path\n");
1262 return NULL;
1264 filename = url;
1267 hr = CreateBindCtx(0, &pbc);
1268 if(SUCCEEDED(hr))
1270 hr = RegisterBindStatusCallback(pbc, (IBindStatusCallback*)&domdoc_bsc.lpVtbl, NULL, 0);
1271 if(SUCCEEDED(hr))
1273 IMoniker *moniker;
1274 hr = CreateURLMoniker(NULL, filename, &moniker);
1275 if(SUCCEEDED(hr))
1277 hr = IMoniker_BindToStorage(moniker, pbc, NULL, &IID_IStream, (LPVOID*)&stream);
1278 IMoniker_Release(moniker);
1281 IBindCtx_Release(pbc);
1283 if(FAILED(hr))
1284 return NULL;
1286 hr = CreateStreamOnHGlobal(NULL, TRUE, &memstream);
1287 if(FAILED(hr))
1289 IStream_Release(stream);
1290 return NULL;
1295 IStream_Read(stream, buf, sizeof(buf), &read);
1296 hr = IStream_Write(memstream, buf, read, &written);
1297 } while(SUCCEEDED(hr) && written != 0 && read != 0);
1299 if(SUCCEEDED(hr))
1301 HGLOBAL hglobal;
1302 hr = GetHGlobalFromStream(memstream, &hglobal);
1303 if(SUCCEEDED(hr))
1305 DWORD len = GlobalSize(hglobal);
1306 char *ptr = GlobalLock(hglobal);
1307 if(len != 0)
1308 xmldoc = doparse( ptr, len );
1309 GlobalUnlock(hglobal);
1312 IStream_Release(memstream);
1313 IStream_Release(stream);
1314 return xmldoc;
1317 static HRESULT WINAPI domdoc_load(
1318 IXMLDOMDocument2 *iface,
1319 VARIANT xmlSource,
1320 VARIANT_BOOL* isSuccessful )
1322 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1323 LPWSTR filename = NULL;
1324 xmlDocPtr xmldoc = NULL;
1325 HRESULT hr = S_FALSE;
1326 IXMLDOMDocument2 *pNewDoc = NULL;
1327 IStream *pStream = NULL;
1329 TRACE("type %d\n", V_VT(&xmlSource) );
1331 *isSuccessful = VARIANT_FALSE;
1333 assert( This->node );
1335 attach_xmlnode(This->node, NULL);
1337 switch( V_VT(&xmlSource) )
1339 case VT_BSTR:
1340 filename = V_BSTR(&xmlSource);
1341 break;
1342 case VT_UNKNOWN:
1343 hr = IUnknown_QueryInterface(V_UNKNOWN(&xmlSource), &IID_IXMLDOMDocument2, (void**)&pNewDoc);
1344 if(hr == S_OK)
1346 if(pNewDoc)
1348 domdoc *newDoc = impl_from_IXMLDOMDocument2( pNewDoc );
1349 xmldoc = xmlCopyDoc(get_doc(newDoc), 1);
1350 attach_xmlnode(This->node, (xmlNodePtr) xmldoc);
1352 *isSuccessful = VARIANT_TRUE;
1354 return S_OK;
1357 hr = IUnknown_QueryInterface(V_UNKNOWN(&xmlSource), &IID_IStream, (void**)&pStream);
1358 if(hr == S_OK)
1360 IPersistStream *pDocStream;
1361 hr = IUnknown_QueryInterface(iface, &IID_IPersistStream, (void**)&pDocStream);
1362 if(hr == S_OK)
1364 hr = xmldoc_IPersistStream_Load(pDocStream, pStream);
1365 IStream_Release(pStream);
1366 if(hr == S_OK)
1368 *isSuccessful = VARIANT_TRUE;
1370 TRACE("Using ID_IStream to load Document\n");
1371 return S_OK;
1373 else
1375 ERR("xmldoc_IPersistStream_Load failed (%d)\n", hr);
1378 else
1380 ERR("QueryInterface IID_IPersistStream failed (%d)\n", hr);
1383 else
1385 /* ISequentialStream */
1386 FIXME("Unknown type not supported (%d) (%p)(%p)\n", hr, pNewDoc, V_UNKNOWN(&xmlSource)->lpVtbl);
1388 break;
1389 default:
1390 FIXME("VT type not supported (%d)\n", V_VT(&xmlSource));
1393 TRACE("filename (%s)\n", debugstr_w(filename));
1395 if ( filename )
1397 xmldoc = doread( filename );
1399 if ( !xmldoc )
1400 This->error = E_FAIL;
1401 else
1403 hr = This->error = S_OK;
1404 *isSuccessful = VARIANT_TRUE;
1408 if(!xmldoc)
1409 xmldoc = xmlNewDoc(NULL);
1411 xmldoc->_private = 0;
1412 attach_xmlnode(This->node, (xmlNodePtr) xmldoc);
1414 TRACE("ret (%d)\n", hr);
1416 return hr;
1420 static HRESULT WINAPI domdoc_get_readyState(
1421 IXMLDOMDocument2 *iface,
1422 long* value )
1424 FIXME("\n");
1425 return E_NOTIMPL;
1429 static HRESULT WINAPI domdoc_get_parseError(
1430 IXMLDOMDocument2 *iface,
1431 IXMLDOMParseError** errorObj )
1433 BSTR error_string = NULL;
1434 static const WCHAR err[] = {'e','r','r','o','r',0};
1435 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1437 FIXME("(%p)->(%p): creating a dummy parseError\n", iface, errorObj);
1439 if(This->error)
1440 error_string = SysAllocString(err);
1442 *errorObj = create_parseError(This->error, NULL, error_string, NULL, 0, 0, 0);
1443 if(!*errorObj) return E_OUTOFMEMORY;
1444 return S_OK;
1448 static HRESULT WINAPI domdoc_get_url(
1449 IXMLDOMDocument2 *iface,
1450 BSTR* urlString )
1452 FIXME("\n");
1453 return E_NOTIMPL;
1457 static HRESULT WINAPI domdoc_get_async(
1458 IXMLDOMDocument2 *iface,
1459 VARIANT_BOOL* isAsync )
1461 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1463 TRACE("%p <- %d\n", isAsync, This->async);
1464 *isAsync = This->async;
1465 return S_OK;
1469 static HRESULT WINAPI domdoc_put_async(
1470 IXMLDOMDocument2 *iface,
1471 VARIANT_BOOL isAsync )
1473 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1475 TRACE("%d\n", isAsync);
1476 This->async = isAsync;
1477 return S_OK;
1481 static HRESULT WINAPI domdoc_abort(
1482 IXMLDOMDocument2 *iface )
1484 FIXME("\n");
1485 return E_NOTIMPL;
1489 static BOOL bstr_to_utf8( BSTR bstr, char **pstr, int *plen )
1491 UINT len, blen = SysStringLen( bstr );
1492 LPSTR str;
1494 len = WideCharToMultiByte( CP_UTF8, 0, bstr, blen, NULL, 0, NULL, NULL );
1495 str = HeapAlloc( GetProcessHeap(), 0, len );
1496 if ( !str )
1497 return FALSE;
1498 WideCharToMultiByte( CP_UTF8, 0, bstr, blen, str, len, NULL, NULL );
1499 *plen = len;
1500 *pstr = str;
1501 return TRUE;
1504 static HRESULT WINAPI domdoc_loadXML(
1505 IXMLDOMDocument2 *iface,
1506 BSTR bstrXML,
1507 VARIANT_BOOL* isSuccessful )
1509 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1510 xmlDocPtr xmldoc = NULL;
1511 char *str;
1512 int len;
1513 HRESULT hr = S_FALSE;
1515 TRACE("%p %s %p\n", This, debugstr_w( bstrXML ), isSuccessful );
1517 assert ( This->node );
1519 attach_xmlnode( This->node, NULL );
1521 if ( isSuccessful )
1523 *isSuccessful = VARIANT_FALSE;
1525 if ( bstrXML && bstr_to_utf8( bstrXML, &str, &len ) )
1527 xmldoc = doparse( str, len );
1528 HeapFree( GetProcessHeap(), 0, str );
1529 if ( !xmldoc )
1530 This->error = E_FAIL;
1531 else
1533 hr = This->error = S_OK;
1534 *isSuccessful = VARIANT_TRUE;
1538 if(!xmldoc)
1539 xmldoc = xmlNewDoc(NULL);
1541 xmldoc->_private = 0;
1542 attach_xmlnode( This->node, (xmlNodePtr) xmldoc );
1544 return hr;
1548 static HRESULT WINAPI domdoc_save(
1549 IXMLDOMDocument2 *iface,
1550 VARIANT destination )
1552 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1553 HANDLE handle;
1554 xmlChar *mem;
1555 int size;
1556 HRESULT ret = S_OK;
1557 DWORD written;
1559 TRACE("(%p)->(var(vt %x, %s))\n", This, V_VT(&destination),
1560 V_VT(&destination) == VT_BSTR ? debugstr_w(V_BSTR(&destination)) : NULL);
1562 if(V_VT(&destination) != VT_BSTR)
1564 FIXME("Unhandled vt %x\n", V_VT(&destination));
1565 return S_FALSE;
1568 handle = CreateFileW( V_BSTR(&destination), GENERIC_WRITE, 0,
1569 NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL );
1570 if( handle == INVALID_HANDLE_VALUE )
1572 WARN("failed to create file\n");
1573 return S_FALSE;
1576 xmlDocDumpMemory(get_doc(This), &mem, &size);
1577 if(!WriteFile(handle, mem, (DWORD)size, &written, NULL) || written != (DWORD)size)
1579 WARN("write error\n");
1580 ret = S_FALSE;
1583 xmlFree(mem);
1584 CloseHandle(handle);
1585 return ret;
1588 static HRESULT WINAPI domdoc_get_validateOnParse(
1589 IXMLDOMDocument2 *iface,
1590 VARIANT_BOOL* isValidating )
1592 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1594 TRACE("%p <- %d\n", isValidating, This->validating);
1595 *isValidating = This->validating;
1596 return S_OK;
1600 static HRESULT WINAPI domdoc_put_validateOnParse(
1601 IXMLDOMDocument2 *iface,
1602 VARIANT_BOOL isValidating )
1604 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1606 TRACE("%d\n", isValidating);
1607 This->validating = isValidating;
1608 return S_OK;
1612 static HRESULT WINAPI domdoc_get_resolveExternals(
1613 IXMLDOMDocument2 *iface,
1614 VARIANT_BOOL* isResolving )
1616 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1618 TRACE("%p <- %d\n", isResolving, This->resolving);
1619 *isResolving = This->resolving;
1620 return S_OK;
1624 static HRESULT WINAPI domdoc_put_resolveExternals(
1625 IXMLDOMDocument2 *iface,
1626 VARIANT_BOOL isResolving )
1628 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1630 TRACE("%d\n", isResolving);
1631 This->resolving = isResolving;
1632 return S_OK;
1636 static HRESULT WINAPI domdoc_get_preserveWhiteSpace(
1637 IXMLDOMDocument2 *iface,
1638 VARIANT_BOOL* isPreserving )
1640 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1642 TRACE("%p <- %d\n", isPreserving, This->preserving);
1643 *isPreserving = This->preserving;
1644 return S_OK;
1648 static HRESULT WINAPI domdoc_put_preserveWhiteSpace(
1649 IXMLDOMDocument2 *iface,
1650 VARIANT_BOOL isPreserving )
1652 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1654 TRACE("%d\n", isPreserving);
1655 This->preserving = isPreserving;
1656 return S_OK;
1660 static HRESULT WINAPI domdoc_put_onReadyStateChange(
1661 IXMLDOMDocument2 *iface,
1662 VARIANT readyStateChangeSink )
1664 FIXME("\n");
1665 return E_NOTIMPL;
1669 static HRESULT WINAPI domdoc_put_onDataAvailable(
1670 IXMLDOMDocument2 *iface,
1671 VARIANT onDataAvailableSink )
1673 FIXME("\n");
1674 return E_NOTIMPL;
1677 static HRESULT WINAPI domdoc_put_onTransformNode(
1678 IXMLDOMDocument2 *iface,
1679 VARIANT onTransformNodeSink )
1681 FIXME("\n");
1682 return E_NOTIMPL;
1685 static HRESULT WINAPI domdoc_get_namespaces(
1686 IXMLDOMDocument2* iface,
1687 IXMLDOMSchemaCollection** schemaCollection )
1689 FIXME("\n");
1690 return E_NOTIMPL;
1693 static HRESULT WINAPI domdoc_get_schemas(
1694 IXMLDOMDocument2* iface,
1695 VARIANT* var1 )
1697 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1698 HRESULT hr = S_FALSE;
1699 IXMLDOMSchemaCollection *cur_schema = This->schema;
1701 TRACE("(%p)->(%p)\n", This, var1);
1703 VariantInit(var1); /* Test shows we don't call VariantClear here */
1704 V_VT(var1) = VT_NULL;
1706 if(cur_schema)
1708 hr = IXMLDOMSchemaCollection_QueryInterface(cur_schema, &IID_IDispatch, (void**)&V_DISPATCH(var1));
1709 if(SUCCEEDED(hr))
1710 V_VT(var1) = VT_DISPATCH;
1712 return hr;
1715 static HRESULT WINAPI domdoc_putref_schemas(
1716 IXMLDOMDocument2* iface,
1717 VARIANT var1)
1719 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1720 HRESULT hr = E_FAIL;
1721 IXMLDOMSchemaCollection *new_schema = NULL;
1723 FIXME("(%p): semi-stub\n", This);
1724 switch(V_VT(&var1))
1726 case VT_UNKNOWN:
1727 hr = IUnknown_QueryInterface(V_UNKNOWN(&var1), &IID_IXMLDOMSchemaCollection, (void**)&new_schema);
1728 break;
1730 case VT_DISPATCH:
1731 hr = IDispatch_QueryInterface(V_DISPATCH(&var1), &IID_IXMLDOMSchemaCollection, (void**)&new_schema);
1732 break;
1734 case VT_NULL:
1735 case VT_EMPTY:
1736 hr = S_OK;
1737 break;
1739 default:
1740 WARN("Can't get schema from vt %x\n", V_VT(&var1));
1743 if(SUCCEEDED(hr))
1745 IXMLDOMSchemaCollection *old_schema = InterlockedExchangePointer((void**)&This->schema, new_schema);
1746 if(old_schema) IXMLDOMSchemaCollection_Release(old_schema);
1749 return hr;
1752 static HRESULT WINAPI domdoc_validate(
1753 IXMLDOMDocument2* iface,
1754 IXMLDOMParseError** err)
1756 FIXME("\n");
1757 return E_NOTIMPL;
1760 static HRESULT WINAPI domdoc_setProperty(
1761 IXMLDOMDocument2* iface,
1762 BSTR p,
1763 VARIANT var)
1765 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1767 if (lstrcmpiW(p, SZ_PROPERTY_SELECTION_LANGUAGE) == 0)
1769 VARIANT varStr;
1770 HRESULT hr;
1771 BSTR bstr;
1773 V_VT(&varStr) = VT_EMPTY;
1774 if (V_VT(&var) != VT_BSTR)
1776 if (FAILED(hr = VariantChangeType(&varStr, &var, 0, VT_BSTR)))
1777 return hr;
1778 bstr = V_BSTR(&varStr);
1780 else
1781 bstr = V_BSTR(&var);
1783 hr = S_OK;
1784 if (lstrcmpiW(bstr, SZ_VALUE_XPATH) == 0)
1785 This->bUseXPath = TRUE;
1786 else if (lstrcmpiW(bstr, SZ_VALUE_XSLPATTERN) == 0)
1787 This->bUseXPath = FALSE;
1788 else
1789 hr = E_FAIL;
1791 VariantClear(&varStr);
1792 return hr;
1795 FIXME("Unknown property %s\n", wine_dbgstr_w(p));
1796 return E_FAIL;
1799 static HRESULT WINAPI domdoc_getProperty(
1800 IXMLDOMDocument2* iface,
1801 BSTR p,
1802 VARIANT* var)
1804 domdoc *This = impl_from_IXMLDOMDocument2( iface );
1806 if (var == NULL)
1807 return E_INVALIDARG;
1808 if (lstrcmpiW(p, SZ_PROPERTY_SELECTION_LANGUAGE) == 0)
1810 V_VT(var) = VT_BSTR;
1811 if (This->bUseXPath)
1812 V_BSTR(var) = SysAllocString(SZ_VALUE_XPATH);
1813 else
1814 V_BSTR(var) = SysAllocString(SZ_VALUE_XSLPATTERN);
1815 return S_OK;
1818 FIXME("Unknown property %s\n", wine_dbgstr_w(p));
1819 return E_FAIL;
1822 static const struct IXMLDOMDocument2Vtbl domdoc_vtbl =
1824 domdoc_QueryInterface,
1825 domdoc_AddRef,
1826 domdoc_Release,
1827 domdoc_GetTypeInfoCount,
1828 domdoc_GetTypeInfo,
1829 domdoc_GetIDsOfNames,
1830 domdoc_Invoke,
1831 domdoc_get_nodeName,
1832 domdoc_get_nodeValue,
1833 domdoc_put_nodeValue,
1834 domdoc_get_nodeType,
1835 domdoc_get_parentNode,
1836 domdoc_get_childNodes,
1837 domdoc_get_firstChild,
1838 domdoc_get_lastChild,
1839 domdoc_get_previousSibling,
1840 domdoc_get_nextSibling,
1841 domdoc_get_attributes,
1842 domdoc_insertBefore,
1843 domdoc_replaceChild,
1844 domdoc_removeChild,
1845 domdoc_appendChild,
1846 domdoc_hasChildNodes,
1847 domdoc_get_ownerDocument,
1848 domdoc_cloneNode,
1849 domdoc_get_nodeTypeString,
1850 domdoc_get_text,
1851 domdoc_put_text,
1852 domdoc_get_specified,
1853 domdoc_get_definition,
1854 domdoc_get_nodeTypedValue,
1855 domdoc_put_nodeTypedValue,
1856 domdoc_get_dataType,
1857 domdoc_put_dataType,
1858 domdoc_get_xml,
1859 domdoc_transformNode,
1860 domdoc_selectNodes,
1861 domdoc_selectSingleNode,
1862 domdoc_get_parsed,
1863 domdoc_get_namespaceURI,
1864 domdoc_get_prefix,
1865 domdoc_get_baseName,
1866 domdoc_transformNodeToObject,
1867 domdoc_get_doctype,
1868 domdoc_get_implementation,
1869 domdoc_get_documentElement,
1870 domdoc_documentElement,
1871 domdoc_createElement,
1872 domdoc_createDocumentFragment,
1873 domdoc_createTextNode,
1874 domdoc_createComment,
1875 domdoc_createCDATASection,
1876 domdoc_createProcessingInstruction,
1877 domdoc_createAttribute,
1878 domdoc_createEntityReference,
1879 domdoc_getElementsByTagName,
1880 domdoc_createNode,
1881 domdoc_nodeFromID,
1882 domdoc_load,
1883 domdoc_get_readyState,
1884 domdoc_get_parseError,
1885 domdoc_get_url,
1886 domdoc_get_async,
1887 domdoc_put_async,
1888 domdoc_abort,
1889 domdoc_loadXML,
1890 domdoc_save,
1891 domdoc_get_validateOnParse,
1892 domdoc_put_validateOnParse,
1893 domdoc_get_resolveExternals,
1894 domdoc_put_resolveExternals,
1895 domdoc_get_preserveWhiteSpace,
1896 domdoc_put_preserveWhiteSpace,
1897 domdoc_put_onReadyStateChange,
1898 domdoc_put_onDataAvailable,
1899 domdoc_put_onTransformNode,
1900 domdoc_get_namespaces,
1901 domdoc_get_schemas,
1902 domdoc_putref_schemas,
1903 domdoc_validate,
1904 domdoc_setProperty,
1905 domdoc_getProperty
1908 HRESULT DOMDocument_create(IUnknown *pUnkOuter, LPVOID *ppObj)
1910 domdoc *doc;
1911 HRESULT hr;
1912 xmlDocPtr xmldoc;
1914 TRACE("(%p,%p)\n", pUnkOuter, ppObj);
1916 doc = HeapAlloc( GetProcessHeap(), 0, sizeof (*doc) );
1917 if( !doc )
1918 return E_OUTOFMEMORY;
1920 doc->lpVtbl = &domdoc_vtbl;
1921 doc->lpvtblIPersistStream = &xmldoc_IPersistStream_VTable;
1922 doc->ref = 1;
1923 doc->async = 0;
1924 doc->validating = 0;
1925 doc->resolving = 0;
1926 doc->preserving = 0;
1927 doc->bUseXPath = FALSE;
1928 doc->error = S_OK;
1929 doc->schema = NULL;
1930 doc->stream = NULL;
1932 xmldoc = xmlNewDoc(NULL);
1933 if(!xmldoc)
1935 HeapFree(GetProcessHeap(), 0, doc);
1936 return E_OUTOFMEMORY;
1939 xmldoc->_private = 0;
1941 doc->node_unk = create_basic_node( (xmlNodePtr)xmldoc, (IUnknown*)&doc->lpVtbl );
1942 if(!doc->node_unk)
1944 xmlFreeDoc(xmldoc);
1945 HeapFree(GetProcessHeap(), 0, doc);
1946 return E_FAIL;
1949 hr = IUnknown_QueryInterface(doc->node_unk, &IID_IXMLDOMNode, (LPVOID*)&doc->node);
1950 if(FAILED(hr))
1952 IUnknown_Release(doc->node_unk);
1953 HeapFree( GetProcessHeap(), 0, doc );
1954 return E_FAIL;
1956 /* The ref on doc->node is actually looped back into this object, so release it */
1957 IXMLDOMNode_Release(doc->node);
1959 *ppObj = &doc->lpVtbl;
1961 TRACE("returning iface %p\n", *ppObj);
1962 return S_OK;
1965 #else
1967 HRESULT DOMDocument_create(IUnknown *pUnkOuter, LPVOID *ppObj)
1969 MESSAGE("This program tried to use a DOMDocument object, but\n"
1970 "libxml2 support was not present at compile time.\n");
1971 return E_NOTIMPL;
1974 #endif