2 * Copyright 2008 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
21 #include "wine/unicode.h"
22 #include "wine/debug.h"
24 WINE_DEFAULT_DEBUG_CHANNEL(jscript
);
27 * This IID is used to get DispatchEx objecto from interface.
28 * We might consider using private insteface instead.
30 static const IID IID_IDispatchJS
=
31 {0x719c3050,0xf9d3,0x11cf,{0xa4,0x93,0x00,0x40,0x05,0x23,0xa8,0xa6}};
40 struct _dispex_prop_t
{
47 const builtin_prop_t
*p
;
52 static inline DISPID
prop_to_id(DispatchEx
*This
, dispex_prop_t
*prop
)
54 return prop
- This
->props
;
57 static inline dispex_prop_t
*get_prop(DispatchEx
*This
, DISPID id
)
59 if(id
< 0 || id
>= This
->prop_cnt
|| This
->props
[id
].type
== PROP_DELETED
)
62 return This
->props
+id
;
65 static DWORD
get_flags(DispatchEx
*This
, dispex_prop_t
*prop
)
67 if(prop
->type
== PROP_PROTREF
) {
68 dispex_prop_t
*parent
= get_prop(This
->prototype
, prop
->u
.ref
);
70 prop
->type
= PROP_DELETED
;
74 return get_flags(This
->prototype
, parent
);
80 static const builtin_prop_t
*find_builtin_prop(DispatchEx
*This
, const WCHAR
*name
)
82 int min
= 0, max
, i
, r
;
84 max
= This
->builtin_info
->props_cnt
-1;
88 r
= strcmpW(name
, This
->builtin_info
->props
[i
].name
);
90 return This
->builtin_info
->props
+ i
;
101 static dispex_prop_t
*alloc_prop(DispatchEx
*This
, const WCHAR
*name
, prop_type_t type
, DWORD flags
)
105 if(This
->buf_size
== This
->prop_cnt
) {
106 dispex_prop_t
*tmp
= heap_realloc(This
->props
, (This
->buf_size
<<=1)*sizeof(*This
->props
));
112 ret
= This
->props
+ This
->prop_cnt
++;
115 ret
->name
= heap_strdupW(name
);
122 static dispex_prop_t
*alloc_protref(DispatchEx
*This
, const WCHAR
*name
, DWORD ref
)
126 ret
= alloc_prop(This
, name
, PROP_PROTREF
, 0);
134 static HRESULT
find_prop_name(DispatchEx
*This
, const WCHAR
*name
, dispex_prop_t
**ret
)
136 const builtin_prop_t
*builtin
;
139 for(prop
= This
->props
; prop
< This
->props
+This
->prop_cnt
; prop
++) {
140 if(prop
->name
&& !strcmpW(prop
->name
, name
)) {
146 builtin
= find_builtin_prop(This
, name
);
148 prop
= alloc_prop(This
, name
, PROP_BUILTIN
, builtin
->flags
);
150 return E_OUTOFMEMORY
;
161 static HRESULT
find_prop_name_prot(DispatchEx
*This
, const WCHAR
*name
, BOOL alloc
, dispex_prop_t
**ret
)
166 hres
= find_prop_name(This
, name
, &prop
);
174 if(This
->prototype
) {
175 hres
= find_prop_name_prot(This
->prototype
, name
, FALSE
, &prop
);
179 prop
= alloc_protref(This
, prop
->name
, prop
- This
->prototype
->props
);
181 return E_OUTOFMEMORY
;
188 TRACE("creating prop %s\n", debugstr_w(name
));
190 prop
= alloc_prop(This
, name
, PROP_VARIANT
, PROPF_ENUM
);
192 return E_OUTOFMEMORY
;
193 VariantInit(&prop
->u
.var
);
200 static HRESULT
set_this(DISPPARAMS
*dp
, DISPPARAMS
*olddp
, IDispatch
*jsthis
)
205 static DISPID this_id
= DISPID_THIS
;
209 for(i
= 0; i
< dp
->cNamedArgs
; i
++) {
210 if(dp
->rgdispidNamedArgs
[i
] == DISPID_THIS
)
214 oldargs
= dp
->rgvarg
;
215 dp
->rgvarg
= heap_alloc((dp
->cArgs
+1) * sizeof(VARIANTARG
));
217 return E_OUTOFMEMORY
;
218 memcpy(dp
->rgvarg
+1, oldargs
, dp
->cArgs
*sizeof(VARIANTARG
));
219 V_VT(dp
->rgvarg
) = VT_DISPATCH
;
220 V_DISPATCH(dp
->rgvarg
) = jsthis
;
224 DISPID
*old
= dp
->rgdispidNamedArgs
;
225 dp
->rgdispidNamedArgs
= heap_alloc((dp
->cNamedArgs
+1)*sizeof(DISPID
));
226 if(!dp
->rgdispidNamedArgs
) {
227 heap_free(dp
->rgvarg
);
228 return E_OUTOFMEMORY
;
231 memcpy(dp
->rgdispidNamedArgs
+1, old
, dp
->cNamedArgs
*sizeof(DISPID
));
232 dp
->rgdispidNamedArgs
[0] = DISPID_THIS
;
235 dp
->rgdispidNamedArgs
= &this_id
;
242 static HRESULT
invoke_prop_func(DispatchEx
*This
, DispatchEx
*jsthis
, dispex_prop_t
*prop
, LCID lcid
, WORD flags
,
243 DISPPARAMS
*dp
, VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
249 if(flags
== DISPATCH_CONSTRUCT
&& (prop
->flags
& DISPATCH_METHOD
)) {
250 WARN("%s is not a constructor\n", debugstr_w(prop
->name
));
253 return prop
->u
.p
->invoke(jsthis
, lcid
, flags
, dp
, retv
, ei
, caller
);
255 return invoke_prop_func(This
->prototype
, jsthis
, This
->prototype
->props
+prop
->u
.ref
, lcid
, flags
, dp
, retv
, ei
, caller
);
259 if(V_VT(&prop
->u
.var
) != VT_DISPATCH
) {
260 FIXME("invoke vt %d\n", V_VT(&prop
->u
.var
));
264 TRACE("call %s %p\n", debugstr_w(prop
->name
), V_DISPATCH(&prop
->u
.var
));
266 hres
= set_this(&new_dp
, dp
, (IDispatch
*)_IDispatchEx_(jsthis
));
270 hres
= disp_call(V_DISPATCH(&prop
->u
.var
), DISPID_VALUE
, lcid
, flags
, &new_dp
, retv
, ei
, caller
);
272 if(new_dp
.rgvarg
!= dp
->rgvarg
) {
273 heap_free(new_dp
.rgvarg
);
274 if(new_dp
.cNamedArgs
> 1)
275 heap_free(new_dp
.rgdispidNamedArgs
);
281 ERR("type %d\n", prop
->type
);
287 static HRESULT
prop_get(DispatchEx
*This
, dispex_prop_t
*prop
, LCID lcid
, DISPPARAMS
*dp
,
288 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
294 if(prop
->u
.p
->flags
& PROPF_METHOD
) {
296 hres
= create_builtin_function(This
->ctx
, prop
->u
.p
->invoke
, prop
->u
.p
->flags
, NULL
, &obj
);
300 prop
->type
= PROP_VARIANT
;
301 V_VT(&prop
->u
.var
) = VT_DISPATCH
;
302 V_DISPATCH(&prop
->u
.var
) = (IDispatch
*)_IDispatchEx_(obj
);
304 hres
= VariantCopy(retv
, &prop
->u
.var
);
306 hres
= prop
->u
.p
->invoke(This
, lcid
, DISPATCH_PROPERTYGET
, dp
, retv
, ei
, caller
);
310 hres
= prop_get(This
->prototype
, This
->prototype
->props
+prop
->u
.ref
, lcid
, dp
, retv
, ei
, caller
);
313 hres
= VariantCopy(retv
, &prop
->u
.var
);
316 ERR("type %d\n", prop
->type
);
321 TRACE("fail %08x\n", hres
);
325 TRACE("%s ret %s\n", debugstr_w(prop
->name
), debugstr_variant(retv
));
329 static HRESULT
prop_put(DispatchEx
*This
, dispex_prop_t
*prop
, LCID lcid
, DISPPARAMS
*dp
,
330 jsexcept_t
*ei
, IServiceProvider
*caller
)
337 if(!(prop
->flags
& PROPF_METHOD
))
338 return prop
->u
.p
->invoke(This
, lcid
, DISPATCH_PROPERTYPUT
, dp
, NULL
, ei
, caller
);
340 prop
->type
= PROP_VARIANT
;
341 prop
->flags
= PROPF_ENUM
;
342 V_VT(&prop
->u
.var
) = VT_EMPTY
;
345 VariantClear(&prop
->u
.var
);
348 ERR("type %d\n", prop
->type
);
352 for(i
=0; i
< dp
->cNamedArgs
; i
++) {
353 if(dp
->rgdispidNamedArgs
[i
] == DISPID_PROPERTYPUT
)
357 if(i
== dp
->cNamedArgs
) {
358 TRACE("no value to set\n");
359 return DISP_E_PARAMNOTOPTIONAL
;
362 hres
= VariantCopy(&prop
->u
.var
, dp
->rgvarg
+i
);
366 if(This
->builtin_info
->on_put
)
367 This
->builtin_info
->on_put(This
, prop
->name
);
369 TRACE("%s = %s\n", debugstr_w(prop
->name
), debugstr_variant(dp
->rgvarg
+i
));
373 static HRESULT
fill_protrefs(DispatchEx
*This
)
375 dispex_prop_t
*iter
, *prop
;
381 fill_protrefs(This
->prototype
);
383 for(iter
= This
->prototype
->props
; iter
< This
->prototype
->props
+This
->prototype
->prop_cnt
; iter
++) {
386 hres
= find_prop_name(This
, iter
->name
, &prop
);
390 prop
= alloc_protref(This
, iter
->name
, iter
- This
->prototype
->props
);
392 return E_OUTOFMEMORY
;
399 #define DISPATCHEX_THIS(iface) DEFINE_THIS(DispatchEx, IDispatchEx, iface)
401 static HRESULT WINAPI
DispatchEx_QueryInterface(IDispatchEx
*iface
, REFIID riid
, void **ppv
)
403 DispatchEx
*This
= DISPATCHEX_THIS(iface
);
405 if(IsEqualGUID(&IID_IUnknown
, riid
)) {
406 TRACE("(%p)->(IID_IUnknown %p)\n", This
, ppv
);
407 *ppv
= _IDispatchEx_(This
);
408 }else if(IsEqualGUID(&IID_IDispatch
, riid
)) {
409 TRACE("(%p)->(IID_IDispatch %p)\n", This
, ppv
);
410 *ppv
= _IDispatchEx_(This
);
411 }else if(IsEqualGUID(&IID_IDispatchEx
, riid
)) {
412 TRACE("(%p)->(IID_IDispatchEx %p)\n", This
, ppv
);
413 *ppv
= _IDispatchEx_(This
);
414 }else if(IsEqualGUID(&IID_IDispatchJS
, riid
)) {
415 TRACE("(%p)->(IID_IDispatchJS %p)\n", This
, ppv
);
416 IUnknown_AddRef(_IDispatchEx_(This
));
420 WARN("(%p)->(%s %p)\n", This
, debugstr_guid(riid
), ppv
);
422 return E_NOINTERFACE
;
425 IUnknown_AddRef((IUnknown
*)*ppv
);
429 static ULONG WINAPI
DispatchEx_AddRef(IDispatchEx
*iface
)
431 DispatchEx
*This
= DISPATCHEX_THIS(iface
);
432 LONG ref
= InterlockedIncrement(&This
->ref
);
434 TRACE("(%p) ref=%d\n", This
, ref
);
439 static ULONG WINAPI
DispatchEx_Release(IDispatchEx
*iface
)
441 DispatchEx
*This
= DISPATCHEX_THIS(iface
);
442 LONG ref
= InterlockedDecrement(&This
->ref
);
444 TRACE("(%p) ref=%d\n", This
, ref
);
449 for(prop
= This
->props
; prop
< This
->props
+This
->prop_cnt
; prop
++) {
450 if(prop
->type
== PROP_VARIANT
)
451 VariantClear(&prop
->u
.var
);
452 heap_free(prop
->name
);
454 heap_free(This
->props
);
455 script_release(This
->ctx
);
457 if(This
->builtin_info
->destructor
)
458 This
->builtin_info
->destructor(This
);
466 static HRESULT WINAPI
DispatchEx_GetTypeInfoCount(IDispatchEx
*iface
, UINT
*pctinfo
)
468 DispatchEx
*This
= DISPATCHEX_THIS(iface
);
470 TRACE("(%p)->(%p)\n", This
, pctinfo
);
476 static HRESULT WINAPI
DispatchEx_GetTypeInfo(IDispatchEx
*iface
, UINT iTInfo
,
477 LCID lcid
, ITypeInfo
**ppTInfo
)
479 DispatchEx
*This
= DISPATCHEX_THIS(iface
);
480 FIXME("(%p)->(%u %u %p)\n", This
, iTInfo
, lcid
, ppTInfo
);
484 static HRESULT WINAPI
DispatchEx_GetIDsOfNames(IDispatchEx
*iface
, REFIID riid
,
485 LPOLESTR
*rgszNames
, UINT cNames
,
486 LCID lcid
, DISPID
*rgDispId
)
488 DispatchEx
*This
= DISPATCHEX_THIS(iface
);
492 TRACE("(%p)->(%s %p %u %u %p)\n", This
, debugstr_guid(riid
), rgszNames
, cNames
,
495 for(i
=0; i
< cNames
; i
++) {
496 hres
= IDispatchEx_GetDispID(_IDispatchEx_(This
), rgszNames
[i
], 0, rgDispId
+i
);
504 static HRESULT WINAPI
DispatchEx_Invoke(IDispatchEx
*iface
, DISPID dispIdMember
,
505 REFIID riid
, LCID lcid
, WORD wFlags
, DISPPARAMS
*pDispParams
,
506 VARIANT
*pVarResult
, EXCEPINFO
*pExcepInfo
, UINT
*puArgErr
)
508 DispatchEx
*This
= DISPATCHEX_THIS(iface
);
510 TRACE("(%p)->(%d %s %d %d %p %p %p %p)\n", This
, dispIdMember
, debugstr_guid(riid
),
511 lcid
, wFlags
, pDispParams
, pVarResult
, pExcepInfo
, puArgErr
);
513 return IDispatchEx_InvokeEx(_IDispatchEx_(This
), dispIdMember
, lcid
, wFlags
,
514 pDispParams
, pVarResult
, pExcepInfo
, NULL
);
517 static HRESULT WINAPI
DispatchEx_GetDispID(IDispatchEx
*iface
, BSTR bstrName
, DWORD grfdex
, DISPID
*pid
)
519 DispatchEx
*This
= DISPATCHEX_THIS(iface
);
523 TRACE("(%p)->(%s %x %p)\n", This
, debugstr_w(bstrName
), grfdex
, pid
);
525 if(grfdex
& ~(fdexNameCaseSensitive
|fdexNameEnsure
|fdexNameImplicit
)) {
526 FIXME("Unsupported grfdex %x\n", grfdex
);
530 hres
= find_prop_name_prot(This
, bstrName
, (grfdex
&fdexNameEnsure
) != 0, &prop
);
534 *pid
= prop_to_id(This
, prop
);
538 TRACE("not found %s\n", debugstr_w(bstrName
));
539 return DISP_E_UNKNOWNNAME
;
542 static HRESULT WINAPI
DispatchEx_InvokeEx(IDispatchEx
*iface
, DISPID id
, LCID lcid
, WORD wFlags
, DISPPARAMS
*pdp
,
543 VARIANT
*pvarRes
, EXCEPINFO
*pei
, IServiceProvider
*pspCaller
)
545 DispatchEx
*This
= DISPATCHEX_THIS(iface
);
550 TRACE("(%p)->(%x %x %x %p %p %p %p)\n", This
, id
, lcid
, wFlags
, pdp
, pvarRes
, pei
, pspCaller
);
553 V_VT(pvarRes
) = VT_EMPTY
;
555 prop
= get_prop(This
, id
);
556 if(!prop
|| prop
->type
== PROP_DELETED
) {
557 TRACE("invalid id\n");
558 return DISP_E_MEMBERNOTFOUND
;
561 memset(&jsexcept
, 0, sizeof(jsexcept
));
564 case DISPATCH_METHOD
:
565 case DISPATCH_CONSTRUCT
:
566 hres
= invoke_prop_func(This
, This
, prop
, lcid
, wFlags
, pdp
, pvarRes
, &jsexcept
, pspCaller
);
568 case DISPATCH_PROPERTYGET
:
569 hres
= prop_get(This
, prop
, lcid
, pdp
, pvarRes
, &jsexcept
, pspCaller
);
571 case DISPATCH_PROPERTYPUT
:
572 hres
= prop_put(This
, prop
, lcid
, pdp
, &jsexcept
, pspCaller
);
575 FIXME("Unimplemented flags %x\n", wFlags
);
585 static HRESULT WINAPI
DispatchEx_DeleteMemberByName(IDispatchEx
*iface
, BSTR bstrName
, DWORD grfdex
)
587 DispatchEx
*This
= DISPATCHEX_THIS(iface
);
591 TRACE("(%p)->(%s %x)\n", This
, debugstr_w(bstrName
), grfdex
);
593 if(grfdex
& ~(fdexNameCaseSensitive
|fdexNameEnsure
|fdexNameImplicit
))
594 FIXME("Unsupported grfdex %x\n", grfdex
);
596 hres
= find_prop_name(This
, bstrName
, &prop
);
600 TRACE("not found\n");
604 heap_free(prop
->name
);
606 prop
->type
= PROP_DELETED
;
610 static HRESULT WINAPI
DispatchEx_DeleteMemberByDispID(IDispatchEx
*iface
, DISPID id
)
612 DispatchEx
*This
= DISPATCHEX_THIS(iface
);
613 FIXME("(%p)->(%x)\n", This
, id
);
617 static HRESULT WINAPI
DispatchEx_GetMemberProperties(IDispatchEx
*iface
, DISPID id
, DWORD grfdexFetch
, DWORD
*pgrfdex
)
619 DispatchEx
*This
= DISPATCHEX_THIS(iface
);
620 FIXME("(%p)->(%x %x %p)\n", This
, id
, grfdexFetch
, pgrfdex
);
624 static HRESULT WINAPI
DispatchEx_GetMemberName(IDispatchEx
*iface
, DISPID id
, BSTR
*pbstrName
)
626 DispatchEx
*This
= DISPATCHEX_THIS(iface
);
629 TRACE("(%p)->(%x %p)\n", This
, id
, pbstrName
);
631 prop
= get_prop(This
, id
);
632 if(!prop
|| !prop
->name
|| prop
->type
== PROP_DELETED
)
633 return DISP_E_MEMBERNOTFOUND
;
635 *pbstrName
= SysAllocString(prop
->name
);
637 return E_OUTOFMEMORY
;
642 static HRESULT WINAPI
DispatchEx_GetNextDispID(IDispatchEx
*iface
, DWORD grfdex
, DISPID id
, DISPID
*pid
)
644 DispatchEx
*This
= DISPATCHEX_THIS(iface
);
648 TRACE("(%p)->(%x %x %p)\n", This
, grfdex
, id
, pid
);
650 if(id
== DISPID_STARTENUM
) {
651 hres
= fill_protrefs(This
);
656 iter
= get_prop(This
, id
+1);
658 *pid
= DISPID_STARTENUM
;
662 while(iter
< This
->props
+ This
->prop_cnt
) {
663 if(iter
->name
&& (get_flags(This
, iter
) & PROPF_ENUM
)) {
664 *pid
= prop_to_id(This
, iter
);
670 *pid
= DISPID_STARTENUM
;
674 static HRESULT WINAPI
DispatchEx_GetNameSpaceParent(IDispatchEx
*iface
, IUnknown
**ppunk
)
676 DispatchEx
*This
= DISPATCHEX_THIS(iface
);
677 FIXME("(%p)->(%p)\n", This
, ppunk
);
681 #undef DISPATCHEX_THIS
683 static IDispatchExVtbl DispatchExVtbl
= {
684 DispatchEx_QueryInterface
,
687 DispatchEx_GetTypeInfoCount
,
688 DispatchEx_GetTypeInfo
,
689 DispatchEx_GetIDsOfNames
,
691 DispatchEx_GetDispID
,
693 DispatchEx_DeleteMemberByName
,
694 DispatchEx_DeleteMemberByDispID
,
695 DispatchEx_GetMemberProperties
,
696 DispatchEx_GetMemberName
,
697 DispatchEx_GetNextDispID
,
698 DispatchEx_GetNameSpaceParent
701 HRESULT
init_dispex(DispatchEx
*dispex
, script_ctx_t
*ctx
, const builtin_info_t
*builtin_info
, DispatchEx
*prototype
)
703 TRACE("%p (%p)\n", dispex
, prototype
);
705 dispex
->lpIDispatchExVtbl
= &DispatchExVtbl
;
707 dispex
->builtin_info
= builtin_info
;
709 dispex
->props
= heap_alloc((dispex
->buf_size
=4) * sizeof(dispex_prop_t
));
711 return E_OUTOFMEMORY
;
713 dispex
->prototype
= prototype
;
715 IDispatchEx_AddRef(_IDispatchEx_(prototype
));
717 dispex
->prop_cnt
= 1;
718 dispex
->props
[0].name
= NULL
;
719 dispex
->props
[0].flags
= 0;
720 if(builtin_info
->value_prop
.invoke
) {
721 dispex
->props
[0].type
= PROP_BUILTIN
;
722 dispex
->props
[0].u
.p
= &builtin_info
->value_prop
;
724 dispex
->props
[0].type
= PROP_DELETED
;
733 static const builtin_info_t dispex_info
= {
741 HRESULT
create_dispex(script_ctx_t
*ctx
, const builtin_info_t
*builtin_info
, DispatchEx
*prototype
, DispatchEx
**dispex
)
746 ret
= heap_alloc_zero(sizeof(DispatchEx
));
748 return E_OUTOFMEMORY
;
750 hres
= init_dispex(ret
, ctx
, builtin_info
? builtin_info
: &dispex_info
, prototype
);
758 HRESULT
init_dispex_from_constr(DispatchEx
*dispex
, script_ctx_t
*ctx
, const builtin_info_t
*builtin_info
, DispatchEx
*constr
)
760 DispatchEx
*prot
= NULL
;
764 static const WCHAR prototypeW
[] = {'p','r','o','t','o','t','y','p','e',0};
766 hres
= find_prop_name_prot(constr
, prototypeW
, FALSE
, &prop
);
767 if(SUCCEEDED(hres
) && prop
) {
771 V_VT(&var
) = VT_EMPTY
;
772 memset(&jsexcept
, 0, sizeof(jsexcept
));
773 hres
= prop_get(constr
, prop
, ctx
->lcid
, NULL
, &var
, &jsexcept
, NULL
/*FIXME*/);
775 ERR("Could not get prototype\n");
779 if(V_VT(&var
) == VT_DISPATCH
)
780 prot
= iface_to_jsdisp((IUnknown
*)V_DISPATCH(&var
));
784 hres
= init_dispex(dispex
, ctx
, builtin_info
, prot
);
787 IDispatchEx_Release(_IDispatchEx_(prot
));
791 DispatchEx
*iface_to_jsdisp(IUnknown
*iface
)
796 hres
= IUnknown_QueryInterface(iface
, &IID_IDispatchJS
, (void**)&ret
);
803 HRESULT
jsdisp_call_value(DispatchEx
*disp
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
, VARIANT
*retv
,
804 jsexcept_t
*ei
, IServiceProvider
*caller
)
806 return disp
->builtin_info
->value_prop
.invoke(disp
, lcid
, flags
, dp
, retv
, ei
, caller
);
809 HRESULT
jsdisp_call(DispatchEx
*disp
, DISPID id
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
, VARIANT
*retv
,
810 jsexcept_t
*ei
, IServiceProvider
*caller
)
814 memset(ei
, 0, sizeof(*ei
));
816 V_VT(retv
) = VT_EMPTY
;
818 prop
= get_prop(disp
, id
);
820 return DISP_E_MEMBERNOTFOUND
;
822 return invoke_prop_func(disp
, disp
, prop
, lcid
, flags
, dp
, retv
, ei
, caller
);
825 HRESULT
disp_call(IDispatch
*disp
, DISPID id
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
, VARIANT
*retv
,
826 jsexcept_t
*ei
, IServiceProvider
*caller
)
832 jsdisp
= iface_to_jsdisp((IUnknown
*)disp
);
834 hres
= jsdisp_call(jsdisp
, id
, lcid
, flags
, dp
, retv
, ei
, caller
);
835 IDispatchEx_Release(_IDispatchEx_(jsdisp
));
839 memset(ei
, 0, sizeof(*ei
));
842 V_VT(retv
) = VT_EMPTY
;
843 hres
= IDispatch_QueryInterface(disp
, &IID_IDispatchEx
, (void**)&dispex
);
847 TRACE("using IDispatch\n");
848 return IDispatch_Invoke(disp
, id
, &IID_NULL
, lcid
, flags
, dp
, retv
, &ei
->ei
, &err
);
851 hres
= IDispatchEx_InvokeEx(dispex
, id
, lcid
, flags
, dp
, retv
, &ei
->ei
, caller
);
852 IDispatchEx_Release(dispex
);
857 HRESULT
jsdisp_propput_name(DispatchEx
*obj
, const WCHAR
*name
, LCID lcid
, VARIANT
*val
, jsexcept_t
*ei
, IServiceProvider
*caller
)
859 DISPID named_arg
= DISPID_PROPERTYPUT
;
860 DISPPARAMS dp
= {val
, &named_arg
, 1, 1};
864 hres
= find_prop_name_prot(obj
, name
, TRUE
, &prop
);
868 return prop_put(obj
, prop
, lcid
, &dp
, ei
, caller
);
871 HRESULT
jsdisp_propput_idx(DispatchEx
*obj
, DWORD idx
, LCID lcid
, VARIANT
*val
, jsexcept_t
*ei
, IServiceProvider
*caller
)
875 static const WCHAR formatW
[] = {'%','d',0};
877 sprintfW(buf
, formatW
, idx
);
878 return jsdisp_propput_name(obj
, buf
, lcid
, val
, ei
, caller
);
881 HRESULT
disp_propput(IDispatch
*disp
, DISPID id
, LCID lcid
, VARIANT
*val
, jsexcept_t
*ei
, IServiceProvider
*caller
)
883 DISPID dispid
= DISPID_PROPERTYPUT
;
884 DISPPARAMS dp
= {val
, &dispid
, 1, 1};
889 jsdisp
= iface_to_jsdisp((IUnknown
*)disp
);
893 prop
= get_prop(jsdisp
, id
);
895 hres
= prop_put(jsdisp
, prop
, lcid
, &dp
, ei
, caller
);
897 hres
= DISP_E_MEMBERNOTFOUND
;
899 IDispatchEx_Release(_IDispatchEx_(jsdisp
));
903 hres
= IDispatch_QueryInterface(disp
, &IID_IDispatchEx
, (void**)&dispex
);
907 TRACE("using IDispatch\n");
908 return IDispatch_Invoke(disp
, id
, &IID_NULL
, DISPATCH_PROPERTYPUT
, lcid
, &dp
, NULL
, &ei
->ei
, &err
);
911 hres
= IDispatchEx_InvokeEx(dispex
, id
, lcid
, DISPATCH_PROPERTYPUT
, &dp
, NULL
, &ei
->ei
, caller
);
913 IDispatchEx_Release(dispex
);
917 static HRESULT
jsdisp_propget_name(DispatchEx
*obj
, const WCHAR
*name
, LCID lcid
, VARIANT
*var
,
918 jsexcept_t
*ei
, IServiceProvider
*caller
)
920 DISPPARAMS dp
= {NULL
, NULL
, 0, 0};
924 hres
= find_prop_name_prot(obj
, name
, FALSE
, &prop
);
928 V_VT(var
) = VT_EMPTY
;
932 return prop_get(obj
, prop
, lcid
, &dp
, var
, ei
, caller
);
935 HRESULT
jsdisp_propget_idx(DispatchEx
*obj
, DWORD idx
, LCID lcid
, VARIANT
*var
, jsexcept_t
*ei
, IServiceProvider
*caller
)
939 static const WCHAR formatW
[] = {'%','d',0};
941 sprintfW(buf
, formatW
, idx
);
942 return jsdisp_propget_name(obj
, buf
, lcid
, var
, ei
, caller
);
945 HRESULT
disp_propget(IDispatch
*disp
, DISPID id
, LCID lcid
, VARIANT
*val
, jsexcept_t
*ei
, IServiceProvider
*caller
)
947 DISPPARAMS dp
= {NULL
,NULL
,0,0};
952 jsdisp
= iface_to_jsdisp((IUnknown
*)disp
);
956 prop
= get_prop(jsdisp
, id
);
958 hres
= prop_get(jsdisp
, prop
, lcid
, &dp
, val
, ei
, caller
);
960 hres
= DISP_E_MEMBERNOTFOUND
;
962 IDispatchEx_Release(_IDispatchEx_(jsdisp
));
966 hres
= IDispatch_QueryInterface(disp
, &IID_IDispatchEx
, (void**)&dispex
);
970 TRACE("uding IDispatch\n");
971 return IDispatchEx_Invoke(dispex
, id
, &IID_NULL
, lcid
, INVOKE_PROPERTYGET
, &dp
, val
, &ei
->ei
, &err
);
974 hres
= IDispatchEx_InvokeEx(dispex
, id
, lcid
, INVOKE_PROPERTYGET
, &dp
, val
, &ei
->ei
, caller
);
975 IDispatchEx_Release(dispex
);