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
**);
26 IHTMLDOMChildrenCollection IHTMLDOMChildrenCollection_iface
;
30 /* FIXME: implement weak reference */
31 HTMLDocumentNode
*doc
;
33 nsIDOMNodeList
*nslist
;
34 } HTMLDOMChildrenCollection
;
36 static inline HTMLDOMChildrenCollection
*impl_from_IHTMLDOMChildrenCollection(IHTMLDOMChildrenCollection
*iface
)
38 return CONTAINING_RECORD(iface
, HTMLDOMChildrenCollection
, IHTMLDOMChildrenCollection_iface
);
41 static HRESULT WINAPI
HTMLDOMChildrenCollection_QueryInterface(IHTMLDOMChildrenCollection
*iface
, REFIID riid
, void **ppv
)
43 HTMLDOMChildrenCollection
*This
= impl_from_IHTMLDOMChildrenCollection(iface
);
47 if(IsEqualGUID(&IID_IUnknown
, riid
)) {
48 TRACE("(%p)->(IID_IUnknown %p)\n", This
, ppv
);
49 *ppv
= &This
->IHTMLDOMChildrenCollection_iface
;
50 }else if(IsEqualGUID(&IID_IHTMLDOMChildrenCollection
, riid
)) {
51 TRACE("(%p)->(IID_IHTMLDOMChildrenCollection %p)\n", This
, ppv
);
52 *ppv
= &This
->IHTMLDOMChildrenCollection_iface
;
53 }else if(dispex_query_interface(&This
->dispex
, riid
, ppv
)) {
54 return *ppv
? S_OK
: E_NOINTERFACE
;
58 IUnknown_AddRef((IUnknown
*)*ppv
);
62 WARN("(%p)->(%s %p)\n", This
, debugstr_guid(riid
), ppv
);
66 static ULONG WINAPI
HTMLDOMChildrenCollection_AddRef(IHTMLDOMChildrenCollection
*iface
)
68 HTMLDOMChildrenCollection
*This
= impl_from_IHTMLDOMChildrenCollection(iface
);
69 LONG ref
= InterlockedIncrement(&This
->ref
);
71 TRACE("(%p) ref=%d\n", This
, ref
);
76 static ULONG WINAPI
HTMLDOMChildrenCollection_Release(IHTMLDOMChildrenCollection
*iface
)
78 HTMLDOMChildrenCollection
*This
= impl_from_IHTMLDOMChildrenCollection(iface
);
79 LONG ref
= InterlockedDecrement(&This
->ref
);
81 TRACE("(%p) ref=%d\n", This
, ref
);
84 htmldoc_release(&This
->doc
->basedoc
);
85 nsIDOMNodeList_Release(This
->nslist
);
92 static HRESULT WINAPI
HTMLDOMChildrenCollection_GetTypeInfoCount(IHTMLDOMChildrenCollection
*iface
, UINT
*pctinfo
)
94 HTMLDOMChildrenCollection
*This
= impl_from_IHTMLDOMChildrenCollection(iface
);
95 return IDispatchEx_GetTypeInfoCount(&This
->dispex
.IDispatchEx_iface
, pctinfo
);
98 static HRESULT WINAPI
HTMLDOMChildrenCollection_GetTypeInfo(IHTMLDOMChildrenCollection
*iface
, UINT iTInfo
,
99 LCID lcid
, ITypeInfo
**ppTInfo
)
101 HTMLDOMChildrenCollection
*This
= impl_from_IHTMLDOMChildrenCollection(iface
);
102 return IDispatchEx_GetTypeInfo(&This
->dispex
.IDispatchEx_iface
, iTInfo
, lcid
, ppTInfo
);
105 static HRESULT WINAPI
HTMLDOMChildrenCollection_GetIDsOfNames(IHTMLDOMChildrenCollection
*iface
, REFIID riid
,
106 LPOLESTR
*rgszNames
, UINT cNames
, LCID lcid
, DISPID
*rgDispId
)
108 HTMLDOMChildrenCollection
*This
= impl_from_IHTMLDOMChildrenCollection(iface
);
109 return IDispatchEx_GetIDsOfNames(&This
->dispex
.IDispatchEx_iface
, riid
, rgszNames
, cNames
,
113 static HRESULT WINAPI
HTMLDOMChildrenCollection_Invoke(IHTMLDOMChildrenCollection
*iface
, DISPID dispIdMember
,
114 REFIID riid
, LCID lcid
, WORD wFlags
, DISPPARAMS
*pDispParams
,
115 VARIANT
*pVarResult
, EXCEPINFO
*pExcepInfo
, UINT
*puArgErr
)
117 HTMLDOMChildrenCollection
*This
= impl_from_IHTMLDOMChildrenCollection(iface
);
118 return IDispatchEx_Invoke(&This
->dispex
.IDispatchEx_iface
, dispIdMember
, riid
, lcid
,
119 wFlags
, pDispParams
, pVarResult
, pExcepInfo
, puArgErr
);
122 static HRESULT WINAPI
HTMLDOMChildrenCollection_get_length(IHTMLDOMChildrenCollection
*iface
, LONG
*p
)
124 HTMLDOMChildrenCollection
*This
= impl_from_IHTMLDOMChildrenCollection(iface
);
127 TRACE("(%p)->(%p)\n", This
, p
);
129 nsIDOMNodeList_GetLength(This
->nslist
, &length
);
134 static HRESULT WINAPI
HTMLDOMChildrenCollection__newEnum(IHTMLDOMChildrenCollection
*iface
, IUnknown
**p
)
136 HTMLDOMChildrenCollection
*This
= impl_from_IHTMLDOMChildrenCollection(iface
);
137 FIXME("(%p)->(%p)\n", This
, p
);
141 static HRESULT WINAPI
HTMLDOMChildrenCollection_item(IHTMLDOMChildrenCollection
*iface
, LONG index
, IDispatch
**ppItem
)
143 HTMLDOMChildrenCollection
*This
= impl_from_IHTMLDOMChildrenCollection(iface
);
144 nsIDOMNode
*nsnode
= NULL
;
150 TRACE("(%p)->(%d %p)\n", This
, index
, ppItem
);
157 nsIDOMNodeList_GetLength(This
->nslist
, &length
);
158 if(index
< 0 || index
>= length
)
161 nsres
= nsIDOMNodeList_Item(This
->nslist
, index
, &nsnode
);
162 if(NS_FAILED(nsres
) || !nsnode
) {
163 ERR("Item failed: %08x\n", nsres
);
167 hres
= get_node(This
->doc
, nsnode
, TRUE
, &node
);
171 *ppItem
= (IDispatch
*)&node
->IHTMLDOMNode_iface
;
175 static const IHTMLDOMChildrenCollectionVtbl HTMLDOMChildrenCollectionVtbl
= {
176 HTMLDOMChildrenCollection_QueryInterface
,
177 HTMLDOMChildrenCollection_AddRef
,
178 HTMLDOMChildrenCollection_Release
,
179 HTMLDOMChildrenCollection_GetTypeInfoCount
,
180 HTMLDOMChildrenCollection_GetTypeInfo
,
181 HTMLDOMChildrenCollection_GetIDsOfNames
,
182 HTMLDOMChildrenCollection_Invoke
,
183 HTMLDOMChildrenCollection_get_length
,
184 HTMLDOMChildrenCollection__newEnum
,
185 HTMLDOMChildrenCollection_item
188 static inline HTMLDOMChildrenCollection
*impl_from_DispatchEx(DispatchEx
*iface
)
190 return CONTAINING_RECORD(iface
, HTMLDOMChildrenCollection
, dispex
);
193 #define DISPID_CHILDCOL_0 MSHTML_DISPID_CUSTOM_MIN
195 static HRESULT
HTMLDOMChildrenCollection_get_dispid(DispatchEx
*dispex
, BSTR name
, DWORD flags
, DISPID
*dispid
)
197 HTMLDOMChildrenCollection
*This
= impl_from_DispatchEx(dispex
);
202 for(ptr
= name
; *ptr
&& isdigitW(*ptr
); ptr
++)
203 idx
= idx
*10 + (*ptr
-'0');
205 return DISP_E_UNKNOWNNAME
;
207 nsIDOMNodeList_GetLength(This
->nslist
, &len
);
209 return DISP_E_UNKNOWNNAME
;
211 *dispid
= DISPID_CHILDCOL_0
+ idx
;
212 TRACE("ret %x\n", *dispid
);
216 static HRESULT
HTMLDOMChildrenCollection_invoke(DispatchEx
*dispex
, DISPID id
, LCID lcid
, WORD flags
, DISPPARAMS
*params
,
217 VARIANT
*res
, EXCEPINFO
*ei
, IServiceProvider
*caller
)
219 HTMLDOMChildrenCollection
*This
= impl_from_DispatchEx(dispex
);
221 TRACE("(%p)->(%x %x %x %p %p %p %p)\n", This
, id
, lcid
, flags
, params
, res
, ei
, caller
);
224 case DISPATCH_PROPERTYGET
: {
225 IDispatch
*disp
= NULL
;
228 hres
= IHTMLDOMChildrenCollection_item(&This
->IHTMLDOMChildrenCollection_iface
,
229 id
- DISPID_CHILDCOL_0
, &disp
);
233 V_VT(res
) = VT_DISPATCH
;
234 V_DISPATCH(res
) = disp
;
239 FIXME("unimplemented flags %x\n", flags
);
246 static const dispex_static_data_vtbl_t HTMLDOMChildrenCollection_dispex_vtbl
= {
248 HTMLDOMChildrenCollection_get_dispid
,
249 HTMLDOMChildrenCollection_invoke
,
253 static const tid_t HTMLDOMChildrenCollection_iface_tids
[] = {
254 IHTMLDOMChildrenCollection_tid
,
258 static dispex_static_data_t HTMLDOMChildrenCollection_dispex
= {
259 &HTMLDOMChildrenCollection_dispex_vtbl
,
260 DispDOMChildrenCollection_tid
,
262 HTMLDOMChildrenCollection_iface_tids
265 static IHTMLDOMChildrenCollection
*create_child_collection(HTMLDocumentNode
*doc
, nsIDOMNodeList
*nslist
)
267 HTMLDOMChildrenCollection
*ret
;
269 ret
= heap_alloc_zero(sizeof(*ret
));
273 ret
->IHTMLDOMChildrenCollection_iface
.lpVtbl
= &HTMLDOMChildrenCollectionVtbl
;
276 nsIDOMNodeList_AddRef(nslist
);
277 ret
->nslist
= nslist
;
279 htmldoc_addref(&doc
->basedoc
);
282 init_dispex(&ret
->dispex
, (IUnknown
*)&ret
->IHTMLDOMChildrenCollection_iface
,
283 &HTMLDOMChildrenCollection_dispex
);
285 return &ret
->IHTMLDOMChildrenCollection_iface
;
288 static inline HTMLDOMNode
*impl_from_IHTMLDOMNode(IHTMLDOMNode
*iface
)
290 return CONTAINING_RECORD(iface
, HTMLDOMNode
, IHTMLDOMNode_iface
);
293 static HRESULT WINAPI
HTMLDOMNode_QueryInterface(IHTMLDOMNode
*iface
,
294 REFIID riid
, void **ppv
)
296 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
298 return This
->vtbl
->qi(This
, riid
, ppv
);
301 static ULONG WINAPI
HTMLDOMNode_AddRef(IHTMLDOMNode
*iface
)
303 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
304 LONG ref
= ccref_incr(&This
->ccref
, (nsISupports
*)&This
->IHTMLDOMNode_iface
);
306 TRACE("(%p) ref=%d\n", This
, ref
);
311 static ULONG WINAPI
HTMLDOMNode_Release(IHTMLDOMNode
*iface
)
313 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
314 LONG ref
= ccref_decr(&This
->ccref
, (nsISupports
*)&This
->IHTMLDOMNode_iface
);
316 TRACE("(%p) ref=%d\n", This
, ref
);
319 This
->vtbl
->destructor(This
);
320 release_dispex(&This
->dispex
);
327 static HRESULT WINAPI
HTMLDOMNode_GetTypeInfoCount(IHTMLDOMNode
*iface
, UINT
*pctinfo
)
329 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
330 return IDispatchEx_GetTypeInfoCount(&This
->dispex
.IDispatchEx_iface
, pctinfo
);
333 static HRESULT WINAPI
HTMLDOMNode_GetTypeInfo(IHTMLDOMNode
*iface
, UINT iTInfo
,
334 LCID lcid
, ITypeInfo
**ppTInfo
)
336 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
337 return IDispatchEx_GetTypeInfo(&This
->dispex
.IDispatchEx_iface
, iTInfo
, lcid
, ppTInfo
);
340 static HRESULT WINAPI
HTMLDOMNode_GetIDsOfNames(IHTMLDOMNode
*iface
, REFIID riid
,
341 LPOLESTR
*rgszNames
, UINT cNames
,
342 LCID lcid
, DISPID
*rgDispId
)
344 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
345 return IDispatchEx_GetIDsOfNames(&This
->dispex
.IDispatchEx_iface
, riid
, rgszNames
, cNames
,
349 static HRESULT WINAPI
HTMLDOMNode_Invoke(IHTMLDOMNode
*iface
, DISPID dispIdMember
,
350 REFIID riid
, LCID lcid
, WORD wFlags
, DISPPARAMS
*pDispParams
,
351 VARIANT
*pVarResult
, EXCEPINFO
*pExcepInfo
, UINT
*puArgErr
)
353 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
354 return IDispatchEx_Invoke(&This
->dispex
.IDispatchEx_iface
, dispIdMember
, riid
, lcid
,
355 wFlags
, pDispParams
, pVarResult
, pExcepInfo
, puArgErr
);
358 static HRESULT WINAPI
HTMLDOMNode_get_nodeType(IHTMLDOMNode
*iface
, LONG
*p
)
360 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
363 TRACE("(%p)->(%p)\n", This
, p
);
365 nsIDOMNode_GetNodeType(This
->nsnode
, &type
);
375 case DOCUMENT_TYPE_NODE
:
381 case DOCUMENT_FRAGMENT_NODE
:
387 * According to MSDN only ELEMENT_NODE and TEXT_NODE are supported.
388 * It needs more tests.
390 FIXME("type %u\n", type
);
397 static HRESULT WINAPI
HTMLDOMNode_get_parentNode(IHTMLDOMNode
*iface
, IHTMLDOMNode
**p
)
399 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
405 TRACE("(%p)->(%p)\n", This
, p
);
407 nsres
= nsIDOMNode_GetParentNode(This
->nsnode
, &nsnode
);
408 if(NS_FAILED(nsres
)) {
409 ERR("GetParentNode failed: %08x\n", nsres
);
418 hres
= get_node(This
->doc
, nsnode
, TRUE
, &node
);
419 nsIDOMNode_Release(nsnode
);
423 *p
= &node
->IHTMLDOMNode_iface
;
427 static HRESULT WINAPI
HTMLDOMNode_hasChildNodes(IHTMLDOMNode
*iface
, VARIANT_BOOL
*fChildren
)
429 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
430 cpp_bool has_child
= FALSE
;
433 TRACE("(%p)->(%p)\n", This
, fChildren
);
435 nsres
= nsIDOMNode_HasChildNodes(This
->nsnode
, &has_child
);
437 ERR("HasChildNodes failed: %08x\n", nsres
);
439 *fChildren
= has_child
? VARIANT_TRUE
: VARIANT_FALSE
;
443 static HRESULT WINAPI
HTMLDOMNode_get_childNodes(IHTMLDOMNode
*iface
, IDispatch
**p
)
445 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
446 nsIDOMNodeList
*nslist
;
449 TRACE("(%p)->(%p)\n", This
, p
);
451 nsres
= nsIDOMNode_GetChildNodes(This
->nsnode
, &nslist
);
452 if(NS_FAILED(nsres
)) {
453 ERR("GetChildNodes failed: %08x\n", nsres
);
457 *p
= (IDispatch
*)create_child_collection(This
->doc
, nslist
);
458 nsIDOMNodeList_Release(nslist
);
460 return *p
? S_OK
: E_OUTOFMEMORY
;
463 static HRESULT WINAPI
HTMLDOMNode_get_attributes(IHTMLDOMNode
*iface
, IDispatch
**p
)
465 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
466 HTMLAttributeCollection
*col
;
469 TRACE("(%p)->(%p)\n", This
, p
);
471 if(This
->vtbl
->get_attr_col
) {
472 hres
= This
->vtbl
->get_attr_col(This
, &col
);
476 *p
= (IDispatch
*)&col
->IHTMLAttributeCollection_iface
;
484 static HRESULT WINAPI
HTMLDOMNode_insertBefore(IHTMLDOMNode
*iface
, IHTMLDOMNode
*newChild
,
485 VARIANT refChild
, IHTMLDOMNode
**node
)
487 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
488 HTMLDOMNode
*new_child
, *node_obj
, *ref_node
= NULL
;
493 TRACE("(%p)->(%p %s %p)\n", This
, newChild
, debugstr_variant(&refChild
), node
);
495 new_child
= get_node_obj(newChild
);
497 ERR("invalid newChild\n");
501 switch(V_VT(&refChild
)) {
505 IHTMLDOMNode
*ref_iface
;
507 if(!V_DISPATCH(&refChild
))
510 hres
= IDispatch_QueryInterface(V_DISPATCH(&refChild
), &IID_IHTMLDOMNode
, (void**)&ref_iface
);
514 ref_node
= get_node_obj(ref_iface
);
515 IHTMLDOMNode_Release(ref_iface
);
517 ERR("unvalid node\n");
524 FIXME("unimplemented refChild %s\n", debugstr_variant(&refChild
));
528 if(SUCCEEDED(hres
)) {
529 nsres
= nsIDOMNode_InsertBefore(This
->nsnode
, new_child
->nsnode
, ref_node
? ref_node
->nsnode
: NULL
, &nsnode
);
530 if(NS_FAILED(nsres
)) {
531 ERR("InsertBefore failed: %08x\n", nsres
);
535 node_release(new_child
);
537 node_release(ref_node
);
541 hres
= get_node(This
->doc
, nsnode
, TRUE
, &node_obj
);
542 nsIDOMNode_Release(nsnode
);
546 *node
= &node_obj
->IHTMLDOMNode_iface
;
550 static HRESULT WINAPI
HTMLDOMNode_removeChild(IHTMLDOMNode
*iface
, IHTMLDOMNode
*oldChild
,
553 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
554 HTMLDOMNode
*node_obj
;
559 TRACE("(%p)->(%p %p)\n", This
, oldChild
, node
);
561 node_obj
= get_node_obj(oldChild
);
565 nsres
= nsIDOMNode_RemoveChild(This
->nsnode
, node_obj
->nsnode
, &nsnode
);
566 node_release(node_obj
);
567 if(NS_FAILED(nsres
)) {
568 ERR("RemoveChild failed: %08x\n", nsres
);
572 hres
= get_node(This
->doc
, nsnode
, TRUE
, &node_obj
);
573 nsIDOMNode_Release(nsnode
);
577 /* FIXME: Make sure that node != newChild */
578 *node
= &node_obj
->IHTMLDOMNode_iface
;
582 static HRESULT WINAPI
HTMLDOMNode_replaceChild(IHTMLDOMNode
*iface
, IHTMLDOMNode
*newChild
,
583 IHTMLDOMNode
*oldChild
, IHTMLDOMNode
**node
)
585 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
586 HTMLDOMNode
*node_new
, *node_old
, *ret_node
;
591 TRACE("(%p)->(%p %p %p)\n", This
, newChild
, oldChild
, node
);
593 node_new
= get_node_obj(newChild
);
597 node_old
= get_node_obj(oldChild
);
599 node_release(node_new
);
603 nsres
= nsIDOMNode_ReplaceChild(This
->nsnode
, node_new
->nsnode
, node_old
->nsnode
, &nsnode
);
604 node_release(node_new
);
605 node_release(node_old
);
609 hres
= get_node(This
->doc
, nsnode
, TRUE
, &ret_node
);
610 nsIDOMNode_Release(nsnode
);
614 *node
= &ret_node
->IHTMLDOMNode_iface
;
618 static HRESULT WINAPI
HTMLDOMNode_cloneNode(IHTMLDOMNode
*iface
, VARIANT_BOOL fDeep
,
619 IHTMLDOMNode
**clonedNode
)
621 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
622 HTMLDOMNode
*new_node
;
627 TRACE("(%p)->(%x %p)\n", This
, fDeep
, clonedNode
);
629 nsres
= nsIDOMNode_CloneNode(This
->nsnode
, fDeep
!= VARIANT_FALSE
, 1, &nsnode
);
630 if(NS_FAILED(nsres
) || !nsnode
) {
631 ERR("CloneNode failed: %08x\n", nsres
);
635 hres
= This
->vtbl
->clone(This
, nsnode
, &new_node
);
639 *clonedNode
= &new_node
->IHTMLDOMNode_iface
;
643 static HRESULT WINAPI
HTMLDOMNode_removeNode(IHTMLDOMNode
*iface
, VARIANT_BOOL fDeep
,
644 IHTMLDOMNode
**removed
)
646 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
647 FIXME("(%p)->(%x %p)\n", This
, fDeep
, removed
);
651 static HRESULT WINAPI
HTMLDOMNode_swapNode(IHTMLDOMNode
*iface
, IHTMLDOMNode
*otherNode
,
652 IHTMLDOMNode
**swappedNode
)
654 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
655 FIXME("(%p)->(%p %p)\n", This
, otherNode
, swappedNode
);
659 static HRESULT WINAPI
HTMLDOMNode_replaceNode(IHTMLDOMNode
*iface
, IHTMLDOMNode
*replacement
,
660 IHTMLDOMNode
**replaced
)
662 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
663 FIXME("(%p)->(%p %p)\n", This
, replacement
, replaced
);
667 static HRESULT WINAPI
HTMLDOMNode_appendChild(IHTMLDOMNode
*iface
, IHTMLDOMNode
*newChild
,
670 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
671 HTMLDOMNode
*node_obj
;
676 TRACE("(%p)->(%p %p)\n", This
, newChild
, node
);
678 node_obj
= get_node_obj(newChild
);
682 nsres
= nsIDOMNode_AppendChild(This
->nsnode
, node_obj
->nsnode
, &nsnode
);
683 node_release(node_obj
);
684 if(NS_FAILED(nsres
)) {
685 ERR("AppendChild failed: %08x\n", nsres
);
689 hres
= get_node(This
->doc
, nsnode
, TRUE
, &node_obj
);
690 nsIDOMNode_Release(nsnode
);
694 /* FIXME: Make sure that node != newChild */
695 *node
= &node_obj
->IHTMLDOMNode_iface
;
699 static HRESULT WINAPI
HTMLDOMNode_get_nodeName(IHTMLDOMNode
*iface
, BSTR
*p
)
701 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
705 TRACE("(%p)->(%p)\n", This
, p
);
707 nsAString_Init(&name
, NULL
);
708 nsres
= nsIDOMNode_GetNodeName(This
->nsnode
, &name
);
709 return return_nsstr(nsres
, &name
, p
);
712 static HRESULT WINAPI
HTMLDOMNode_put_nodeValue(IHTMLDOMNode
*iface
, VARIANT v
)
714 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
716 TRACE("(%p)->(%s)\n", This
, debugstr_variant(&v
));
722 nsAString_InitDepend(&val_str
, V_BSTR(&v
));
723 nsIDOMNode_SetNodeValue(This
->nsnode
, &val_str
);
724 nsAString_Finish(&val_str
);
730 FIXME("unsupported value %s\n", debugstr_variant(&v
));
736 static HRESULT WINAPI
HTMLDOMNode_get_nodeValue(IHTMLDOMNode
*iface
, VARIANT
*p
)
738 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
739 const PRUnichar
*val
;
742 TRACE("(%p)->(%p)\n", This
, p
);
744 nsAString_Init(&val_str
, NULL
);
745 nsIDOMNode_GetNodeValue(This
->nsnode
, &val_str
);
746 nsAString_GetData(&val_str
, &val
);
750 V_BSTR(p
) = SysAllocString(val
);
755 nsAString_Finish(&val_str
);
760 static HRESULT WINAPI
HTMLDOMNode_get_firstChild(IHTMLDOMNode
*iface
, IHTMLDOMNode
**p
)
762 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
763 nsIDOMNode
*nschild
= NULL
;
767 TRACE("(%p)->(%p)\n", This
, p
);
769 nsIDOMNode_GetFirstChild(This
->nsnode
, &nschild
);
775 hres
= get_node(This
->doc
, nschild
, TRUE
, &node
);
776 nsIDOMNode_Release(nschild
);
780 *p
= &node
->IHTMLDOMNode_iface
;
784 static HRESULT WINAPI
HTMLDOMNode_get_lastChild(IHTMLDOMNode
*iface
, IHTMLDOMNode
**p
)
786 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
787 nsIDOMNode
*nschild
= NULL
;
791 TRACE("(%p)->(%p)\n", This
, p
);
793 nsIDOMNode_GetLastChild(This
->nsnode
, &nschild
);
799 hres
= get_node(This
->doc
, nschild
, TRUE
, &node
);
800 nsIDOMNode_Release(nschild
);
804 *p
= &node
->IHTMLDOMNode_iface
;
808 static HRESULT WINAPI
HTMLDOMNode_get_previousSibling(IHTMLDOMNode
*iface
, IHTMLDOMNode
**p
)
810 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
811 nsIDOMNode
*nschild
= NULL
;
815 TRACE("(%p)->(%p)\n", This
, p
);
817 nsIDOMNode_GetPreviousSibling(This
->nsnode
, &nschild
);
823 hres
= get_node(This
->doc
, nschild
, TRUE
, &node
);
824 nsIDOMNode_Release(nschild
);
828 *p
= &node
->IHTMLDOMNode_iface
;
832 static HRESULT WINAPI
HTMLDOMNode_get_nextSibling(IHTMLDOMNode
*iface
, IHTMLDOMNode
**p
)
834 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(iface
);
835 nsIDOMNode
*nssibling
= NULL
;
839 TRACE("(%p)->(%p)\n", This
, p
);
841 nsIDOMNode_GetNextSibling(This
->nsnode
, &nssibling
);
847 hres
= get_node(This
->doc
, nssibling
, TRUE
, &node
);
848 nsIDOMNode_Release(nssibling
);
852 *p
= &node
->IHTMLDOMNode_iface
;
856 static const IHTMLDOMNodeVtbl HTMLDOMNodeVtbl
= {
857 HTMLDOMNode_QueryInterface
,
860 HTMLDOMNode_GetTypeInfoCount
,
861 HTMLDOMNode_GetTypeInfo
,
862 HTMLDOMNode_GetIDsOfNames
,
864 HTMLDOMNode_get_nodeType
,
865 HTMLDOMNode_get_parentNode
,
866 HTMLDOMNode_hasChildNodes
,
867 HTMLDOMNode_get_childNodes
,
868 HTMLDOMNode_get_attributes
,
869 HTMLDOMNode_insertBefore
,
870 HTMLDOMNode_removeChild
,
871 HTMLDOMNode_replaceChild
,
872 HTMLDOMNode_cloneNode
,
873 HTMLDOMNode_removeNode
,
874 HTMLDOMNode_swapNode
,
875 HTMLDOMNode_replaceNode
,
876 HTMLDOMNode_appendChild
,
877 HTMLDOMNode_get_nodeName
,
878 HTMLDOMNode_put_nodeValue
,
879 HTMLDOMNode_get_nodeValue
,
880 HTMLDOMNode_get_firstChild
,
881 HTMLDOMNode_get_lastChild
,
882 HTMLDOMNode_get_previousSibling
,
883 HTMLDOMNode_get_nextSibling
886 static HTMLDOMNode
*get_node_obj(IHTMLDOMNode
*iface
)
890 if(iface
->lpVtbl
!= &HTMLDOMNodeVtbl
)
893 ret
= impl_from_IHTMLDOMNode(iface
);
898 static inline HTMLDOMNode
*impl_from_IHTMLDOMNode2(IHTMLDOMNode2
*iface
)
900 return CONTAINING_RECORD(iface
, HTMLDOMNode
, IHTMLDOMNode2_iface
);
903 static HRESULT WINAPI
HTMLDOMNode2_QueryInterface(IHTMLDOMNode2
*iface
,
904 REFIID riid
, void **ppv
)
906 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode2(iface
);
908 return IHTMLDOMNode_QueryInterface(&This
->IHTMLDOMNode_iface
, riid
, ppv
);
911 static ULONG WINAPI
HTMLDOMNode2_AddRef(IHTMLDOMNode2
*iface
)
913 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode2(iface
);
915 return IHTMLDOMNode_AddRef(&This
->IHTMLDOMNode_iface
);
918 static ULONG WINAPI
HTMLDOMNode2_Release(IHTMLDOMNode2
*iface
)
920 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode2(iface
);
922 return IHTMLDOMNode_Release(&This
->IHTMLDOMNode_iface
);
925 static HRESULT WINAPI
HTMLDOMNode2_GetTypeInfoCount(IHTMLDOMNode2
*iface
, UINT
*pctinfo
)
927 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode2(iface
);
928 return IDispatchEx_GetTypeInfoCount(&This
->dispex
.IDispatchEx_iface
, pctinfo
);
931 static HRESULT WINAPI
HTMLDOMNode2_GetTypeInfo(IHTMLDOMNode2
*iface
, UINT iTInfo
,
932 LCID lcid
, ITypeInfo
**ppTInfo
)
934 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode2(iface
);
935 return IDispatchEx_GetTypeInfo(&This
->dispex
.IDispatchEx_iface
, iTInfo
, lcid
, ppTInfo
);
938 static HRESULT WINAPI
HTMLDOMNode2_GetIDsOfNames(IHTMLDOMNode2
*iface
, REFIID riid
,
939 LPOLESTR
*rgszNames
, UINT cNames
,
940 LCID lcid
, DISPID
*rgDispId
)
942 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode2(iface
);
943 return IDispatchEx_GetIDsOfNames(&This
->dispex
.IDispatchEx_iface
, riid
, rgszNames
, cNames
,
947 static HRESULT WINAPI
HTMLDOMNode2_Invoke(IHTMLDOMNode2
*iface
, DISPID dispIdMember
,
948 REFIID riid
, LCID lcid
, WORD wFlags
, DISPPARAMS
*pDispParams
,
949 VARIANT
*pVarResult
, EXCEPINFO
*pExcepInfo
, UINT
*puArgErr
)
951 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode2(iface
);
952 return IDispatchEx_Invoke(&This
->dispex
.IDispatchEx_iface
, dispIdMember
, riid
, lcid
,
953 wFlags
, pDispParams
, pVarResult
, pExcepInfo
, puArgErr
);
956 static HRESULT WINAPI
HTMLDOMNode2_get_ownerDocument(IHTMLDOMNode2
*iface
, IDispatch
**p
)
958 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode2(iface
);
960 TRACE("(%p)->(%p)\n", This
, p
);
962 /* FIXME: Better check for document node */
963 if(This
== &This
->doc
->node
) {
966 *p
= (IDispatch
*)&This
->doc
->basedoc
.IHTMLDocument2_iface
;
967 IDispatch_AddRef(*p
);
972 static const IHTMLDOMNode2Vtbl HTMLDOMNode2Vtbl
= {
973 HTMLDOMNode2_QueryInterface
,
975 HTMLDOMNode2_Release
,
976 HTMLDOMNode2_GetTypeInfoCount
,
977 HTMLDOMNode2_GetTypeInfo
,
978 HTMLDOMNode2_GetIDsOfNames
,
980 HTMLDOMNode2_get_ownerDocument
983 static nsXPCOMCycleCollectionParticipant node_ccp
;
985 HRESULT
HTMLDOMNode_QI(HTMLDOMNode
*This
, REFIID riid
, void **ppv
)
989 if(IsEqualGUID(&IID_IUnknown
, riid
)) {
990 TRACE("(%p)->(IID_IUnknown %p)\n", This
, ppv
);
991 *ppv
= &This
->IHTMLDOMNode_iface
;
992 }else if(IsEqualGUID(&IID_IDispatch
, riid
)) {
993 TRACE("(%p)->(IID_IDispatch %p)\n", This
, ppv
);
994 *ppv
= &This
->IHTMLDOMNode_iface
;
995 }else if(IsEqualGUID(&IID_IDispatchEx
, riid
)) {
996 if(This
->dispex
.data
) {
997 TRACE("(%p)->(IID_IDispatchEx %p)\n", This
, ppv
);
998 *ppv
= &This
->dispex
.IDispatchEx_iface
;
1000 FIXME("(%p)->(IID_IDispatchEx %p)\n", This
, ppv
);
1001 return E_NOINTERFACE
;
1003 }else if(IsEqualGUID(&IID_IHTMLDOMNode
, riid
)) {
1004 TRACE("(%p)->(IID_IHTMLDOMNode %p)\n", This
, ppv
);
1005 *ppv
= &This
->IHTMLDOMNode_iface
;
1006 }else if(IsEqualGUID(&IID_IHTMLDOMNode2
, riid
)) {
1007 TRACE("(%p)->(IID_IHTMLDOMNode2 %p)\n", This
, ppv
);
1008 *ppv
= &This
->IHTMLDOMNode2_iface
;
1009 }else if(IsEqualGUID(&IID_nsXPCOMCycleCollectionParticipant
, riid
)) {
1010 TRACE("(%p)->(IID_nsXPCOMCycleCollectionParticipant %p)\n", This
, ppv
);
1013 }else if(IsEqualGUID(&IID_nsCycleCollectionISupports
, riid
)) {
1014 TRACE("(%p)->(IID_nsCycleCollectionISupports %p)\n", This
, ppv
);
1015 *ppv
= &This
->IHTMLDOMNode_iface
;
1017 }else if(dispex_query_interface(&This
->dispex
, riid
, ppv
)) {
1018 return *ppv
? S_OK
: E_NOINTERFACE
;
1022 IUnknown_AddRef((IUnknown
*)*ppv
);
1026 WARN("(%p)->(%s %p)\n", This
, debugstr_guid(riid
), ppv
);
1027 return E_NOINTERFACE
;
1030 void HTMLDOMNode_destructor(HTMLDOMNode
*This
)
1033 nsIDOMNode_Release(This
->nsnode
);
1034 if(This
->doc
&& &This
->doc
->node
!= This
)
1035 htmldoc_release(&This
->doc
->basedoc
);
1036 if(This
->event_target
)
1037 release_event_target(This
->event_target
);
1040 static HRESULT
HTMLDOMNode_clone(HTMLDOMNode
*This
, nsIDOMNode
*nsnode
, HTMLDOMNode
**ret
)
1042 return create_node(This
->doc
, nsnode
, ret
);
1045 static const cpc_entry_t HTMLDOMNode_cpc
[] = {{NULL
}};
1047 static const NodeImplVtbl HTMLDOMNodeImplVtbl
= {
1049 HTMLDOMNode_destructor
,
1054 void HTMLDOMNode_Init(HTMLDocumentNode
*doc
, HTMLDOMNode
*node
, nsIDOMNode
*nsnode
)
1058 node
->IHTMLDOMNode_iface
.lpVtbl
= &HTMLDOMNodeVtbl
;
1059 node
->IHTMLDOMNode2_iface
.lpVtbl
= &HTMLDOMNode2Vtbl
;
1061 ccref_init(&node
->ccref
, 1);
1063 if(&doc
->node
!= node
)
1064 htmldoc_addref(&doc
->basedoc
);
1067 nsIDOMNode_AddRef(nsnode
);
1068 node
->nsnode
= nsnode
;
1070 nsres
= nsIDOMNode_SetMshtmlNode(nsnode
, (nsISupports
*)&node
->IHTMLDOMNode_iface
);
1071 assert(nsres
== NS_OK
);
1074 static HRESULT
create_node(HTMLDocumentNode
*doc
, nsIDOMNode
*nsnode
, HTMLDOMNode
**ret
)
1079 nsIDOMNode_GetNodeType(nsnode
, &node_type
);
1082 case ELEMENT_NODE
: {
1084 hres
= HTMLElement_Create(doc
, nsnode
, FALSE
, &elem
);
1091 hres
= HTMLDOMTextNode_Create(doc
, nsnode
, ret
);
1095 /* doctype nodes are represented as comment nodes (at least in quirks mode) */
1096 case DOCUMENT_TYPE_NODE
:
1097 case COMMENT_NODE
: {
1098 HTMLElement
*comment
;
1099 hres
= HTMLCommentElement_Create(doc
, nsnode
, &comment
);
1102 *ret
= &comment
->node
;
1105 case ATTRIBUTE_NODE
:
1106 ERR("Called on attribute node\n");
1107 return E_UNEXPECTED
;
1111 node
= heap_alloc_zero(sizeof(HTMLDOMNode
));
1113 return E_OUTOFMEMORY
;
1115 node
->vtbl
= &HTMLDOMNodeImplVtbl
;
1116 HTMLDOMNode_Init(doc
, node
, nsnode
);
1121 TRACE("type %d ret %p\n", node_type
, *ret
);
1125 static void NSAPI
HTMLDOMNode_unmark_if_purple(void *p
)
1127 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(p
);
1128 ccref_unmark_if_purple(&This
->ccref
);
1131 static nsresult NSAPI
HTMLDOMNode_traverse(void *ccp
, void *p
, nsCycleCollectionTraversalCallback
*cb
)
1133 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(p
);
1135 TRACE("%p\n", This
);
1137 describe_cc_node(&This
->ccref
, "HTMLDOMNode", cb
);
1140 note_cc_edge((nsISupports
*)This
->nsnode
, "This->nsnode", cb
);
1141 if(This
->doc
&& &This
->doc
->node
!= This
)
1142 note_cc_edge((nsISupports
*)&This
->doc
->node
.IHTMLDOMNode_iface
, "This->doc", cb
);
1143 dispex_traverse(&This
->dispex
, cb
);
1145 if(This
->vtbl
->traverse
)
1146 This
->vtbl
->traverse(This
, cb
);
1151 static nsresult NSAPI
HTMLDOMNode_unlink(void *p
)
1153 HTMLDOMNode
*This
= impl_from_IHTMLDOMNode(p
);
1155 TRACE("%p\n", This
);
1157 if(This
->vtbl
->unlink
)
1158 This
->vtbl
->unlink(This
);
1160 dispex_unlink(&This
->dispex
);
1163 nsIDOMNode
*nsnode
= This
->nsnode
;
1164 This
->nsnode
= NULL
;
1165 nsIDOMNode_Release(nsnode
);
1168 if(This
->doc
&& &This
->doc
->node
!= This
) {
1169 HTMLDocument
*doc
= &This
->doc
->basedoc
;
1171 htmldoc_release(doc
);
1179 void init_node_cc(void)
1181 static const CCObjCallback node_ccp_callback
= {
1182 HTMLDOMNode_unmark_if_purple
,
1183 HTMLDOMNode_traverse
,
1187 ccp_init(&node_ccp
, &node_ccp_callback
);
1190 HRESULT
get_node(HTMLDocumentNode
*This
, nsIDOMNode
*nsnode
, BOOL create
, HTMLDOMNode
**ret
)
1192 nsISupports
*unk
= NULL
;
1195 nsres
= nsIDOMNode_GetMshtmlNode(nsnode
, &unk
);
1196 assert(nsres
== NS_OK
);
1199 *ret
= get_node_obj((IHTMLDOMNode
*)unk
);
1200 nsISupports_Release(unk
);
1209 return create_node(This
, nsnode
, ret
);