1350 lines
44 KiB
C
1350 lines
44 KiB
C
/*
|
|
* Copyright 2011 Jacek Caban for CodeWeavers
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
|
|
#define COBJMACROS
|
|
#define CONST_VTABLE
|
|
|
|
#include <ole2.h>
|
|
#include <dispex.h>
|
|
#include <activscp.h>
|
|
|
|
#include "wine/test.h"
|
|
|
|
extern const CLSID CLSID_VBScript;
|
|
|
|
#define DEFINE_EXPECT(func) \
|
|
static BOOL expect_ ## func = FALSE, called_ ## func = FALSE
|
|
|
|
#define SET_EXPECT(func) \
|
|
expect_ ## func = TRUE
|
|
|
|
#define CHECK_EXPECT2(func) \
|
|
do { \
|
|
ok(expect_ ##func, "unexpected call " #func "\n"); \
|
|
called_ ## func = TRUE; \
|
|
}while(0)
|
|
|
|
#define CHECK_EXPECT(func) \
|
|
do { \
|
|
CHECK_EXPECT2(func); \
|
|
expect_ ## func = FALSE; \
|
|
}while(0)
|
|
|
|
#define CHECK_CALLED(func) \
|
|
do { \
|
|
ok(called_ ## func, "expected " #func "\n"); \
|
|
expect_ ## func = called_ ## func = FALSE; \
|
|
}while(0)
|
|
|
|
DEFINE_EXPECT(global_success_d);
|
|
DEFINE_EXPECT(global_success_i);
|
|
DEFINE_EXPECT(global_vbvar_d);
|
|
DEFINE_EXPECT(global_vbvar_i);
|
|
DEFINE_EXPECT(testobj_propget_d);
|
|
DEFINE_EXPECT(testobj_propget_i);
|
|
DEFINE_EXPECT(testobj_propput_d);
|
|
DEFINE_EXPECT(testobj_propput_i);
|
|
DEFINE_EXPECT(global_propargput_d);
|
|
DEFINE_EXPECT(global_propargput_i);
|
|
|
|
#define DISPID_GLOBAL_REPORTSUCCESS 1000
|
|
#define DISPID_GLOBAL_TRACE 1001
|
|
#define DISPID_GLOBAL_OK 1002
|
|
#define DISPID_GLOBAL_GETVT 1003
|
|
#define DISPID_GLOBAL_ISENGLANG 1004
|
|
#define DISPID_GLOBAL_VBVAR 1005
|
|
#define DISPID_GLOBAL_TESTOBJ 1006
|
|
#define DISPID_GLOBAL_ISNULLDISP 1007
|
|
#define DISPID_GLOBAL_TESTDISP 1008
|
|
#define DISPID_GLOBAL_REFOBJ 1009
|
|
#define DISPID_GLOBAL_PROPARGPUT 1010
|
|
#define DISPID_GLOBAL_COUNTER 1011
|
|
|
|
#define DISPID_TESTOBJ_PROPGET 2000
|
|
#define DISPID_TESTOBJ_PROPPUT 2001
|
|
|
|
static const WCHAR testW[] = {'t','e','s','t',0};
|
|
|
|
static BOOL strict_dispid_check;
|
|
static const char *test_name = "(null)";
|
|
static int test_counter;
|
|
|
|
static BSTR a2bstr(const char *str)
|
|
{
|
|
BSTR ret;
|
|
int len;
|
|
|
|
len = MultiByteToWideChar(CP_ACP, 0, str, -1, NULL, 0);
|
|
ret = SysAllocStringLen(NULL, len-1);
|
|
MultiByteToWideChar(CP_ACP, 0, str, -1, ret, len);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int strcmp_wa(LPCWSTR strw, const char *stra)
|
|
{
|
|
CHAR buf[512];
|
|
WideCharToMultiByte(CP_ACP, 0, strw, -1, buf, sizeof(buf), 0, 0);
|
|
return lstrcmpA(buf, stra);
|
|
}
|
|
|
|
static const char *vt2a(VARIANT *v)
|
|
{
|
|
if(V_VT(v) == (VT_BYREF|VT_VARIANT)) {
|
|
static char buf[64];
|
|
sprintf(buf, "%s*", vt2a(V_BYREF(v)));
|
|
return buf;
|
|
}
|
|
|
|
switch(V_VT(v)) {
|
|
case VT_EMPTY:
|
|
return "VT_EMPTY";
|
|
case VT_NULL:
|
|
return "VT_NULL";
|
|
case VT_I2:
|
|
return "VT_I2";
|
|
case VT_I4:
|
|
return "VT_I4";
|
|
case VT_R8:
|
|
return "VT_R8";
|
|
case VT_BSTR:
|
|
return "VT_BSTR";
|
|
case VT_DISPATCH:
|
|
return "VT_DISPATCH";
|
|
case VT_BOOL:
|
|
return "VT_BOOL";
|
|
case VT_ARRAY|VT_VARIANT:
|
|
return "VT_ARRAY|VT_VARIANT";
|
|
default:
|
|
ok(0, "unknown vt %d\n", V_VT(v));
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
/* Returns true if the user interface is in English. Note that this does not
|
|
* presume of the formatting of dates, numbers, etc.
|
|
*/
|
|
static BOOL is_lang_english(void)
|
|
{
|
|
static HMODULE hkernel32 = NULL;
|
|
static LANGID (WINAPI *pGetThreadUILanguage)(void) = NULL;
|
|
static LANGID (WINAPI *pGetUserDefaultUILanguage)(void) = NULL;
|
|
|
|
if (!hkernel32)
|
|
{
|
|
hkernel32 = GetModuleHandleA("kernel32.dll");
|
|
pGetThreadUILanguage = (void*)GetProcAddress(hkernel32, "GetThreadUILanguage");
|
|
pGetUserDefaultUILanguage = (void*)GetProcAddress(hkernel32, "GetUserDefaultUILanguage");
|
|
}
|
|
if (pGetThreadUILanguage)
|
|
return PRIMARYLANGID(pGetThreadUILanguage()) == LANG_ENGLISH;
|
|
if (pGetUserDefaultUILanguage)
|
|
return PRIMARYLANGID(pGetUserDefaultUILanguage()) == LANG_ENGLISH;
|
|
|
|
return PRIMARYLANGID(GetUserDefaultLangID()) == LANG_ENGLISH;
|
|
}
|
|
|
|
static void test_disp(IDispatch *disp)
|
|
{
|
|
DISPID id, public_prop_id, public_prop2_id, public_func_id, public_sub_id, defvalget_id;
|
|
DISPID named_args[5] = {DISPID_PROPERTYPUT};
|
|
VARIANT v, args[5];
|
|
DISPPARAMS dp = {args, named_args};
|
|
IDispatchEx *dispex;
|
|
EXCEPINFO ei = {0};
|
|
BSTR str;
|
|
HRESULT hres;
|
|
|
|
hres = IDispatch_QueryInterface(disp, &IID_IDispatchEx, (void**)&dispex);
|
|
ok(hres == S_OK, "Could not get IDispatchEx iface: %08x\n", hres);
|
|
|
|
str = a2bstr("publicProp");
|
|
hres = IDispatchEx_GetDispID(dispex, str, fdexNameCaseInsensitive, &public_prop_id);
|
|
SysFreeString(str);
|
|
ok(hres == S_OK, "GetDispID(publicProp) failed: %08x\n", hres);
|
|
|
|
str = a2bstr("PUBLICPROP");
|
|
hres = IDispatchEx_GetDispID(dispex, str, 0, &id);
|
|
SysFreeString(str);
|
|
ok(hres == S_OK, "GetDispID(PUBLICPROP) failed: %08x\n", hres);
|
|
ok(public_prop_id == id, "id = %d\n", public_prop_id);
|
|
|
|
str = a2bstr("publicPROP2");
|
|
hres = IDispatchEx_GetDispID(dispex, str, fdexNameCaseInsensitive, &public_prop2_id);
|
|
SysFreeString(str);
|
|
ok(hres == S_OK, "GetDispID(publicProp2) failed: %08x\n", hres);
|
|
|
|
str = a2bstr("defValGet");
|
|
hres = IDispatchEx_GetDispID(dispex, str, fdexNameCaseInsensitive, &defvalget_id);
|
|
SysFreeString(str);
|
|
ok(hres == S_OK, "GetDispID(defValGet) failed: %08x\n", hres);
|
|
ok(defvalget_id == DISPID_VALUE, "id = %d\n", defvalget_id);
|
|
|
|
str = a2bstr("privateProp");
|
|
hres = IDispatchEx_GetDispID(dispex, str, fdexNameCaseInsensitive, &id);
|
|
SysFreeString(str);
|
|
ok(hres == DISP_E_UNKNOWNNAME, "GetDispID(privateProp) failed: %08x, expected DISP_E_UNKNOWNNAME\n", hres);
|
|
ok(id == -1, "id = %d\n", id);
|
|
|
|
str = a2bstr("class_initialize");
|
|
hres = IDispatchEx_GetDispID(dispex, str, fdexNameCaseInsensitive, &id);
|
|
SysFreeString(str);
|
|
ok(hres == S_OK, "GetDispID(publicProp2) failed: %08x\n", hres);
|
|
|
|
hres = IDispatchEx_InvokeEx(dispex, public_prop_id, 0, DISPATCH_PROPERTYGET|DISPATCH_METHOD, &dp, &v, &ei, NULL);
|
|
ok(hres == S_OK, "InvokeEx failed: %08x\n", hres);
|
|
ok(V_VT(&v) == VT_EMPTY, "V_VT(v) = %d\n", V_VT(&v));
|
|
|
|
V_VT(args) = VT_BOOL;
|
|
V_BOOL(args) = VARIANT_TRUE;
|
|
dp.cArgs = dp.cNamedArgs = 1;
|
|
V_VT(&v) = VT_BOOL;
|
|
hres = IDispatchEx_InvokeEx(dispex, public_prop_id, 0, DISPATCH_PROPERTYPUT, &dp, &v, &ei, NULL);
|
|
ok(hres == S_OK, "InvokeEx failed: %08x\n", hres);
|
|
ok(V_VT(&v) == VT_EMPTY, "V_VT(v) = %d\n", V_VT(&v));
|
|
|
|
dp.cArgs = dp.cNamedArgs = 0;
|
|
hres = IDispatchEx_InvokeEx(dispex, public_prop_id, 0, DISPATCH_PROPERTYGET|DISPATCH_METHOD, &dp, &v, &ei, NULL);
|
|
ok(hres == S_OK, "InvokeEx failed: %08x\n", hres);
|
|
ok(V_VT(&v) == VT_BOOL, "V_VT(v) = %d\n", V_VT(&v));
|
|
ok(V_BOOL(&v), "V_BOOL(v) = %x\n", V_BOOL(&v));
|
|
|
|
dp.cArgs = 1;
|
|
hres = IDispatchEx_InvokeEx(dispex, public_prop_id, 0, DISPATCH_PROPERTYGET|DISPATCH_METHOD, &dp, &v, &ei, NULL);
|
|
ok(hres == DISP_E_MEMBERNOTFOUND, "InvokeEx failed: %08x, expected DISP_E_MEMBERNOTFOUND\n", hres);
|
|
ok(V_VT(&v) == VT_EMPTY, "V_VT(v) = %d\n", V_VT(&v));
|
|
|
|
V_VT(args) = VT_BOOL;
|
|
V_BOOL(args) = VARIANT_FALSE;
|
|
dp.cArgs = 1;
|
|
V_VT(&v) = VT_BOOL;
|
|
hres = IDispatchEx_InvokeEx(dispex, public_prop_id, 0, DISPATCH_PROPERTYPUT, &dp, NULL, &ei, NULL);
|
|
ok(hres == DISP_E_PARAMNOTOPTIONAL, "InvokeEx failed: %08x, expected DISP_E_PARAMNOTOPTIONAL\n", hres);
|
|
|
|
str = a2bstr("publicFunction");
|
|
hres = IDispatchEx_GetDispID(dispex, str, fdexNameCaseInsensitive, &public_func_id);
|
|
SysFreeString(str);
|
|
ok(hres == S_OK, "GetDispID(publicFunction) failed: %08x\n", hres);
|
|
ok(public_func_id != -1, "public_func_id = -1\n");
|
|
|
|
str = a2bstr("publicSub");
|
|
hres = IDispatchEx_GetDispID(dispex, str, fdexNameCaseInsensitive, &public_sub_id);
|
|
SysFreeString(str);
|
|
ok(hres == S_OK, "GetDispID(publicSub) failed: %08x\n", hres);
|
|
ok(public_sub_id != -1, "public_func_id = -1\n");
|
|
|
|
dp.cArgs = dp.cNamedArgs = 0;
|
|
hres = IDispatchEx_InvokeEx(dispex, public_func_id, 0, DISPATCH_PROPERTYGET|DISPATCH_METHOD, &dp, &v, &ei, NULL);
|
|
ok(hres == S_OK, "InvokeEx failed: %08x\n", hres);
|
|
ok(V_VT(&v) == VT_I2, "V_VT(v) = %d\n", V_VT(&v));
|
|
ok(V_I2(&v) == 4, "V_I2(v) = %d\n", V_I2(&v));
|
|
|
|
dp.cArgs = dp.cNamedArgs = 0;
|
|
hres = IDispatchEx_InvokeEx(dispex, public_func_id, 0, DISPATCH_PROPERTYGET, &dp, &v, &ei, NULL);
|
|
ok(hres == DISP_E_MEMBERNOTFOUND, "InvokeEx failed: %08x, expected DISP_E_MEMBERNOTFOUND\n", hres);
|
|
ok(V_VT(&v) == VT_EMPTY, "V_VT(v) = %d\n", V_VT(&v));
|
|
|
|
dp.cArgs = dp.cNamedArgs = 0;
|
|
hres = IDispatchEx_InvokeEx(dispex, public_func_id, 0, DISPATCH_METHOD, &dp, &v, &ei, NULL);
|
|
ok(hres == S_OK, "InvokeEx failed: %08x\n", hres);
|
|
ok(V_VT(&v) == VT_I2, "V_VT(v) = %d\n", V_VT(&v));
|
|
ok(V_I2(&v) == 4, "V_I2(v) = %d\n", V_I2(&v));
|
|
|
|
dp.cArgs = dp.cNamedArgs = 0;
|
|
hres = IDispatchEx_InvokeEx(dispex, public_sub_id, 0, DISPATCH_PROPERTYGET|DISPATCH_METHOD, &dp, &v, &ei, NULL);
|
|
ok(hres == S_OK, "InvokeEx failed: %08x\n", hres);
|
|
ok(V_VT(&v) == VT_EMPTY, "V_VT(v) = %d\n", V_VT(&v));
|
|
|
|
dp.cArgs = dp.cNamedArgs = 0;
|
|
hres = IDispatchEx_InvokeEx(dispex, public_sub_id, 0, DISPATCH_PROPERTYGET, &dp, &v, &ei, NULL);
|
|
ok(hres == DISP_E_MEMBERNOTFOUND, "InvokeEx failed: %08x, expected DISP_E_MEMBERNOTFOUND\n", hres);
|
|
ok(V_VT(&v) == VT_EMPTY, "V_VT(v) = %d\n", V_VT(&v));
|
|
|
|
dp.cArgs = dp.cNamedArgs = 0;
|
|
hres = IDispatchEx_InvokeEx(dispex, public_sub_id, 0, DISPATCH_METHOD, &dp, &v, &ei, NULL);
|
|
ok(hres == S_OK, "InvokeEx failed: %08x\n", hres);
|
|
ok(V_VT(&v) == VT_EMPTY, "V_VT(v) = %d\n", V_VT(&v));
|
|
|
|
V_VT(args) = VT_BOOL;
|
|
V_BOOL(args) = VARIANT_TRUE;
|
|
dp.cArgs = dp.cNamedArgs = 1;
|
|
hres = IDispatchEx_InvokeEx(dispex, public_sub_id, 0, DISPATCH_PROPERTYPUT, &dp, NULL, &ei, NULL);
|
|
ok(FAILED(hres), "InvokeEx succeeded: %08x\n", hres);
|
|
|
|
dp.cArgs = dp.cNamedArgs = 0;
|
|
hres = IDispatchEx_InvokeEx(dispex, public_func_id, 0, DISPATCH_PROPERTYGET|DISPATCH_METHOD, &dp, &v, &ei, NULL);
|
|
ok(hres == S_OK, "InvokeEx failed: %08x\n", hres);
|
|
ok(V_VT(&v) == VT_I2, "V_VT(v) = %d\n", V_VT(&v));
|
|
ok(V_I2(&v) == 4, "V_I2(v) = %d\n", V_I2(&v));
|
|
|
|
dp.cArgs = dp.cNamedArgs = 0;
|
|
hres = IDispatchEx_InvokeEx(dispex, public_func_id, 0, DISPATCH_METHOD, &dp, &v, &ei, NULL);
|
|
ok(hres == S_OK, "InvokeEx failed: %08x\n", hres);
|
|
ok(V_VT(&v) == VT_I2, "V_VT(v) = %d\n", V_VT(&v));
|
|
ok(V_I2(&v) == 4, "V_I2(v) = %d\n", V_I2(&v));
|
|
|
|
dp.cArgs = dp.cNamedArgs = 0;
|
|
hres = IDispatchEx_InvokeEx(dispex, public_sub_id, 0, DISPATCH_PROPERTYGET|DISPATCH_METHOD, &dp, &v, &ei, NULL);
|
|
ok(hres == S_OK, "InvokeEx failed: %08x\n", hres);
|
|
ok(V_VT(&v) == VT_EMPTY, "V_VT(v) = %d\n", V_VT(&v));
|
|
|
|
dp.cArgs = dp.cNamedArgs = 0;
|
|
hres = IDispatchEx_InvokeEx(dispex, public_sub_id, 0, DISPATCH_METHOD, &dp, &v, &ei, NULL);
|
|
ok(hres == S_OK, "InvokeEx failed: %08x\n", hres);
|
|
ok(V_VT(&v) == VT_EMPTY, "V_VT(v) = %d\n", V_VT(&v));
|
|
|
|
str = a2bstr("privateSub");
|
|
hres = IDispatchEx_GetDispID(dispex, str, fdexNameCaseInsensitive, &id);
|
|
SysFreeString(str);
|
|
ok(hres == DISP_E_UNKNOWNNAME, "GetDispID(privateSub) failed: %08x, expected DISP_E_UNKNOWNNAME\n", hres);
|
|
ok(id == -1, "id = %d\n", id);
|
|
|
|
str = a2bstr("dynprop");
|
|
hres = IDispatchEx_GetDispID(dispex, str, fdexNameCaseInsensitive|fdexNameEnsure, &id);
|
|
ok(hres == DISP_E_UNKNOWNNAME, "GetDispID(privateProp) failed: %08x, expected DISP_E_UNKNOWNNAME\n", hres);
|
|
ok(id == -1, "id = %d\n", id);
|
|
hres = IDispatchEx_GetDispID(dispex, str, fdexNameEnsure, &id);
|
|
ok(hres == DISP_E_UNKNOWNNAME, "GetDispID(privateProp) failed: %08x, expected DISP_E_UNKNOWNNAME\n", hres);
|
|
ok(id == -1, "id = %d\n", id);
|
|
SysFreeString(str);
|
|
|
|
IDispatchEx_Release(dispex);
|
|
}
|
|
|
|
#define test_grfdex(a,b) _test_grfdex(__LINE__,a,b)
|
|
static void _test_grfdex(unsigned line, DWORD grfdex, DWORD expect)
|
|
{
|
|
ok_(__FILE__,line)(grfdex == expect, "grfdex = %x, expected %x\n", grfdex, expect);
|
|
}
|
|
|
|
static HRESULT WINAPI DispatchEx_QueryInterface(IDispatchEx *iface, REFIID riid, void **ppv)
|
|
{
|
|
*ppv = NULL;
|
|
|
|
if(IsEqualGUID(riid, &IID_IUnknown)
|
|
|| IsEqualGUID(riid, &IID_IDispatch)
|
|
|| IsEqualGUID(riid, &IID_IDispatchEx))
|
|
*ppv = iface;
|
|
else
|
|
return E_NOINTERFACE;
|
|
|
|
IUnknown_AddRef((IUnknown*)*ppv);
|
|
return S_OK;
|
|
}
|
|
|
|
static ULONG WINAPI DispatchEx_AddRef(IDispatchEx *iface)
|
|
{
|
|
return 2;
|
|
}
|
|
|
|
static ULONG WINAPI DispatchEx_Release(IDispatchEx *iface)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
static HRESULT WINAPI DispatchEx_GetTypeInfoCount(IDispatchEx *iface, UINT *pctinfo)
|
|
{
|
|
ok(0, "unexpected call\n");
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI DispatchEx_GetTypeInfo(IDispatchEx *iface, UINT iTInfo,
|
|
LCID lcid, ITypeInfo **ppTInfo)
|
|
{
|
|
ok(0, "unexpected call\n");
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI DispatchEx_GetIDsOfNames(IDispatchEx *iface, REFIID riid,
|
|
LPOLESTR *rgszNames, UINT cNames,
|
|
LCID lcid, DISPID *rgDispId)
|
|
{
|
|
ok(0, "unexpected call\n");
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI DispatchEx_Invoke(IDispatchEx *iface, DISPID dispIdMember,
|
|
REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS *pDispParams,
|
|
VARIANT *pVarResult, EXCEPINFO *pExcepInfo, UINT *puArgErr)
|
|
{
|
|
ok(0, "unexpected call\n");
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI DispatchEx_DeleteMemberByName(IDispatchEx *iface, BSTR bstrName, DWORD grfdex)
|
|
{
|
|
ok(0, "unexpected call %s %x\n", wine_dbgstr_w(bstrName), grfdex);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI DispatchEx_DeleteMemberByDispID(IDispatchEx *iface, DISPID id)
|
|
{
|
|
ok(0, "unexpected call\n");
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI DispatchEx_GetMemberProperties(IDispatchEx *iface, DISPID id, DWORD grfdexFetch, DWORD *pgrfdex)
|
|
{
|
|
ok(0, "unexpected call\n");
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI DispatchEx_GetMemberName(IDispatchEx *iface, DISPID id, BSTR *pbstrName)
|
|
{
|
|
ok(0, "unexpected call\n");
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI DispatchEx_GetNextDispID(IDispatchEx *iface, DWORD grfdex, DISPID id, DISPID *pid)
|
|
{
|
|
ok(0, "unexpected call\n");
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI DispatchEx_GetNameSpaceParent(IDispatchEx *iface, IUnknown **ppunk)
|
|
{
|
|
ok(0, "unexpected call\n");
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI DispatchEx_GetDispID(IDispatchEx *iface, BSTR bstrName, DWORD grfdex, DISPID *pid)
|
|
{
|
|
ok(0, "unexpected call\n");
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI DispatchEx_InvokeEx(IDispatchEx *iface, DISPID id, LCID lcid, WORD wFlags, DISPPARAMS *pdp,
|
|
VARIANT *pvarRes, EXCEPINFO *pei, IServiceProvider *pspCaller)
|
|
{
|
|
ok(0, "unexpected call\n");
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI testObj_GetDispID(IDispatchEx *iface, BSTR bstrName, DWORD grfdex, DISPID *pid)
|
|
{
|
|
if(!strcmp_wa(bstrName, "propget")) {
|
|
CHECK_EXPECT(testobj_propget_d);
|
|
test_grfdex(grfdex, fdexNameCaseInsensitive);
|
|
*pid = DISPID_TESTOBJ_PROPGET;
|
|
return S_OK;
|
|
}
|
|
if(!strcmp_wa(bstrName, "propput")) {
|
|
CHECK_EXPECT(testobj_propput_d);
|
|
test_grfdex(grfdex, fdexNameCaseInsensitive);
|
|
*pid = DISPID_TESTOBJ_PROPPUT;
|
|
return S_OK;
|
|
}
|
|
|
|
ok(0, "unexpected call %s\n", wine_dbgstr_w(bstrName));
|
|
return DISP_E_UNKNOWNNAME;
|
|
}
|
|
|
|
static HRESULT WINAPI testObj_InvokeEx(IDispatchEx *iface, DISPID id, LCID lcid, WORD wFlags, DISPPARAMS *pdp,
|
|
VARIANT *pvarRes, EXCEPINFO *pei, IServiceProvider *pspCaller)
|
|
{
|
|
switch(id) {
|
|
case DISPID_TESTOBJ_PROPGET:
|
|
CHECK_EXPECT(testobj_propget_i);
|
|
|
|
ok(wFlags == (DISPATCH_PROPERTYGET|DISPATCH_METHOD), "wFlags = %x\n", wFlags);
|
|
ok(pdp != NULL, "pdp == NULL\n");
|
|
ok(!pdp->rgvarg, "rgvarg == NULL\n");
|
|
ok(!pdp->rgdispidNamedArgs, "rgdispidNamedArgs != NULL\n");
|
|
ok(!pdp->cArgs, "cArgs = %d\n", pdp->cArgs);
|
|
ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
|
|
ok(pvarRes != NULL, "pvarRes == NULL\n");
|
|
ok(pei != NULL, "pei == NULL\n");
|
|
|
|
V_VT(pvarRes) = VT_I2;
|
|
V_I2(pvarRes) = 10;
|
|
return S_OK;
|
|
case DISPID_TESTOBJ_PROPPUT:
|
|
CHECK_EXPECT(testobj_propput_i);
|
|
|
|
ok(wFlags == DISPATCH_PROPERTYPUT, "wFlags = %x\n", wFlags);
|
|
ok(pdp != NULL, "pdp == NULL\n");
|
|
ok(pdp->rgvarg != NULL, "rgvarg == NULL\n");
|
|
ok(pdp->rgdispidNamedArgs != NULL, "rgdispidNamedArgs == NULL\n");
|
|
ok(pdp->cArgs == 1, "cArgs = %d\n", pdp->cArgs);
|
|
ok(pdp->cNamedArgs == 1, "cNamedArgs = %d\n", pdp->cNamedArgs);
|
|
ok(pdp->rgdispidNamedArgs[0] == DISPID_PROPERTYPUT, "pdp->rgdispidNamedArgs[0] = %d\n", pdp->rgdispidNamedArgs[0]);
|
|
ok(!pvarRes, "pvarRes != NULL\n");
|
|
ok(pei != NULL, "pei == NULL\n");
|
|
|
|
ok(V_VT(pdp->rgvarg) == VT_I2, "V_VT(psp->rgvargs) = %d\n", V_VT(pdp->rgvarg));
|
|
ok(V_I2(pdp->rgvarg) == 1, "V_I2(psp->rgvargs) = %d\n", V_I2(pdp->rgvarg));
|
|
return S_OK;
|
|
}
|
|
|
|
ok(0, "unexpected call %d\n", id);
|
|
return E_FAIL;
|
|
}
|
|
|
|
static IDispatchExVtbl testObjVtbl = {
|
|
DispatchEx_QueryInterface,
|
|
DispatchEx_AddRef,
|
|
DispatchEx_Release,
|
|
DispatchEx_GetTypeInfoCount,
|
|
DispatchEx_GetTypeInfo,
|
|
DispatchEx_GetIDsOfNames,
|
|
DispatchEx_Invoke,
|
|
testObj_GetDispID,
|
|
testObj_InvokeEx,
|
|
DispatchEx_DeleteMemberByName,
|
|
DispatchEx_DeleteMemberByDispID,
|
|
DispatchEx_GetMemberProperties,
|
|
DispatchEx_GetMemberName,
|
|
DispatchEx_GetNextDispID,
|
|
DispatchEx_GetNameSpaceParent
|
|
};
|
|
|
|
static IDispatchEx testObj = { &testObjVtbl };
|
|
|
|
static ULONG refobj_ref;
|
|
|
|
static ULONG WINAPI RefObj_AddRef(IDispatchEx *iface)
|
|
{
|
|
return ++refobj_ref;
|
|
}
|
|
|
|
static ULONG WINAPI RefObj_Release(IDispatchEx *iface)
|
|
{
|
|
return --refobj_ref;
|
|
}
|
|
|
|
static IDispatchExVtbl RefObjVtbl = {
|
|
DispatchEx_QueryInterface,
|
|
RefObj_AddRef,
|
|
RefObj_Release,
|
|
DispatchEx_GetTypeInfoCount,
|
|
DispatchEx_GetTypeInfo,
|
|
DispatchEx_GetIDsOfNames,
|
|
DispatchEx_Invoke,
|
|
DispatchEx_GetDispID,
|
|
DispatchEx_InvokeEx,
|
|
DispatchEx_DeleteMemberByName,
|
|
DispatchEx_DeleteMemberByDispID,
|
|
DispatchEx_GetMemberProperties,
|
|
DispatchEx_GetMemberName,
|
|
DispatchEx_GetNextDispID,
|
|
DispatchEx_GetNameSpaceParent
|
|
};
|
|
|
|
static IDispatchEx RefObj = { &RefObjVtbl };
|
|
|
|
static HRESULT WINAPI Global_GetDispID(IDispatchEx *iface, BSTR bstrName, DWORD grfdex, DISPID *pid)
|
|
{
|
|
if(!strcmp_wa(bstrName, "ok")) {
|
|
test_grfdex(grfdex, fdexNameCaseInsensitive);
|
|
*pid = DISPID_GLOBAL_OK;
|
|
return S_OK;
|
|
}
|
|
if(!strcmp_wa(bstrName, "trace")) {
|
|
test_grfdex(grfdex, fdexNameCaseInsensitive);
|
|
*pid = DISPID_GLOBAL_TRACE;
|
|
return S_OK;
|
|
}
|
|
if(!strcmp_wa(bstrName, "reportSuccess")) {
|
|
CHECK_EXPECT(global_success_d);
|
|
test_grfdex(grfdex, fdexNameCaseInsensitive);
|
|
*pid = DISPID_GLOBAL_REPORTSUCCESS;
|
|
return S_OK;
|
|
}
|
|
if(!strcmp_wa(bstrName, "getVT")) {
|
|
test_grfdex(grfdex, fdexNameCaseInsensitive);
|
|
*pid = DISPID_GLOBAL_GETVT;
|
|
return S_OK;
|
|
}
|
|
if(!strcmp_wa(bstrName, "isEnglishLang")) {
|
|
test_grfdex(grfdex, fdexNameCaseInsensitive);
|
|
*pid = DISPID_GLOBAL_ISENGLANG;
|
|
return S_OK;
|
|
}
|
|
if(!strcmp_wa(bstrName, "testObj")) {
|
|
test_grfdex(grfdex, fdexNameCaseInsensitive);
|
|
*pid = DISPID_GLOBAL_TESTOBJ;
|
|
return S_OK;
|
|
}
|
|
if(!strcmp_wa(bstrName, "vbvar")) {
|
|
CHECK_EXPECT(global_vbvar_d);
|
|
test_grfdex(grfdex, fdexNameCaseInsensitive);
|
|
*pid = DISPID_GLOBAL_VBVAR;
|
|
return S_OK;
|
|
}
|
|
if(!strcmp_wa(bstrName, "isNullDisp")) {
|
|
test_grfdex(grfdex, fdexNameCaseInsensitive);
|
|
*pid = DISPID_GLOBAL_ISNULLDISP;
|
|
return S_OK;
|
|
}
|
|
if(!strcmp_wa(bstrName, "testDisp")) {
|
|
test_grfdex(grfdex, fdexNameCaseInsensitive);
|
|
*pid = DISPID_GLOBAL_TESTDISP;
|
|
return S_OK;
|
|
}
|
|
if(!strcmp_wa(bstrName, "RefObj")) {
|
|
test_grfdex(grfdex, fdexNameCaseInsensitive);
|
|
*pid = DISPID_GLOBAL_REFOBJ;
|
|
return S_OK;
|
|
}
|
|
if(!strcmp_wa(bstrName, "propargput")) {
|
|
CHECK_EXPECT(global_propargput_d);
|
|
test_grfdex(grfdex, fdexNameCaseInsensitive);
|
|
*pid = DISPID_GLOBAL_PROPARGPUT;
|
|
return S_OK;
|
|
}
|
|
if(!strcmp_wa(bstrName, "counter")) {
|
|
test_grfdex(grfdex, fdexNameCaseInsensitive);
|
|
*pid = DISPID_GLOBAL_COUNTER;
|
|
return S_OK;
|
|
}
|
|
|
|
if(strict_dispid_check && strcmp_wa(bstrName, "x"))
|
|
ok(0, "unexpected call %s %x\n", wine_dbgstr_w(bstrName), grfdex);
|
|
return DISP_E_UNKNOWNNAME;
|
|
}
|
|
|
|
static HRESULT WINAPI Global_InvokeEx(IDispatchEx *iface, DISPID id, LCID lcid, WORD wFlags, DISPPARAMS *pdp,
|
|
VARIANT *pvarRes, EXCEPINFO *pei, IServiceProvider *pspCaller)
|
|
{
|
|
switch(id) {
|
|
case DISPID_GLOBAL_OK: {
|
|
VARIANT *b;
|
|
|
|
ok(wFlags == INVOKE_FUNC || wFlags == (INVOKE_FUNC|INVOKE_PROPERTYGET), "wFlags = %x\n", wFlags);
|
|
ok(pdp != NULL, "pdp == NULL\n");
|
|
ok(pdp->rgvarg != NULL, "rgvarg == NULL\n");
|
|
ok(!pdp->rgdispidNamedArgs, "rgdispidNamedArgs != NULL\n");
|
|
ok(pdp->cArgs == 2, "cArgs = %d\n", pdp->cArgs);
|
|
ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
|
|
if(wFlags & INVOKE_PROPERTYGET)
|
|
ok(pvarRes != NULL, "pvarRes == NULL\n");
|
|
else
|
|
ok(!pvarRes, "pvarRes != NULL\n");
|
|
ok(pei != NULL, "pei == NULL\n");
|
|
|
|
ok(V_VT(pdp->rgvarg) == VT_BSTR, "V_VT(psp->rgvargs) = %d\n", V_VT(pdp->rgvarg));
|
|
|
|
b = pdp->rgvarg+1;
|
|
if(V_VT(b) == (VT_BYREF|VT_VARIANT))
|
|
b = V_BYREF(b);
|
|
|
|
ok(V_VT(b) == VT_BOOL, "V_VT(b) = %d\n", V_VT(b));
|
|
|
|
ok(V_BOOL(b), "%s: %s\n", test_name, wine_dbgstr_w(V_BSTR(pdp->rgvarg)));
|
|
return S_OK;
|
|
}
|
|
|
|
case DISPID_GLOBAL_TRACE:
|
|
ok(wFlags == INVOKE_FUNC, "wFlags = %x\n", wFlags);
|
|
ok(pdp != NULL, "pdp == NULL\n");
|
|
ok(pdp->rgvarg != NULL, "rgvarg == NULL\n");
|
|
ok(!pdp->rgdispidNamedArgs, "rgdispidNamedArgs != NULL\n");
|
|
ok(pdp->cArgs == 1, "cArgs = %d\n", pdp->cArgs);
|
|
ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
|
|
ok(!pvarRes, "pvarRes != NULL\n");
|
|
ok(pei != NULL, "pei == NULL\n");
|
|
|
|
ok(V_VT(pdp->rgvarg) == VT_BSTR, "V_VT(psp->rgvargs) = %d\n", V_VT(pdp->rgvarg));
|
|
if(V_VT(pdp->rgvarg) == VT_BSTR)
|
|
trace("%s: %s\n", test_name, wine_dbgstr_w(V_BSTR(pdp->rgvarg)));
|
|
|
|
return S_OK;
|
|
|
|
case DISPID_GLOBAL_REPORTSUCCESS:
|
|
CHECK_EXPECT(global_success_i);
|
|
|
|
ok(wFlags == INVOKE_FUNC, "wFlags = %x\n", wFlags);
|
|
ok(pdp != NULL, "pdp == NULL\n");
|
|
ok(!pdp->rgdispidNamedArgs, "rgdispidNamedArgs != NULL\n");
|
|
ok(pdp->cArgs == 0, "cArgs = %d\n", pdp->cArgs);
|
|
ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
|
|
ok(!pvarRes, "pvarRes != NULL\n");
|
|
ok(pei != NULL, "pei == NULL\n");
|
|
|
|
return S_OK;
|
|
|
|
case DISPID_GLOBAL_GETVT:
|
|
ok(pdp != NULL, "pdp == NULL\n");
|
|
ok(pdp->rgvarg != NULL, "rgvarg == NULL\n");
|
|
ok(!pdp->rgdispidNamedArgs, "rgdispidNamedArgs != NULL\n");
|
|
ok(pdp->cArgs == 1, "cArgs = %d\n", pdp->cArgs);
|
|
ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
|
|
ok(pvarRes != NULL, "pvarRes == NULL\n");
|
|
ok(V_VT(pvarRes) == VT_EMPTY, "V_VT(pvarRes) = %d\n", V_VT(pvarRes));
|
|
ok(pei != NULL, "pei == NULL\n");
|
|
|
|
V_VT(pvarRes) = VT_BSTR;
|
|
V_BSTR(pvarRes) = a2bstr(vt2a(pdp->rgvarg));
|
|
return S_OK;
|
|
|
|
case DISPID_GLOBAL_ISENGLANG:
|
|
ok(wFlags == (INVOKE_FUNC|INVOKE_PROPERTYGET), "wFlags = %x\n", wFlags);
|
|
ok(pdp != NULL, "pdp == NULL\n");
|
|
ok(!pdp->rgdispidNamedArgs, "rgdispidNamedArgs != NULL\n");
|
|
ok(pdp->cArgs == 0, "cArgs = %d\n", pdp->cArgs);
|
|
ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
|
|
ok(pvarRes != NULL, "pvarRes == NULL\n");
|
|
ok(pei != NULL, "pei == NULL\n");
|
|
|
|
V_VT(pvarRes) = VT_BOOL;
|
|
if(is_lang_english()) {
|
|
V_BOOL(pvarRes) = VARIANT_TRUE;
|
|
}else {
|
|
skip("Skipping some tests in non-English UIs\n");
|
|
V_BOOL(pvarRes) = VARIANT_FALSE;
|
|
}
|
|
return S_OK;
|
|
|
|
case DISPID_GLOBAL_VBVAR:
|
|
CHECK_EXPECT(global_vbvar_i);
|
|
|
|
ok(wFlags == DISPATCH_PROPERTYPUT, "wFlags = %x\n", wFlags);
|
|
ok(pdp != NULL, "pdp == NULL\n");
|
|
ok(pdp->rgvarg != NULL, "rgvarg == NULL\n");
|
|
ok(pdp->rgdispidNamedArgs != NULL, "rgdispidNamedArgs == NULL\n");
|
|
ok(pdp->cArgs == 1, "cArgs = %d\n", pdp->cArgs);
|
|
ok(pdp->cNamedArgs == 1, "cNamedArgs = %d\n", pdp->cNamedArgs);
|
|
ok(pdp->rgdispidNamedArgs[0] == DISPID_PROPERTYPUT, "pdp->rgdispidNamedArgs[0] = %d\n", pdp->rgdispidNamedArgs[0]);
|
|
ok(!pvarRes, "pvarRes != NULL\n");
|
|
ok(pei != NULL, "pei == NULL\n");
|
|
|
|
ok(V_VT(pdp->rgvarg) == VT_I2, "V_VT(psp->rgvargs) = %d\n", V_VT(pdp->rgvarg));
|
|
ok(V_I2(pdp->rgvarg) == 3, "V_I2(psp->rgvargs) = %d\n", V_I2(pdp->rgvarg));
|
|
return S_OK;
|
|
|
|
case DISPID_GLOBAL_TESTOBJ:
|
|
ok(wFlags == (DISPATCH_PROPERTYGET|DISPATCH_METHOD), "wFlags = %x\n", wFlags);
|
|
|
|
ok(pdp != NULL, "pdp == NULL\n");
|
|
ok(!pdp->rgvarg, "rgvarg != NULL\n");
|
|
ok(!pdp->rgdispidNamedArgs, "rgdispidNamedArgs != NULL\n");
|
|
ok(!pdp->cArgs, "cArgs = %d\n", pdp->cArgs);
|
|
ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
|
|
ok(pvarRes != NULL, "pvarRes == NULL\n");
|
|
ok(pei != NULL, "pei == NULL\n");
|
|
|
|
V_VT(pvarRes) = VT_DISPATCH;
|
|
V_DISPATCH(pvarRes) = (IDispatch*)&testObj;
|
|
return S_OK;
|
|
|
|
case DISPID_GLOBAL_REFOBJ:
|
|
ok(wFlags == (DISPATCH_PROPERTYGET|DISPATCH_METHOD), "wFlags = %x\n", wFlags);
|
|
|
|
ok(pdp != NULL, "pdp == NULL\n");
|
|
ok(!pdp->rgvarg, "rgvarg == NULL\n");
|
|
ok(!pdp->rgdispidNamedArgs, "rgdispidNamedArgs != NULL\n");
|
|
ok(!pdp->cArgs, "cArgs = %d\n", pdp->cArgs);
|
|
ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
|
|
ok(pvarRes != NULL, "pvarRes == NULL\n");
|
|
ok(pei != NULL, "pei == NULL\n");
|
|
|
|
IDispatchEx_AddRef(&RefObj);
|
|
V_VT(pvarRes) = VT_DISPATCH;
|
|
V_DISPATCH(pvarRes) = (IDispatch*)&RefObj;
|
|
return S_OK;
|
|
|
|
case DISPID_GLOBAL_ISNULLDISP: {
|
|
VARIANT *v;
|
|
|
|
ok(wFlags == (INVOKE_FUNC|INVOKE_PROPERTYGET), "wFlags = %x\n", wFlags);
|
|
ok(pdp != NULL, "pdp == NULL\n");
|
|
ok(pdp->rgvarg != NULL, "rgvarg == NULL\n");
|
|
ok(!pdp->rgdispidNamedArgs, "rgdispidNamedArgs != NULL\n");
|
|
ok(pdp->cArgs == 1, "cArgs = %d\n", pdp->cArgs);
|
|
ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
|
|
ok(pvarRes != NULL, "pvarRes == NULL\n");
|
|
ok(pei != NULL, "pei == NULL\n");
|
|
|
|
v = pdp->rgvarg;
|
|
if(V_VT(v) == (VT_VARIANT|VT_BYREF))
|
|
v = V_VARIANTREF(v);
|
|
|
|
ok(V_VT(v) == VT_DISPATCH, "V_VT(psp->rgvargs) = %d\n", V_VT(pdp->rgvarg));
|
|
V_VT(pvarRes) = VT_BOOL;
|
|
V_BOOL(pvarRes) = V_DISPATCH(v) ? VARIANT_FALSE : VARIANT_TRUE;
|
|
return S_OK;
|
|
}
|
|
|
|
case DISPID_GLOBAL_TESTDISP:
|
|
ok(wFlags == INVOKE_FUNC, "wFlags = %x\n", wFlags);
|
|
ok(pdp != NULL, "pdp == NULL\n");
|
|
ok(pdp->rgvarg != NULL, "rgvarg == NULL\n");
|
|
ok(!pdp->rgdispidNamedArgs, "rgdispidNamedArgs != NULL\n");
|
|
ok(pdp->cArgs == 1, "cArgs = %d\n", pdp->cArgs);
|
|
ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
|
|
ok(!pvarRes, "pvarRes != NULL\n");
|
|
ok(pei != NULL, "pei == NULL\n");
|
|
|
|
ok(V_VT(pdp->rgvarg) == VT_DISPATCH, "V_VT(psp->rgvargs) = %d\n", V_VT(pdp->rgvarg));
|
|
test_disp(V_DISPATCH(pdp->rgvarg));
|
|
return S_OK;
|
|
|
|
case DISPID_GLOBAL_PROPARGPUT:
|
|
CHECK_EXPECT(global_propargput_i);
|
|
|
|
ok(wFlags == DISPATCH_PROPERTYPUT, "wFlags = %x\n", wFlags);
|
|
ok(pdp != NULL, "pdp == NULL\n");
|
|
ok(pdp->rgvarg != NULL, "rgvarg == NULL\n");
|
|
ok(pdp->rgdispidNamedArgs != NULL, "rgdispidNamedArgs == NULL\n");
|
|
ok(pdp->cArgs == 3, "cArgs = %d\n", pdp->cArgs);
|
|
ok(pdp->cNamedArgs == 1, "cNamedArgs = %d\n", pdp->cNamedArgs);
|
|
ok(pdp->rgdispidNamedArgs[0] == DISPID_PROPERTYPUT, "pdp->rgdispidNamedArgs[0] = %d\n", pdp->rgdispidNamedArgs[0]);
|
|
ok(!pvarRes, "pvarRes != NULL\n");
|
|
ok(pei != NULL, "pei == NULL\n");
|
|
|
|
ok(V_VT(pdp->rgvarg) == VT_I2, "V_VT(psp->rgvargs) = %d\n", V_VT(pdp->rgvarg));
|
|
ok(V_I2(pdp->rgvarg) == 0, "V_I2(psp->rgvargs) = %d\n", V_I2(pdp->rgvarg));
|
|
|
|
ok(V_VT(pdp->rgvarg+1) == VT_I2, "V_VT(psp->rgvargs+1) = %d\n", V_VT(pdp->rgvarg+1));
|
|
ok(V_I2(pdp->rgvarg+1) == 2, "V_I2(psp->rgvargs+1) = %d\n", V_I2(pdp->rgvarg+1));
|
|
|
|
ok(V_VT(pdp->rgvarg+2) == VT_I2, "V_VT(psp->rgvargs+2) = %d\n", V_VT(pdp->rgvarg+2));
|
|
ok(V_I2(pdp->rgvarg+2) == 1, "V_I2(psp->rgvargs+2) = %d\n", V_I2(pdp->rgvarg+2));
|
|
return S_OK;
|
|
|
|
case DISPID_GLOBAL_COUNTER:
|
|
ok(pdp != NULL, "pdp == NULL\n");
|
|
todo_wine ok(pdp->rgvarg != NULL, "rgvarg == NULL\n");
|
|
ok(!pdp->rgdispidNamedArgs, "rgdispidNamedArgs != NULL\n");
|
|
ok(!pdp->cArgs, "cArgs = %d\n", pdp->cArgs);
|
|
ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
|
|
ok(pvarRes != NULL, "pvarRes == NULL\n");
|
|
ok(V_VT(pvarRes) == VT_EMPTY, "V_VT(pvarRes) = %d\n", V_VT(pvarRes));
|
|
ok(pei != NULL, "pei == NULL\n");
|
|
|
|
V_VT(pvarRes) = VT_I2;
|
|
V_I2(pvarRes) = test_counter++;
|
|
return S_OK;
|
|
}
|
|
|
|
ok(0, "unexpected call %d\n", id);
|
|
return DISP_E_MEMBERNOTFOUND;
|
|
}
|
|
|
|
static IDispatchExVtbl GlobalVtbl = {
|
|
DispatchEx_QueryInterface,
|
|
DispatchEx_AddRef,
|
|
DispatchEx_Release,
|
|
DispatchEx_GetTypeInfoCount,
|
|
DispatchEx_GetTypeInfo,
|
|
DispatchEx_GetIDsOfNames,
|
|
DispatchEx_Invoke,
|
|
Global_GetDispID,
|
|
Global_InvokeEx,
|
|
DispatchEx_DeleteMemberByName,
|
|
DispatchEx_DeleteMemberByDispID,
|
|
DispatchEx_GetMemberProperties,
|
|
DispatchEx_GetMemberName,
|
|
DispatchEx_GetNextDispID,
|
|
DispatchEx_GetNameSpaceParent
|
|
};
|
|
|
|
static IDispatchEx Global = { &GlobalVtbl };
|
|
|
|
static HRESULT WINAPI ActiveScriptSite_QueryInterface(IActiveScriptSite *iface, REFIID riid, void **ppv)
|
|
{
|
|
*ppv = NULL;
|
|
|
|
if(IsEqualGUID(&IID_IUnknown, riid))
|
|
*ppv = iface;
|
|
else if(IsEqualGUID(&IID_IActiveScriptSite, riid))
|
|
*ppv = iface;
|
|
else
|
|
return E_NOINTERFACE;
|
|
|
|
IUnknown_AddRef((IUnknown*)*ppv);
|
|
return S_OK;
|
|
}
|
|
|
|
static ULONG WINAPI ActiveScriptSite_AddRef(IActiveScriptSite *iface)
|
|
{
|
|
return 2;
|
|
}
|
|
|
|
static ULONG WINAPI ActiveScriptSite_Release(IActiveScriptSite *iface)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
static HRESULT WINAPI ActiveScriptSite_GetLCID(IActiveScriptSite *iface, LCID *plcid)
|
|
{
|
|
*plcid = GetUserDefaultLCID();
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ActiveScriptSite_GetItemInfo(IActiveScriptSite *iface, LPCOLESTR pstrName,
|
|
DWORD dwReturnMask, IUnknown **ppiunkItem, ITypeInfo **ppti)
|
|
{
|
|
ok(dwReturnMask == SCRIPTINFO_IUNKNOWN, "unexpected dwReturnMask %x\n", dwReturnMask);
|
|
ok(!ppti, "ppti != NULL\n");
|
|
|
|
if(strcmp_wa(pstrName, "test"))
|
|
ok(0, "unexpected pstrName %s\n", wine_dbgstr_w(pstrName));
|
|
|
|
*ppiunkItem = (IUnknown*)&Global;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ActiveScriptSite_GetDocVersionString(IActiveScriptSite *iface, BSTR *pbstrVersion)
|
|
{
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI ActiveScriptSite_OnScriptTerminate(IActiveScriptSite *iface,
|
|
const VARIANT *pvarResult, const EXCEPINFO *pexcepinfo)
|
|
{
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI ActiveScriptSite_OnStateChange(IActiveScriptSite *iface, SCRIPTSTATE ssScriptState)
|
|
{
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI ActiveScriptSite_OnScriptError(IActiveScriptSite *iface, IActiveScriptError *pscripterror)
|
|
{
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI ActiveScriptSite_OnEnterScript(IActiveScriptSite *iface)
|
|
{
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI ActiveScriptSite_OnLeaveScript(IActiveScriptSite *iface)
|
|
{
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
#undef ACTSCPSITE_THIS
|
|
|
|
static const IActiveScriptSiteVtbl ActiveScriptSiteVtbl = {
|
|
ActiveScriptSite_QueryInterface,
|
|
ActiveScriptSite_AddRef,
|
|
ActiveScriptSite_Release,
|
|
ActiveScriptSite_GetLCID,
|
|
ActiveScriptSite_GetItemInfo,
|
|
ActiveScriptSite_GetDocVersionString,
|
|
ActiveScriptSite_OnScriptTerminate,
|
|
ActiveScriptSite_OnStateChange,
|
|
ActiveScriptSite_OnScriptError,
|
|
ActiveScriptSite_OnEnterScript,
|
|
ActiveScriptSite_OnLeaveScript
|
|
};
|
|
|
|
static IActiveScriptSite ActiveScriptSite = { &ActiveScriptSiteVtbl };
|
|
|
|
static IActiveScript *create_script(void)
|
|
{
|
|
IActiveScript *script;
|
|
HRESULT hres;
|
|
|
|
hres = CoCreateInstance(&CLSID_VBScript, NULL, CLSCTX_INPROC_SERVER|CLSCTX_INPROC_HANDLER,
|
|
&IID_IActiveScript, (void**)&script);
|
|
ok(hres == S_OK, "CoCreateInstance failed: %08x\n", hres);
|
|
|
|
return script;
|
|
}
|
|
|
|
static HRESULT parse_script(DWORD flags, BSTR script_str)
|
|
{
|
|
IActiveScriptParse *parser;
|
|
IActiveScript *engine;
|
|
IDispatch *script_disp;
|
|
LONG ref;
|
|
HRESULT hres;
|
|
|
|
engine = create_script();
|
|
if(!engine)
|
|
return S_OK;
|
|
|
|
hres = IActiveScript_QueryInterface(engine, &IID_IActiveScriptParse, (void**)&parser);
|
|
ok(hres == S_OK, "Could not get IActiveScriptParse: %08x\n", hres);
|
|
if (FAILED(hres))
|
|
{
|
|
IActiveScript_Release(engine);
|
|
return hres;
|
|
}
|
|
|
|
hres = IActiveScriptParse64_InitNew(parser);
|
|
ok(hres == S_OK, "InitNew failed: %08x\n", hres);
|
|
|
|
hres = IActiveScript_SetScriptSite(engine, &ActiveScriptSite);
|
|
ok(hres == S_OK, "SetScriptSite failed: %08x\n", hres);
|
|
|
|
hres = IActiveScript_AddNamedItem(engine, testW,
|
|
SCRIPTITEM_ISVISIBLE|SCRIPTITEM_ISSOURCE|flags);
|
|
ok(hres == S_OK, "AddNamedItem failed: %08x\n", hres);
|
|
|
|
hres = IActiveScript_SetScriptState(engine, SCRIPTSTATE_STARTED);
|
|
ok(hres == S_OK, "SetScriptState(SCRIPTSTATE_STARTED) failed: %08x\n", hres);
|
|
|
|
hres = IActiveScript_GetScriptDispatch(engine, NULL, &script_disp);
|
|
ok(hres == S_OK, "GetScriptDispatch failed: %08x\n", hres);
|
|
ok(script_disp != NULL, "script_disp == NULL\n");
|
|
ok(script_disp != (IDispatch*)&Global, "script_disp == Global\n");
|
|
|
|
test_counter = 0;
|
|
|
|
hres = IActiveScriptParse64_ParseScriptText(parser, script_str, NULL, NULL, NULL, 0, 0, 0, NULL, NULL);
|
|
|
|
IActiveScript_Close(engine);
|
|
|
|
IDispatch_Release(script_disp);
|
|
IActiveScript_Release(engine);
|
|
|
|
ref = IUnknown_Release(parser);
|
|
ok(!ref, "ref=%d\n", ref);
|
|
return hres;
|
|
}
|
|
|
|
static void parse_script_af(DWORD flags, const char *src)
|
|
{
|
|
BSTR tmp;
|
|
HRESULT hres;
|
|
|
|
tmp = a2bstr(src);
|
|
hres = parse_script(flags, tmp);
|
|
SysFreeString(tmp);
|
|
ok(hres == S_OK, "parse_script failed: %08x\n", hres);
|
|
}
|
|
|
|
static void parse_script_a(const char *src)
|
|
{
|
|
parse_script_af(SCRIPTITEM_GLOBALMEMBERS, src);
|
|
}
|
|
|
|
static void test_gc(void)
|
|
{
|
|
IActiveScriptParse *parser;
|
|
IActiveScript *engine;
|
|
BSTR src;
|
|
HRESULT hres;
|
|
|
|
strict_dispid_check = FALSE;
|
|
|
|
engine = create_script();
|
|
if(!engine)
|
|
return;
|
|
|
|
hres = IActiveScript_QueryInterface(engine, &IID_IActiveScriptParse, (void**)&parser);
|
|
ok(hres == S_OK, "Could not get IActiveScriptParse: %08x\n", hres);
|
|
|
|
hres = IActiveScriptParse64_InitNew(parser);
|
|
ok(hres == S_OK, "InitNew failed: %08x\n", hres);
|
|
|
|
hres = IActiveScript_SetScriptSite(engine, &ActiveScriptSite);
|
|
ok(hres == S_OK, "SetScriptSite failed: %08x\n", hres);
|
|
|
|
hres = IActiveScript_AddNamedItem(engine, testW,
|
|
SCRIPTITEM_ISVISIBLE|SCRIPTITEM_ISSOURCE|SCRIPTITEM_GLOBALMEMBERS);
|
|
ok(hres == S_OK, "AddNamedItem failed: %08x\n", hres);
|
|
|
|
hres = IActiveScript_SetScriptState(engine, SCRIPTSTATE_STARTED);
|
|
ok(hres == S_OK, "SetScriptState(SCRIPTSTATE_STARTED) failed: %08x\n", hres);
|
|
|
|
src = a2bstr(
|
|
"class C\n"
|
|
" Public ref\n"
|
|
" Public Sub Class_Terminate\n"
|
|
" Call reportSuccess()\n"
|
|
" End Sub\n"
|
|
"End Class\n"
|
|
"Dim x\n"
|
|
"set x = new C\n"
|
|
"set x.ref = x\n"
|
|
"set x = nothing\n");
|
|
|
|
hres = IActiveScriptParse64_ParseScriptText(parser, src, NULL, NULL, NULL, 0, 0, 0, NULL, NULL);
|
|
ok(hres == S_OK, "ParseScriptText failed: %08x\n", hres);
|
|
SysFreeString(src);
|
|
|
|
SET_EXPECT(global_success_d);
|
|
SET_EXPECT(global_success_i);
|
|
IActiveScript_Close(engine);
|
|
CHECK_CALLED(global_success_d);
|
|
CHECK_CALLED(global_success_i);
|
|
|
|
IActiveScript_Release(engine);
|
|
IUnknown_Release(parser);
|
|
}
|
|
|
|
static HRESULT test_global_vars_ref(BOOL use_close)
|
|
{
|
|
IActiveScriptParse *parser;
|
|
IActiveScript *engine;
|
|
BSTR script_str;
|
|
LONG ref;
|
|
HRESULT hres;
|
|
|
|
engine = create_script();
|
|
if(!engine)
|
|
return S_OK;
|
|
|
|
hres = IActiveScript_QueryInterface(engine, &IID_IActiveScriptParse, (void**)&parser);
|
|
ok(hres == S_OK, "Could not get IActiveScriptParse: %08x\n", hres);
|
|
if (FAILED(hres))
|
|
{
|
|
IActiveScript_Release(engine);
|
|
return hres;
|
|
}
|
|
|
|
hres = IActiveScriptParse64_InitNew(parser);
|
|
ok(hres == S_OK, "InitNew failed: %08x\n", hres);
|
|
|
|
hres = IActiveScript_SetScriptSite(engine, &ActiveScriptSite);
|
|
ok(hres == S_OK, "SetScriptSite failed: %08x\n", hres);
|
|
|
|
hres = IActiveScript_AddNamedItem(engine, testW, SCRIPTITEM_ISVISIBLE|SCRIPTITEM_ISSOURCE|SCRIPTITEM_GLOBALMEMBERS);
|
|
ok(hres == S_OK, "AddNamedItem failed: %08x\n", hres);
|
|
|
|
hres = IActiveScript_SetScriptState(engine, SCRIPTSTATE_STARTED);
|
|
ok(hres == S_OK, "SetScriptState(SCRIPTSTATE_STARTED) failed: %08x\n", hres);
|
|
|
|
refobj_ref = 0;
|
|
|
|
script_str = a2bstr("Dim x\nset x = RefObj\n");
|
|
hres = IActiveScriptParse64_ParseScriptText(parser, script_str, NULL, NULL, NULL, 0, 0, 0, NULL, NULL);
|
|
SysFreeString(script_str);
|
|
|
|
ok(refobj_ref, "refobj_ref = 0\n");
|
|
|
|
if(use_close) {
|
|
hres = IActiveScript_Close(engine);
|
|
ok(hres == S_OK, "Close failed: %08x\n", hres);
|
|
}else {
|
|
hres = IActiveScript_SetScriptState(engine, SCRIPTSTATE_UNINITIALIZED);
|
|
ok(hres == S_OK, "SetScriptState(SCRIPTSTATE_STARTED) failed: %08x\n", hres);
|
|
}
|
|
|
|
ok(!refobj_ref, "refobj_ref = %d\n", refobj_ref);
|
|
|
|
IActiveScript_Release(engine);
|
|
|
|
ref = IUnknown_Release(parser);
|
|
ok(!ref, "ref=%d\n", ref);
|
|
return hres;
|
|
}
|
|
|
|
static BSTR get_script_from_file(const char *filename)
|
|
{
|
|
DWORD size, len;
|
|
HANDLE file, map;
|
|
const char *file_map;
|
|
BSTR ret;
|
|
|
|
file = CreateFileA(filename, GENERIC_READ, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_READONLY, NULL);
|
|
if(file == INVALID_HANDLE_VALUE) {
|
|
trace("Could not open file: %u\n", GetLastError());
|
|
return NULL;
|
|
}
|
|
|
|
size = GetFileSize(file, NULL);
|
|
|
|
map = CreateFileMapping(file, NULL, PAGE_READONLY, 0, 0, NULL);
|
|
CloseHandle(file);
|
|
if(map == INVALID_HANDLE_VALUE) {
|
|
trace("Could not create file mapping: %u\n", GetLastError());
|
|
return NULL;
|
|
}
|
|
|
|
file_map = MapViewOfFile(map, FILE_MAP_READ, 0, 0, 0);
|
|
CloseHandle(map);
|
|
if(!file_map) {
|
|
trace("MapViewOfFile failed: %u\n", GetLastError());
|
|
return NULL;
|
|
}
|
|
|
|
len = MultiByteToWideChar(CP_ACP, 0, file_map, size, NULL, 0);
|
|
ret = SysAllocStringLen(NULL, len);
|
|
MultiByteToWideChar(CP_ACP, 0, file_map, size, ret, len);
|
|
|
|
UnmapViewOfFile(file_map);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static void run_from_file(const char *filename)
|
|
{
|
|
BSTR script_str;
|
|
HRESULT hres;
|
|
|
|
script_str = get_script_from_file(filename);
|
|
if(!script_str)
|
|
return;
|
|
|
|
strict_dispid_check = FALSE;
|
|
hres = parse_script(SCRIPTITEM_GLOBALMEMBERS, script_str);
|
|
SysFreeString(script_str);
|
|
ok(hres == S_OK, "parse_script failed: %08x\n", hres);
|
|
}
|
|
|
|
static void run_from_res(const char *name)
|
|
{
|
|
const char *data;
|
|
DWORD size, len;
|
|
BSTR str;
|
|
HRSRC src;
|
|
HRESULT hres;
|
|
|
|
strict_dispid_check = FALSE;
|
|
test_name = name;
|
|
|
|
src = FindResourceA(NULL, name, (LPCSTR)40);
|
|
ok(src != NULL, "Could not find resource %s\n", name);
|
|
|
|
size = SizeofResource(NULL, src);
|
|
data = LoadResource(NULL, src);
|
|
|
|
len = MultiByteToWideChar(CP_ACP, 0, data, size, NULL, 0);
|
|
str = SysAllocStringLen(NULL, len);
|
|
MultiByteToWideChar(CP_ACP, 0, data, size, str, len);
|
|
|
|
SET_EXPECT(global_success_d);
|
|
SET_EXPECT(global_success_i);
|
|
hres = parse_script(SCRIPTITEM_GLOBALMEMBERS, str);
|
|
CHECK_CALLED(global_success_d);
|
|
CHECK_CALLED(global_success_i);
|
|
|
|
ok(hres == S_OK, "parse_script failed: %08x\n", hres);
|
|
SysFreeString(str);
|
|
}
|
|
|
|
static void run_tests(void)
|
|
{
|
|
strict_dispid_check = TRUE;
|
|
|
|
parse_script_a("");
|
|
parse_script_a("' empty ;");
|
|
|
|
SET_EXPECT(global_success_d);
|
|
SET_EXPECT(global_success_i);
|
|
parse_script_a("reportSuccess");
|
|
CHECK_CALLED(global_success_d);
|
|
CHECK_CALLED(global_success_i);
|
|
|
|
SET_EXPECT(global_success_d);
|
|
SET_EXPECT(global_success_i);
|
|
parse_script_a("reportSuccess()");
|
|
CHECK_CALLED(global_success_d);
|
|
CHECK_CALLED(global_success_i);
|
|
|
|
SET_EXPECT(global_success_d);
|
|
SET_EXPECT(global_success_i);
|
|
parse_script_a("Call reportSuccess");
|
|
CHECK_CALLED(global_success_d);
|
|
CHECK_CALLED(global_success_i);
|
|
|
|
SET_EXPECT(global_success_d);
|
|
SET_EXPECT(global_success_i);
|
|
parse_script_a("test.reportSuccess()");
|
|
CHECK_CALLED(global_success_d);
|
|
CHECK_CALLED(global_success_i);
|
|
|
|
SET_EXPECT(global_vbvar_d);
|
|
SET_EXPECT(global_vbvar_i);
|
|
parse_script_a("Option Explicit\nvbvar = 3");
|
|
CHECK_CALLED(global_vbvar_d);
|
|
CHECK_CALLED(global_vbvar_i);
|
|
|
|
SET_EXPECT(global_vbvar_d);
|
|
SET_EXPECT(global_vbvar_i);
|
|
parse_script_a("Option Explicit\nvbvar() = 3");
|
|
CHECK_CALLED(global_vbvar_d);
|
|
CHECK_CALLED(global_vbvar_i);
|
|
|
|
SET_EXPECT(testobj_propget_d);
|
|
SET_EXPECT(testobj_propget_i);
|
|
parse_script_a("dim x\nx = testObj.propget");
|
|
CHECK_CALLED(testobj_propget_d);
|
|
CHECK_CALLED(testobj_propget_i);
|
|
|
|
SET_EXPECT(testobj_propput_d);
|
|
SET_EXPECT(testobj_propput_i);
|
|
parse_script_a("testObj.propput = 1");
|
|
CHECK_CALLED(testobj_propput_d);
|
|
CHECK_CALLED(testobj_propput_i);
|
|
|
|
SET_EXPECT(global_propargput_d);
|
|
SET_EXPECT(global_propargput_i);
|
|
parse_script_a("propargput(counter(), counter()) = counter()");
|
|
CHECK_CALLED(global_propargput_d);
|
|
CHECK_CALLED(global_propargput_i);
|
|
|
|
SET_EXPECT(global_propargput_d);
|
|
SET_EXPECT(global_propargput_i);
|
|
parse_script_a("test.propargput(counter(), counter()) = counter()");
|
|
CHECK_CALLED(global_propargput_d);
|
|
CHECK_CALLED(global_propargput_i);
|
|
|
|
parse_script_a("x = 1\n Call ok(x = 1, \"x = \" & x)");
|
|
|
|
parse_script_a("x = _ \n3");
|
|
|
|
test_global_vars_ref(TRUE);
|
|
test_global_vars_ref(FALSE);
|
|
|
|
strict_dispid_check = FALSE;
|
|
|
|
parse_script_a("Sub testsub\n"
|
|
"x = 1\n"
|
|
"Call ok(x = 1, \"x = \" & x)\n"
|
|
"End Sub\n"
|
|
"Call testsub()");
|
|
|
|
run_from_res("lang.vbs");
|
|
run_from_res("api.vbs");
|
|
|
|
test_gc();
|
|
}
|
|
|
|
static BOOL check_vbscript(void)
|
|
{
|
|
IActiveScript *vbscript;
|
|
HRESULT hres;
|
|
|
|
hres = CoCreateInstance(&CLSID_VBScript, NULL, CLSCTX_INPROC_SERVER|CLSCTX_INPROC_HANDLER,
|
|
&IID_IActiveScript, (void**)&vbscript);
|
|
if(SUCCEEDED(hres))
|
|
IActiveScript_Release(vbscript);
|
|
|
|
return hres == S_OK;
|
|
}
|
|
|
|
START_TEST(run)
|
|
{
|
|
int argc;
|
|
char **argv;
|
|
|
|
argc = winetest_get_mainargs(&argv);
|
|
|
|
CoInitialize(NULL);
|
|
|
|
if(!check_vbscript()) {
|
|
win_skip("Broken engine, probably too old\n");
|
|
}else if(argc > 2) {
|
|
run_from_file(argv[2]);
|
|
}else {
|
|
run_tests();
|
|
}
|
|
|
|
CoUninitialize();
|
|
}
|