2 * Copyright 2006 Jacek Caban for CodeWeavers
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Lesser General Public
6 * License as published by the Free Software Foundation; either
7 * version 2.1 of the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Lesser General Public License for more details.
14 * You should have received a copy of the GNU Lesser General Public
15 * License along with this library; if not, write to the Free Software
16 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
19 #include "mshtml_private.h"
21 static HTMLDOMNode
*get_node_obj(IHTMLDOMNode
*);
22 static HRESULT
create_node(HTMLDocumentNode
*,nsIDOMNode
*,HTMLDOMNode
**);
24 static ExternalCycleCollectionParticipant node_ccp
;
28 IHTMLDOMChildrenCollection IHTMLDOMChildrenCollection_iface
;
32 /* FIXME: implement weak reference */
33 HTMLDocumentNode
*doc
;
35 nsIDOMNodeList
*nslist
;
36 } HTMLDOMChildrenCollection
;
39 IEnumVARIANT IEnumVARIANT_iface
;
44 HTMLDOMChildrenCollection
*col
;
45 } HTMLDOMChildrenCollectionEnum
;
47 static inline HTMLDOMChildrenCollectionEnum
*impl_from_IEnumVARIANT(IEnumVARIANT
*iface
)
49 return CONTAINING_RECORD(iface
, HTMLDOMChildrenCollectionEnum
, IEnumVARIANT_iface
);
52 static HRESULT WINAPI
HTMLDOMChildrenCollectionEnum_QueryInterface(IEnumVARIANT
*iface
, REFIID riid
, void **ppv
)
54 HTMLDOMChildrenCollectionEnum
*This
= impl_from_IEnumVARIANT(iface
);
56 TRACE("(%p)->(%s %p)\n", This
, debugstr_mshtml_guid(riid
), ppv
);
58 if(IsEqualGUID(riid
, &IID_IUnknown
)) {
59 *ppv
= &This
->IEnumVARIANT_iface
;
60 }else if(IsEqualGUID(riid
, &IID_IEnumVARIANT
)) {
61 *ppv
= &This
->IEnumVARIANT_iface
;
63 FIXME("(%p)->(%s %p)\n", This
, debugstr_mshtml_guid(riid
), ppv
);
68 IUnknown_AddRef((IUnknown
*)*ppv
);
72 static ULONG WINAPI
HTMLDOMChildrenCollectionEnum_AddRef(IEnumVARIANT
*iface
)
74 HTMLDOMChildrenCollectionEnum
*This
= impl_from_IEnumVARIANT(iface
);
75 LONG ref
= InterlockedIncrement(&This
->ref
);
77 TRACE("(%p) ref=%d\n", This
, ref
);
82 static ULONG WINAPI
HTMLDOMChildrenCollectionEnum_Release(IEnumVARIANT
*iface
)
84 HTMLDOMChildrenCollectionEnum
*This
= impl_from_IEnumVARIANT(iface
);
85 LONG ref
= InterlockedDecrement(&This
->ref
);
87 TRACE("(%p) ref=%d\n", This
, ref
);
90 IHTMLDOMChildrenCollection_Release(&This
->col
->IHTMLDOMChildrenCollection_iface
);
97 static ULONG
get_enum_len(HTMLDOMChildrenCollectionEnum
*This
)
102 nsres
= nsIDOMNodeList_GetLength(This
->col
->nslist
, &len
);
103 assert(nsres
== NS_OK
);
107 static HRESULT WINAPI
HTMLDOMChildrenCollectionEnum_Next(IEnumVARIANT
*iface
, ULONG celt
, VARIANT
*rgVar
, ULONG
*pCeltFetched
)
109 HTMLDOMChildrenCollectionEnum
*This
= impl_from_IEnumVARIANT(iface
);
110 ULONG fetched
= 0, len
;
116 TRACE("(%p)->(%d %p %p)\n", This
, celt
, rgVar
, pCeltFetched
);
118 len
= get_enum_len(This
);
120 while(This
->iter
+fetched
< len
&& fetched
< celt
) {
121 nsres
= nsIDOMNodeList_Item(This
->col
->nslist
, This
->iter
+fetched
, &nsnode
);
122 assert(nsres
== NS_OK
);
124 hres
= get_node(This
->col
->doc
, nsnode
, TRUE
, &node
);
125 nsIDOMNode_Release(nsnode
);
127 ERR("get_node failed: %08x\n", hres
);
131 V_VT(rgVar
+fetched
) = VT_DISPATCH
;
132 IHTMLDOMNode_AddRef(&node
->IHTMLDOMNode_iface
);
133 V_DISPATCH(rgVar
+fetched
) = (IDispatch
*)&node
->IHTMLDOMNode_iface
;
137 This
->iter
+= fetched
;
139 *pCeltFetched
= fetched
;
140 return fetched
== celt
? S_OK
: S_FALSE
;
143 static HRESULT WINAPI
HTMLDOMChildrenCollectionEnum_Skip(IEnumVARIANT
*iface
, ULONG celt
)
145 HTMLDOMChildrenCollectionEnum
*This
= impl_from_IEnumVARIANT(iface
);
148 TRACE("(%p)->(%d)\n", This
, celt
);
150 len
= get_enum_len(This
);
151 if(This
->iter
+ celt
> len
) {
160 static HRESULT WINAPI
HTMLDOMChildrenCollectionEnum_Reset(IEnumVARIANT
*iface
)
162 HTMLDOMChildrenCollectionEnum
*This
= impl_from_IEnumVARIANT(iface
);
164 TRACE("(%p)->()\n", This
);
170 static HRESULT WINAPI
HTMLDOMChildrenCollectionEnum_Clone(IEnumVARIANT
*iface
, IEnumVARIANT
**ppEnum
)
172 HTMLDOMChildrenCollectionEnum
*This
= impl_from_IEnumVARIANT(iface
);
173 FIXME("(%p)->(%p)\n", This
, ppEnum
);
177 static const IEnumVARIANTVtbl HTMLDOMChildrenCollectionEnumVtbl
= {
178 HTMLDOMChildrenCollectionEnum_QueryInterface
,
179 HTMLDOMChildrenCollectionEnum_AddRef
,
180 HTMLDOMChildrenCollectionEnum_Release
,
181 HTMLDOMChildrenCollectionEnum_Next
,
182 HTMLDOMChildrenCollectionEnum_Skip
,
183 HTMLDOMChildrenCollectionEnum_Reset
,
184 HTMLDOMChildrenCollectionEnum_Clone
187 static inline HTMLDOMChildrenCollection
*impl_from_IHTMLDOMChildrenCollection(IHTMLDOMChildrenCollection
*iface
)
189 return CONTAINING_RECORD(iface
, HTMLDOMChildrenCollection
, IHTMLDOMChildrenCollection_iface
);
192 static HRESULT WINAPI
HTMLDOMChildrenCollection_QueryInterface(IHTMLDOMChildrenCollection
*iface
, REFIID riid
, void **ppv
)
194 HTMLDOMChildrenCollection
*This
= impl_from_IHTMLDOMChildrenCollection(iface
);
196 TRACE("(%p)->(%s %p)\n", This
, debugstr_mshtml_guid(riid
), ppv
);
198 if(IsEqualGUID(&IID_IUnknown
, riid
)) {
199 *ppv
= &This
->IHTMLDOMChildrenCollection_iface
;
200 }else if(IsEqualGUID(&IID_IHTMLDOMChildrenCollection
, riid
)) {
201 *ppv
= &This
->IHTMLDOMChildrenCollection_iface
;
202 }else if(dispex_query_interface(&This
->dispex
, riid
, ppv
)) {
203 return *ppv
? S_OK
: E_NOINTERFACE
;
206 WARN("(%p)->(%s %p)\n", This
, debugstr_mshtml_guid(riid
), ppv
);
207 return E_NOINTERFACE
;
210 IUnknown_AddRef((IUnknown
*)*ppv
);
214 static ULONG WINAPI
HTMLDOMChildrenCollection_AddRef(IHTMLDOMChildrenCollection
*iface
)
216 HTMLDOMChildrenCollection
*This
= impl_from_IHTMLDOMChildrenCollection(iface
);
217 LONG ref
= InterlockedIncrement(&This
->ref
);
219 TRACE("(%p) ref=%d\n", This
, ref
);
224 static ULONG WINAPI
HTMLDOMChildrenCollection_Release(IHTMLDOMChildrenCollection
*iface
)
226 HTMLDOMChildrenCollection
*This
= impl_from_IHTMLDOMChildrenCollection(iface
);
227 LONG ref
= InterlockedDecrement(&This
->ref
);
229 TRACE("(%p) ref=%d\n", This
, ref
);
232 htmldoc_release(&This
->doc
->basedoc
);
233 nsIDOMNodeList_Release(This
->nslist
);
240 static HRESULT WINAPI
HTMLDOMChildrenCollection_GetTypeInfoCount(IHTMLDOMChildrenCollection
*iface
, UINT
*pctinfo
)
242 HTMLDOMChildrenCollection
*This
= impl_from_IHTMLDOMChildrenCollection(iface
);
243 return IDispatchEx_GetTypeInfoCount(&This
->dispex
.IDispatchEx_iface
, pctinfo
);
246 static HRESULT WINAPI
HTMLDOMChildrenCollection_GetTypeInfo(IHTMLDOMChildrenCollection
*iface
, UINT iTInfo
,
247 LCID lcid
, ITypeInfo
**ppTInfo
)
249 HTMLDOMChildrenCollection
*This
= impl_from_IHTMLDOMChildrenCollection(iface
);
250 return IDispatchEx_GetTypeInfo(&This
->dispex
.IDispatchEx_iface
, iTInfo
, lcid
, ppTInfo
);
253 static HRESULT WINAPI
HTMLDOMChildrenCollection_GetIDsOfNames(IHTMLDOMChildrenCollection
*iface
, REFIID riid
,
254 LPOLESTR
*rgszNames
, UINT cNames
, LCID lcid
, DISPID
*rgDispId
)
256 HTMLDOMChildrenCollection
*This
= impl_from_IHTMLDOMChildrenCollection(iface
);
257 return IDispatchEx_GetIDsOfNames(&This
->dispex
.IDispatchEx_iface
, riid
, rgszNames
, cNames
,
261 static HRESULT WINAPI
HTMLDOMChildrenCollection_Invoke(IHTMLDOMChildrenCollection
*iface
, DISPID dispIdMember
,
262 REFIID riid
, LCID lcid
, WORD wFlags
, DISPPARAMS
*pDispParams
,
263 VARIANT
*pVarResult
, EXCEPINFO
*pExcepInfo
, UINT
*puArgErr
)
265 HTMLDOMChildrenCollection
*This
= impl_from_IHTMLDOMChildrenCollection(iface
);
266 return IDispatchEx_Invoke(&This
->dispex
.IDispatchEx_iface
, dispIdMember
, riid
, lcid
,
267 wFlags
, pDispParams
, pVarResult
, pExcepInfo
, puArgErr
);
270 static HRESULT WINAPI
HTMLDOMChildrenCollection_get_length(IHTMLDOMChildrenCollection
*iface
, LONG
*p
)
272 HTMLDOMChildrenCollection
*This
= impl_from_IHTMLDOMChildrenCollection(iface
);
275 TRACE("(%p)->(%p)\n", This
, p
);
277 nsIDOMNodeList_GetLength(This
->nslist
, &length
);
282 static HRESULT WINAPI
HTMLDOMChildrenCollection_get__newEnum(IHTMLDOMChildrenCollection
*iface
, IUnknown
**p
)
284 HTMLDOMChildrenCollection
*This
= impl_from_IHTMLDOMChildrenCollection(iface
);
285 HTMLDOMChildrenCollectionEnum
*ret
;
287 TRACE("(%p)->(%p)\n", This
, p
);
289 ret
= heap_alloc(sizeof(*ret
));
291 return E_OUTOFMEMORY
;
293 ret
->IEnumVARIANT_iface
.lpVtbl
= &HTMLDOMChildrenCollectionEnumVtbl
;
297 IHTMLDOMChildrenCollection_AddRef(&This
->IHTMLDOMChildrenCollection_iface
);
300 *p
= (IUnknown
*)&ret
->IEnumVARIANT_iface
;
304 static HRESULT WINAPI
HTMLDOMChildrenCollection_item(IHTMLDOMChildrenCollection
*iface
, LONG index
, IDispatch
**ppItem
)
306 HTMLDOMChildrenCollection
*This
= impl_from_IHTMLDOMChildrenCollection(iface
);
307 nsIDOMNode
*nsnode
= NULL
;
313 TRACE("(%p)->(%d %p)\n", This
, index
, ppItem
);
320 nsIDOMNodeList_GetLength(This
->nslist
, &length
);
321 if(index
< 0 || index
>= length
)
324 nsres
= nsIDOMNodeList_Item(This
->nslist
, index
, &nsnode
);
325 if(NS_FAILED(nsres
) || !nsnode
) {
326 ERR("Item failed: %08x\n", nsres
);
330 hres
= get_node(This
->doc
, nsnode
, TRUE
, &node
);
334 *ppItem
= (IDispatch
*)&node
->IHTMLDOMNode_iface
;
338 static const IHTMLDOMChildrenCollectionVtbl HTMLDOMChildrenCollectionVtbl
= {
339 HTMLDOMChildrenCollection_QueryInterface
,
340 HTMLDOMChildrenCollection_AddRef
,
341 HTMLDOMChildrenCollection_Release
,
342 HTMLDOMChildrenCollection_GetTypeInfoCount
,
343 HTMLDOMChildrenCollection_GetTypeInfo
,
344 HTMLDOMChildrenCollection_GetIDsOfNames
,
345 HTMLDOMChildrenCollection_Invoke
,
346 HTMLDOMChildrenCollection_get_length
,
347 HTMLDOMChildrenCollection_get__newEnum
,
348 HTMLDOMChildrenCollection_item
351 static inline HTMLDOMChildrenCollection
*impl_from_DispatchEx(DispatchEx
*iface
)
353 return CONTAINING_RECORD(iface
, HTMLDOMChildrenCollection
, dispex
);
356 #define DISPID_CHILDCOL_0 MSHTML_DISPID_CUSTOM_MIN
358 static HRESULT
HTMLDOMChildrenCollection_get_dispid(DispatchEx
*dispex
, BSTR name
, DWORD flags
, DISPID
*dispid
)
360 HTMLDOMChildrenCollection
*This
= impl_from_DispatchEx(dispex
);
365 for(ptr
= name
; *ptr
&& isdigitW(*ptr
); ptr
++)
366 idx
= idx
*10 + (*ptr
-'0');
368 return DISP_E_UNKNOWNNAME
;
370 nsIDOMNodeList_GetLength(This
->nslist
, &len
);
372 return DISP_E_UNKNOWNNAME
;
374 *dispid
= DISPID_CHILDCOL_0
+ idx
;
375 TRACE("ret %x\n", *dispid
);
379 static HRESULT
HTMLDOMChildrenCollection_invoke(DispatchEx
*dispex
, DISPID id
, LCID lcid
, WORD flags
, DISPPARAMS
*params
,
380 VARIANT
*res
, EXCEPINFO
*ei
, IServiceProvider
*caller
)
382 HTMLDOMChildrenCollection
*This
= impl_from_DispatchEx(dispex
);
384 TRACE("(%p)->(%x %x %x %p %p %p %p)\n", This
, id
, lcid
, flags
, params
, res
, ei
, caller
);
387 case DISPATCH_PROPERTYGET
: {
388 IDispatch
*disp
= NULL
;
391 hres
= IHTMLDOMChildrenCollection_item(&This
->IHTMLDOMChildrenCollection_iface
,
392 id
- DISPID_CHILDCOL_0
, &disp
);
396 V_VT(res
) = VT_DISPATCH
;
397 V_DISPATCH(res
) = disp
;
402 FIXME("unimplemented flags %x\n", flags
);
409 static const dispex_static_data_vtbl_t HTMLDOMChildrenCollection_dispex_vtbl
= {
411 HTMLDOMChildrenCollection_get_dispid
,
412 HTMLDOMChildrenCollection_invoke
,
416 static const tid_t HTMLDOMChildrenCollection_iface_tids
[] = {
417 IHTMLDOMChildrenCollection_tid
,
421 static dispex_static_data_t HTMLDOMChildrenCollection_dispex
= {
422 &HTMLDOMChildrenCollection_dispex_vtbl
,
423 DispDOMChildrenCollection_tid
,
425 HTMLDOMChildrenCollection_iface_tids
428 static IHTMLDOMChildrenCollection
*create_child_collection(HTMLDocumentNode
*doc
, nsIDOMNodeList
*nslist
)
430 HTMLDOMChildrenCollection
*ret
;
432 ret
= heap_alloc_zero(sizeof(*ret
));
436 ret
->IHTMLDOMChildrenCollection_iface
.lpVtbl
= &HTMLDOMChildrenCollectionVtbl
;
439 nsIDOMNodeList_AddRef(nslist
);
440 ret
->nslist
= nslist
;
442 htmldoc_addref(&doc
->basedoc
);
445 init_dispex(&ret
->dispex
, (IUnknown
*)&ret
->IHTMLDOMChildrenCollection_iface
,
446 &HTMLDOMChildrenCollection_dispex
);
448 return &ret
->IHTMLDOMChildrenCollection_iface
;
451 static inline HTMLDOMNode
*impl_from_IHTMLDOMNode(IHTMLDOMNode
*iface
)
453 return CONTAINING_RECORD(iface
, HTMLDOMNode
, IHTMLDOMNode_iface
);
456 static HRESULT WINAPI
HTMLDOMNode_QueryInterface(IHTMLDOMNode
*iface
,
457 REFIID riid
, void **ppv
)
459 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
461 return This
->vtbl
->qi(This
, riid
, ppv
);
464 static ULONG WINAPI
HTMLDOMNode_AddRef(IHTMLDOMNode
*iface
)
466 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
469 ref
= ccref_incr(&This
->ccref
, (nsISupports
*)&This
->IHTMLDOMNode_iface
);
471 TRACE("(%p) ref=%d\n", This
, ref
);
476 static ULONG WINAPI
HTMLDOMNode_Release(IHTMLDOMNode
*iface
)
478 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
479 LONG ref
= ccref_decr(&This
->ccref
, (nsISupports
*)&This
->IHTMLDOMNode_iface
, /*&node_ccp*/ NULL
);
481 TRACE("(%p) ref=%d\n", This
, ref
);
486 static HRESULT WINAPI
HTMLDOMNode_GetTypeInfoCount(IHTMLDOMNode
*iface
, UINT
*pctinfo
)
488 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
489 return IDispatchEx_GetTypeInfoCount(&This
->event_target
.dispex
.IDispatchEx_iface
, pctinfo
);
492 static HRESULT WINAPI
HTMLDOMNode_GetTypeInfo(IHTMLDOMNode
*iface
, UINT iTInfo
,
493 LCID lcid
, ITypeInfo
**ppTInfo
)
495 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
496 return IDispatchEx_GetTypeInfo(&This
->event_target
.dispex
.IDispatchEx_iface
, iTInfo
, lcid
, ppTInfo
);
499 static HRESULT WINAPI
HTMLDOMNode_GetIDsOfNames(IHTMLDOMNode
*iface
, REFIID riid
,
500 LPOLESTR
*rgszNames
, UINT cNames
,
501 LCID lcid
, DISPID
*rgDispId
)
503 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
504 return IDispatchEx_GetIDsOfNames(&This
->event_target
.dispex
.IDispatchEx_iface
, riid
, rgszNames
, cNames
,
508 static HRESULT WINAPI
HTMLDOMNode_Invoke(IHTMLDOMNode
*iface
, DISPID dispIdMember
,
509 REFIID riid
, LCID lcid
, WORD wFlags
, DISPPARAMS
*pDispParams
,
510 VARIANT
*pVarResult
, EXCEPINFO
*pExcepInfo
, UINT
*puArgErr
)
512 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
513 return IDispatchEx_Invoke(&This
->event_target
.dispex
.IDispatchEx_iface
, dispIdMember
, riid
, lcid
,
514 wFlags
, pDispParams
, pVarResult
, pExcepInfo
, puArgErr
);
517 static HRESULT WINAPI
HTMLDOMNode_get_nodeType(IHTMLDOMNode
*iface
, LONG
*p
)
519 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
522 TRACE("(%p)->(%p)\n", This
, p
);
524 nsIDOMNode_GetNodeType(This
->nsnode
, &type
);
534 case DOCUMENT_TYPE_NODE
:
540 case DOCUMENT_FRAGMENT_NODE
:
546 * According to MSDN only ELEMENT_NODE and TEXT_NODE are supported.
547 * It needs more tests.
549 FIXME("type %u\n", type
);
556 static HRESULT WINAPI
HTMLDOMNode_get_parentNode(IHTMLDOMNode
*iface
, IHTMLDOMNode
**p
)
558 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
564 TRACE("(%p)->(%p)\n", This
, p
);
566 nsres
= nsIDOMNode_GetParentNode(This
->nsnode
, &nsnode
);
567 if(NS_FAILED(nsres
)) {
568 ERR("GetParentNode failed: %08x\n", nsres
);
577 hres
= get_node(This
->doc
, nsnode
, TRUE
, &node
);
578 nsIDOMNode_Release(nsnode
);
582 *p
= &node
->IHTMLDOMNode_iface
;
586 static HRESULT WINAPI
HTMLDOMNode_hasChildNodes(IHTMLDOMNode
*iface
, VARIANT_BOOL
*fChildren
)
588 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
589 cpp_bool has_child
= FALSE
;
592 TRACE("(%p)->(%p)\n", This
, fChildren
);
594 nsres
= nsIDOMNode_HasChildNodes(This
->nsnode
, &has_child
);
596 ERR("HasChildNodes failed: %08x\n", nsres
);
598 *fChildren
= has_child
? VARIANT_TRUE
: VARIANT_FALSE
;
602 static HRESULT WINAPI
HTMLDOMNode_get_childNodes(IHTMLDOMNode
*iface
, IDispatch
**p
)
604 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
605 nsIDOMNodeList
*nslist
;
608 TRACE("(%p)->(%p)\n", This
, p
);
610 nsres
= nsIDOMNode_GetChildNodes(This
->nsnode
, &nslist
);
611 if(NS_FAILED(nsres
)) {
612 ERR("GetChildNodes failed: %08x\n", nsres
);
616 *p
= (IDispatch
*)create_child_collection(This
->doc
, nslist
);
617 nsIDOMNodeList_Release(nslist
);
619 return *p
? S_OK
: E_OUTOFMEMORY
;
622 static HRESULT WINAPI
HTMLDOMNode_get_attributes(IHTMLDOMNode
*iface
, IDispatch
**p
)
624 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
625 HTMLAttributeCollection
*col
;
628 TRACE("(%p)->(%p)\n", This
, p
);
630 if(This
->vtbl
->get_attr_col
) {
631 hres
= This
->vtbl
->get_attr_col(This
, &col
);
635 *p
= (IDispatch
*)&col
->IHTMLAttributeCollection_iface
;
643 static HRESULT WINAPI
HTMLDOMNode_insertBefore(IHTMLDOMNode
*iface
, IHTMLDOMNode
*newChild
,
644 VARIANT refChild
, IHTMLDOMNode
**node
)
646 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
647 HTMLDOMNode
*new_child
, *node_obj
, *ref_node
= NULL
;
652 TRACE("(%p)->(%p %s %p)\n", This
, newChild
, debugstr_variant(&refChild
), node
);
654 new_child
= get_node_obj(newChild
);
656 ERR("invalid newChild\n");
660 switch(V_VT(&refChild
)) {
664 IHTMLDOMNode
*ref_iface
;
666 if(!V_DISPATCH(&refChild
))
669 hres
= IDispatch_QueryInterface(V_DISPATCH(&refChild
), &IID_IHTMLDOMNode
, (void**)&ref_iface
);
673 ref_node
= get_node_obj(ref_iface
);
674 IHTMLDOMNode_Release(ref_iface
);
676 ERR("unvalid node\n");
683 FIXME("unimplemented refChild %s\n", debugstr_variant(&refChild
));
687 if(SUCCEEDED(hres
)) {
688 nsres
= nsIDOMNode_InsertBefore(This
->nsnode
, new_child
->nsnode
, ref_node
? ref_node
->nsnode
: NULL
, &nsnode
);
689 if(NS_FAILED(nsres
)) {
690 ERR("InsertBefore failed: %08x\n", nsres
);
694 node_release(new_child
);
696 node_release(ref_node
);
700 hres
= get_node(This
->doc
, nsnode
, TRUE
, &node_obj
);
701 nsIDOMNode_Release(nsnode
);
705 *node
= &node_obj
->IHTMLDOMNode_iface
;
709 static HRESULT WINAPI
HTMLDOMNode_removeChild(IHTMLDOMNode
*iface
, IHTMLDOMNode
*oldChild
,
712 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
713 HTMLDOMNode
*node_obj
;
718 TRACE("(%p)->(%p %p)\n", This
, oldChild
, node
);
720 node_obj
= get_node_obj(oldChild
);
724 nsres
= nsIDOMNode_RemoveChild(This
->nsnode
, node_obj
->nsnode
, &nsnode
);
725 node_release(node_obj
);
726 if(NS_FAILED(nsres
)) {
727 ERR("RemoveChild failed: %08x\n", nsres
);
731 hres
= get_node(This
->doc
, nsnode
, TRUE
, &node_obj
);
732 nsIDOMNode_Release(nsnode
);
736 /* FIXME: Make sure that node != newChild */
737 *node
= &node_obj
->IHTMLDOMNode_iface
;
741 static HRESULT WINAPI
HTMLDOMNode_replaceChild(IHTMLDOMNode
*iface
, IHTMLDOMNode
*newChild
,
742 IHTMLDOMNode
*oldChild
, IHTMLDOMNode
**node
)
744 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
745 HTMLDOMNode
*node_new
, *node_old
, *ret_node
;
750 TRACE("(%p)->(%p %p %p)\n", This
, newChild
, oldChild
, node
);
752 node_new
= get_node_obj(newChild
);
756 node_old
= get_node_obj(oldChild
);
758 node_release(node_new
);
762 nsres
= nsIDOMNode_ReplaceChild(This
->nsnode
, node_new
->nsnode
, node_old
->nsnode
, &nsnode
);
763 node_release(node_new
);
764 node_release(node_old
);
768 hres
= get_node(This
->doc
, nsnode
, TRUE
, &ret_node
);
769 nsIDOMNode_Release(nsnode
);
773 *node
= &ret_node
->IHTMLDOMNode_iface
;
777 static HRESULT WINAPI
HTMLDOMNode_cloneNode(IHTMLDOMNode
*iface
, VARIANT_BOOL fDeep
,
778 IHTMLDOMNode
**clonedNode
)
780 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
781 HTMLDOMNode
*new_node
;
786 TRACE("(%p)->(%x %p)\n", This
, fDeep
, clonedNode
);
788 nsres
= nsIDOMNode_CloneNode(This
->nsnode
, fDeep
!= VARIANT_FALSE
, 1, &nsnode
);
789 if(NS_FAILED(nsres
) || !nsnode
) {
790 ERR("CloneNode failed: %08x\n", nsres
);
794 hres
= This
->vtbl
->clone(This
, nsnode
, &new_node
);
798 *clonedNode
= &new_node
->IHTMLDOMNode_iface
;
802 static HRESULT WINAPI
HTMLDOMNode_removeNode(IHTMLDOMNode
*iface
, VARIANT_BOOL fDeep
,
803 IHTMLDOMNode
**removed
)
805 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
806 FIXME("(%p)->(%x %p)\n", This
, fDeep
, removed
);
810 static HRESULT WINAPI
HTMLDOMNode_swapNode(IHTMLDOMNode
*iface
, IHTMLDOMNode
*otherNode
,
811 IHTMLDOMNode
**swappedNode
)
813 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
814 FIXME("(%p)->(%p %p)\n", This
, otherNode
, swappedNode
);
818 static HRESULT WINAPI
HTMLDOMNode_replaceNode(IHTMLDOMNode
*iface
, IHTMLDOMNode
*replacement
,
819 IHTMLDOMNode
**replaced
)
821 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
822 FIXME("(%p)->(%p %p)\n", This
, replacement
, replaced
);
826 static HRESULT WINAPI
HTMLDOMNode_appendChild(IHTMLDOMNode
*iface
, IHTMLDOMNode
*newChild
,
829 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
830 HTMLDOMNode
*node_obj
;
835 TRACE("(%p)->(%p %p)\n", This
, newChild
, node
);
837 node_obj
= get_node_obj(newChild
);
841 nsres
= nsIDOMNode_AppendChild(This
->nsnode
, node_obj
->nsnode
, &nsnode
);
842 node_release(node_obj
);
843 if(NS_FAILED(nsres
)) {
844 ERR("AppendChild failed: %08x\n", nsres
);
848 hres
= get_node(This
->doc
, nsnode
, TRUE
, &node_obj
);
849 nsIDOMNode_Release(nsnode
);
853 /* FIXME: Make sure that node != newChild */
854 *node
= &node_obj
->IHTMLDOMNode_iface
;
858 static HRESULT WINAPI
HTMLDOMNode_get_nodeName(IHTMLDOMNode
*iface
, BSTR
*p
)
860 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
864 TRACE("(%p)->(%p)\n", This
, p
);
866 nsAString_Init(&name
, NULL
);
867 nsres
= nsIDOMNode_GetNodeName(This
->nsnode
, &name
);
868 return return_nsstr(nsres
, &name
, p
);
871 static HRESULT WINAPI
HTMLDOMNode_put_nodeValue(IHTMLDOMNode
*iface
, VARIANT v
)
873 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
875 TRACE("(%p)->(%s)\n", This
, debugstr_variant(&v
));
881 nsAString_InitDepend(&val_str
, V_BSTR(&v
));
882 nsIDOMNode_SetNodeValue(This
->nsnode
, &val_str
);
883 nsAString_Finish(&val_str
);
889 FIXME("unsupported value %s\n", debugstr_variant(&v
));
895 static HRESULT WINAPI
HTMLDOMNode_get_nodeValue(IHTMLDOMNode
*iface
, VARIANT
*p
)
897 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
898 const PRUnichar
*val
;
901 TRACE("(%p)->(%p)\n", This
, p
);
903 nsAString_Init(&val_str
, NULL
);
904 nsIDOMNode_GetNodeValue(This
->nsnode
, &val_str
);
905 nsAString_GetData(&val_str
, &val
);
909 V_BSTR(p
) = SysAllocString(val
);
914 nsAString_Finish(&val_str
);
919 static HRESULT WINAPI
HTMLDOMNode_get_firstChild(IHTMLDOMNode
*iface
, IHTMLDOMNode
**p
)
921 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
922 nsIDOMNode
*nschild
= NULL
;
926 TRACE("(%p)->(%p)\n", This
, p
);
928 nsIDOMNode_GetFirstChild(This
->nsnode
, &nschild
);
934 hres
= get_node(This
->doc
, nschild
, TRUE
, &node
);
935 nsIDOMNode_Release(nschild
);
939 *p
= &node
->IHTMLDOMNode_iface
;
943 static HRESULT WINAPI
HTMLDOMNode_get_lastChild(IHTMLDOMNode
*iface
, IHTMLDOMNode
**p
)
945 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
946 nsIDOMNode
*nschild
= NULL
;
950 TRACE("(%p)->(%p)\n", This
, p
);
952 nsIDOMNode_GetLastChild(This
->nsnode
, &nschild
);
958 hres
= get_node(This
->doc
, nschild
, TRUE
, &node
);
959 nsIDOMNode_Release(nschild
);
963 *p
= &node
->IHTMLDOMNode_iface
;
967 static HRESULT WINAPI
HTMLDOMNode_get_previousSibling(IHTMLDOMNode
*iface
, IHTMLDOMNode
**p
)
969 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
970 nsIDOMNode
*nschild
= NULL
;
974 TRACE("(%p)->(%p)\n", This
, p
);
976 nsIDOMNode_GetPreviousSibling(This
->nsnode
, &nschild
);
982 hres
= get_node(This
->doc
, nschild
, TRUE
, &node
);
983 nsIDOMNode_Release(nschild
);
987 *p
= &node
->IHTMLDOMNode_iface
;
991 static HRESULT WINAPI
HTMLDOMNode_get_nextSibling(IHTMLDOMNode
*iface
, IHTMLDOMNode
**p
)
993 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
994 nsIDOMNode
*nssibling
= NULL
;
998 TRACE("(%p)->(%p)\n", This
, p
);
1000 nsIDOMNode_GetNextSibling(This
->nsnode
, &nssibling
);
1006 hres
= get_node(This
->doc
, nssibling
, TRUE
, &node
);
1007 nsIDOMNode_Release(nssibling
);
1011 *p
= &node
->IHTMLDOMNode_iface
;
1015 static const IHTMLDOMNodeVtbl HTMLDOMNodeVtbl
= {
1016 HTMLDOMNode_QueryInterface
,
1018 HTMLDOMNode_Release
,
1019 HTMLDOMNode_GetTypeInfoCount
,
1020 HTMLDOMNode_GetTypeInfo
,
1021 HTMLDOMNode_GetIDsOfNames
,
1023 HTMLDOMNode_get_nodeType
,
1024 HTMLDOMNode_get_parentNode
,
1025 HTMLDOMNode_hasChildNodes
,
1026 HTMLDOMNode_get_childNodes
,
1027 HTMLDOMNode_get_attributes
,
1028 HTMLDOMNode_insertBefore
,
1029 HTMLDOMNode_removeChild
,
1030 HTMLDOMNode_replaceChild
,
1031 HTMLDOMNode_cloneNode
,
1032 HTMLDOMNode_removeNode
,
1033 HTMLDOMNode_swapNode
,
1034 HTMLDOMNode_replaceNode
,
1035 HTMLDOMNode_appendChild
,
1036 HTMLDOMNode_get_nodeName
,
1037 HTMLDOMNode_put_nodeValue
,
1038 HTMLDOMNode_get_nodeValue
,
1039 HTMLDOMNode_get_firstChild
,
1040 HTMLDOMNode_get_lastChild
,
1041 HTMLDOMNode_get_previousSibling
,
1042 HTMLDOMNode_get_nextSibling
1045 static HTMLDOMNode
*get_node_obj(IHTMLDOMNode
*iface
)
1049 if(iface
->lpVtbl
!= &HTMLDOMNodeVtbl
)
1052 ret
= impl_from_IHTMLDOMNode(iface
);
1057 static inline HTMLDOMNode
*impl_from_IHTMLDOMNode2(IHTMLDOMNode2
*iface
)
1059 return CONTAINING_RECORD(iface
, HTMLDOMNode
, IHTMLDOMNode2_iface
);
1062 static HRESULT WINAPI
HTMLDOMNode2_QueryInterface(IHTMLDOMNode2
*iface
,
1063 REFIID riid
, void **ppv
)
1065 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode2(iface
);
1067 return IHTMLDOMNode_QueryInterface(&This
->IHTMLDOMNode_iface
, riid
, ppv
);
1070 static ULONG WINAPI
HTMLDOMNode2_AddRef(IHTMLDOMNode2
*iface
)
1072 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode2(iface
);
1074 return IHTMLDOMNode_AddRef(&This
->IHTMLDOMNode_iface
);
1077 static ULONG WINAPI
HTMLDOMNode2_Release(IHTMLDOMNode2
*iface
)
1079 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode2(iface
);
1081 return IHTMLDOMNode_Release(&This
->IHTMLDOMNode_iface
);
1084 static HRESULT WINAPI
HTMLDOMNode2_GetTypeInfoCount(IHTMLDOMNode2
*iface
, UINT
*pctinfo
)
1086 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode2(iface
);
1087 return IDispatchEx_GetTypeInfoCount(&This
->event_target
.dispex
.IDispatchEx_iface
, pctinfo
);
1090 static HRESULT WINAPI
HTMLDOMNode2_GetTypeInfo(IHTMLDOMNode2
*iface
, UINT iTInfo
,
1091 LCID lcid
, ITypeInfo
**ppTInfo
)
1093 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode2(iface
);
1094 return IDispatchEx_GetTypeInfo(&This
->event_target
.dispex
.IDispatchEx_iface
, iTInfo
, lcid
, ppTInfo
);
1097 static HRESULT WINAPI
HTMLDOMNode2_GetIDsOfNames(IHTMLDOMNode2
*iface
, REFIID riid
,
1098 LPOLESTR
*rgszNames
, UINT cNames
,
1099 LCID lcid
, DISPID
*rgDispId
)
1101 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode2(iface
);
1102 return IDispatchEx_GetIDsOfNames(&This
->event_target
.dispex
.IDispatchEx_iface
, riid
, rgszNames
, cNames
,
1106 static HRESULT WINAPI
HTMLDOMNode2_Invoke(IHTMLDOMNode2
*iface
, DISPID dispIdMember
,
1107 REFIID riid
, LCID lcid
, WORD wFlags
, DISPPARAMS
*pDispParams
,
1108 VARIANT
*pVarResult
, EXCEPINFO
*pExcepInfo
, UINT
*puArgErr
)
1110 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode2(iface
);
1111 return IDispatchEx_Invoke(&This
->event_target
.dispex
.IDispatchEx_iface
, dispIdMember
, riid
, lcid
,
1112 wFlags
, pDispParams
, pVarResult
, pExcepInfo
, puArgErr
);
1115 static HRESULT WINAPI
HTMLDOMNode2_get_ownerDocument(IHTMLDOMNode2
*iface
, IDispatch
**p
)
1117 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode2(iface
);
1119 TRACE("(%p)->(%p)\n", This
, p
);
1121 /* FIXME: Better check for document node */
1122 if(This
== &This
->doc
->node
) {
1125 *p
= (IDispatch
*)&This
->doc
->basedoc
.IHTMLDocument2_iface
;
1126 IDispatch_AddRef(*p
);
1131 static const IHTMLDOMNode2Vtbl HTMLDOMNode2Vtbl
= {
1132 HTMLDOMNode2_QueryInterface
,
1133 HTMLDOMNode2_AddRef
,
1134 HTMLDOMNode2_Release
,
1135 HTMLDOMNode2_GetTypeInfoCount
,
1136 HTMLDOMNode2_GetTypeInfo
,
1137 HTMLDOMNode2_GetIDsOfNames
,
1138 HTMLDOMNode2_Invoke
,
1139 HTMLDOMNode2_get_ownerDocument
1142 HRESULT
HTMLDOMNode_QI(HTMLDOMNode
*This
, REFIID riid
, void **ppv
)
1144 TRACE("(%p)->(%s %p)\n", This
, debugstr_mshtml_guid(riid
), ppv
);
1146 if(IsEqualGUID(&IID_IUnknown
, riid
)) {
1147 *ppv
= &This
->IHTMLDOMNode_iface
;
1148 }else if(IsEqualGUID(&IID_IDispatch
, riid
)) {
1149 *ppv
= &This
->IHTMLDOMNode_iface
;
1150 }else if(IsEqualGUID(&IID_IHTMLDOMNode
, riid
)) {
1151 *ppv
= &This
->IHTMLDOMNode_iface
;
1152 }else if(IsEqualGUID(&IID_IHTMLDOMNode2
, riid
)) {
1153 *ppv
= &This
->IHTMLDOMNode2_iface
;
1154 }else if(IsEqualGUID(&IID_nsXPCOMCycleCollectionParticipant
, riid
)) {
1157 }else if(IsEqualGUID(&IID_nsCycleCollectionISupports
, riid
)) {
1158 *ppv
= &This
->IHTMLDOMNode_iface
;
1160 }else if(dispex_query_interface(&This
->event_target
.dispex
, riid
, ppv
)) {
1161 return *ppv
? S_OK
: E_NOINTERFACE
;
1164 WARN("(%p)->(%s %p)\n", This
, debugstr_mshtml_guid(riid
), ppv
);
1165 return E_NOINTERFACE
;
1168 IUnknown_AddRef((IUnknown
*)*ppv
);
1172 void HTMLDOMNode_destructor(HTMLDOMNode
*This
)
1175 nsIDOMNode_Release(This
->nsnode
);
1176 if(This
->doc
&& &This
->doc
->node
!= This
)
1177 htmldoc_release(&This
->doc
->basedoc
);
1178 if(This
->event_target
.ptr
)
1179 release_event_target(This
->event_target
.ptr
);
1182 static HRESULT
HTMLDOMNode_clone(HTMLDOMNode
*This
, nsIDOMNode
*nsnode
, HTMLDOMNode
**ret
)
1184 return create_node(This
->doc
, nsnode
, ret
);
1187 static const cpc_entry_t HTMLDOMNode_cpc
[] = {{NULL
}};
1189 static const NodeImplVtbl HTMLDOMNodeImplVtbl
= {
1191 HTMLDOMNode_destructor
,
1196 void HTMLDOMNode_Init(HTMLDocumentNode
*doc
, HTMLDOMNode
*node
, nsIDOMNode
*nsnode
)
1200 node
->IHTMLDOMNode_iface
.lpVtbl
= &HTMLDOMNodeVtbl
;
1201 node
->IHTMLDOMNode2_iface
.lpVtbl
= &HTMLDOMNode2Vtbl
;
1203 ccref_init(&node
->ccref
, 1);
1205 if(&doc
->node
!= node
)
1206 htmldoc_addref(&doc
->basedoc
);
1209 nsIDOMNode_AddRef(nsnode
);
1210 node
->nsnode
= nsnode
;
1212 nsres
= nsIDOMNode_SetMshtmlNode(nsnode
, (nsISupports
*)&node
->IHTMLDOMNode_iface
);
1213 assert(nsres
== NS_OK
);
1216 static HRESULT
create_node(HTMLDocumentNode
*doc
, nsIDOMNode
*nsnode
, HTMLDOMNode
**ret
)
1221 nsIDOMNode_GetNodeType(nsnode
, &node_type
);
1224 case ELEMENT_NODE
: {
1226 hres
= HTMLElement_Create(doc
, nsnode
, FALSE
, &elem
);
1233 hres
= HTMLDOMTextNode_Create(doc
, nsnode
, ret
);
1237 /* doctype nodes are represented as comment nodes (at least in quirks mode) */
1238 case DOCUMENT_TYPE_NODE
:
1239 case COMMENT_NODE
: {
1240 HTMLElement
*comment
;
1241 hres
= HTMLCommentElement_Create(doc
, nsnode
, &comment
);
1244 *ret
= &comment
->node
;
1247 case ATTRIBUTE_NODE
:
1248 ERR("Called on attribute node\n");
1249 return E_UNEXPECTED
;
1253 node
= heap_alloc_zero(sizeof(HTMLDOMNode
));
1255 return E_OUTOFMEMORY
;
1257 node
->vtbl
= &HTMLDOMNodeImplVtbl
;
1258 HTMLDOMNode_Init(doc
, node
, nsnode
);
1263 TRACE("type %d ret %p\n", node_type
, *ret
);
1267 static nsresult NSAPI
HTMLDOMNode_traverse(void *ccp
, void *p
, nsCycleCollectionTraversalCallback
*cb
)
1269 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(p
);
1271 TRACE("%p\n", This
);
1273 describe_cc_node(&This
->ccref
, "HTMLDOMNode", cb
);
1276 note_cc_edge((nsISupports
*)This
->nsnode
, "This->nsnode", cb
);
1277 if(This
->doc
&& &This
->doc
->node
!= This
)
1278 note_cc_edge((nsISupports
*)&This
->doc
->node
.IHTMLDOMNode_iface
, "This->doc", cb
);
1279 dispex_traverse(&This
->event_target
.dispex
, cb
);
1281 if(This
->vtbl
->traverse
)
1282 This
->vtbl
->traverse(This
, cb
);
1287 static nsresult NSAPI
HTMLDOMNode_unlink(void *p
)
1289 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(p
);
1291 TRACE("%p\n", This
);
1293 if(This
->vtbl
->unlink
)
1294 This
->vtbl
->unlink(This
);
1296 dispex_unlink(&This
->event_target
.dispex
);
1299 nsIDOMNode
*nsnode
= This
->nsnode
;
1300 This
->nsnode
= NULL
;
1301 nsIDOMNode_Release(nsnode
);
1304 if(This
->doc
&& &This
->doc
->node
!= This
) {
1305 HTMLDocument
*doc
= &This
->doc
->basedoc
;
1307 htmldoc_release(doc
);
1315 static void NSAPI
HTMLDOMNode_delete_cycle_collectable(void *p
)
1317 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(p
);
1319 TRACE("(%p)\n", This
);
1321 if(This
->vtbl
->unlink
)
1322 This
->vtbl
->unlink(This
);
1323 This
->vtbl
->destructor(This
);
1324 release_dispex(&This
->event_target
.dispex
);
1328 void init_node_cc(void)
1330 static const CCObjCallback node_ccp_callback
= {
1331 HTMLDOMNode_traverse
,
1333 HTMLDOMNode_delete_cycle_collectable
1336 ccp_init(&node_ccp
, &node_ccp_callback
);
1339 HRESULT
get_node(HTMLDocumentNode
*This
, nsIDOMNode
*nsnode
, BOOL create
, HTMLDOMNode
**ret
)
1341 nsISupports
*unk
= NULL
;
1344 nsres
= nsIDOMNode_GetMshtmlNode(nsnode
, &unk
);
1345 assert(nsres
== NS_OK
);
1348 *ret
= get_node_obj((IHTMLDOMNode
*)unk
);
1349 nsISupports_Release(unk
);
1358 return create_node(This
, nsnode
, ret
);