2 * Copyright 2011 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
28 #include "wine/test.h"
30 extern const CLSID CLSID_VBScript;
32 #define DEFINE_EXPECT(func) \
33 static BOOL expect_ ## func = FALSE, called_ ## func = FALSE
35 #define SET_EXPECT(func) \
36 expect_ ## func = TRUE
38 #define SET_CALLED(func) \
39 called_ ## func = TRUE
41 #define CHECK_EXPECT2(func) \
43 ok(expect_ ##func, "unexpected call " #func "\n"); \
44 called_ ## func = TRUE; \
47 #define CHECK_EXPECT(func) \
49 CHECK_EXPECT2(func); \
50 expect_ ## func = FALSE; \
53 #define CHECK_CALLED(func) \
55 ok(called_ ## func, "expected " #func "\n"); \
56 expect_ ## func = called_ ## func = FALSE; \
59 DEFINE_EXPECT(global_success_d);
60 DEFINE_EXPECT(global_success_i);
61 DEFINE_EXPECT(global_vbvar_d);
62 DEFINE_EXPECT(global_vbvar_i);
63 DEFINE_EXPECT(testobj_propget_d);
64 DEFINE_EXPECT(testobj_propget_i);
65 DEFINE_EXPECT(testobj_propput_d);
66 DEFINE_EXPECT(testobj_propput_i);
68 #define DISPID_GLOBAL_REPORTSUCCESS 1000
69 #define DISPID_GLOBAL_TRACE 1001
70 #define DISPID_GLOBAL_OK 1002
71 #define DISPID_GLOBAL_GETVT 1003
72 #define DISPID_GLOBAL_ISENGLANG 1004
73 #define DISPID_GLOBAL_VBVAR 1005
74 #define DISPID_GLOBAL_TESTOBJ 1006
75 #define DISPID_GLOBAL_ISNULLDISP 1007
76 #define DISPID_GLOBAL_TESTDISP 1008
78 #define DISPID_TESTOBJ_PROPGET 2000
79 #define DISPID_TESTOBJ_PROPPUT 2001
81 static const WCHAR testW[] = {'t','e','s','t',0};
83 static BOOL strict_dispid_check;
84 static const char *test_name = "(null)";
86 static BSTR a2bstr(const char *str)
91 len = MultiByteToWideChar(CP_ACP, 0, str, -1, NULL, 0);
92 ret = SysAllocStringLen(NULL, len-1);
93 MultiByteToWideChar(CP_ACP, 0, str, -1, ret, len);
98 static int strcmp_wa(LPCWSTR strw, const char *stra)
101 WideCharToMultiByte(CP_ACP, 0, strw, -1, buf, sizeof(buf), 0, 0);
102 return lstrcmpA(buf, stra);
105 static const char *vt2a(VARIANT *v)
107 if(V_VT(v) == (VT_BYREF|VT_VARIANT)) {
109 sprintf(buf, "%s*", vt2a(V_BYREF(v)));
127 return "VT_DISPATCH";
130 case VT_ARRAY|VT_VARIANT:
131 return "VT_ARRAY|VT_VARIANT";
133 ok(0, "unknown vt %d\n", V_VT(v));
138 /* Returns true if the user interface is in English. Note that this does not
139 * presume of the formatting of dates, numbers, etc.
141 static BOOL is_lang_english(void)
143 static HMODULE hkernel32 = NULL;
144 static LANGID (WINAPI *pGetThreadUILanguage)(void) = NULL;
145 static LANGID (WINAPI *pGetUserDefaultUILanguage)(void) = NULL;
149 hkernel32 = GetModuleHandleA("kernel32.dll");
150 pGetThreadUILanguage = (void*)GetProcAddress(hkernel32, "GetThreadUILanguage");
151 pGetUserDefaultUILanguage = (void*)GetProcAddress(hkernel32, "GetUserDefaultUILanguage");
153 if (pGetThreadUILanguage)
154 return PRIMARYLANGID(pGetThreadUILanguage()) == LANG_ENGLISH;
155 if (pGetUserDefaultUILanguage)
156 return PRIMARYLANGID(pGetUserDefaultUILanguage()) == LANG_ENGLISH;
158 return PRIMARYLANGID(GetUserDefaultLangID()) == LANG_ENGLISH;
161 static void test_disp(IDispatch *disp)
163 DISPID id, public_prop_id, public_prop2_id, public_func_id, public_sub_id;
164 DISPID named_args[5] = {DISPID_PROPERTYPUT};
166 DISPPARAMS dp = {args, named_args};
172 hres = IDispatch_QueryInterface(disp, &IID_IDispatchEx, (void**)&dispex);
173 ok(hres == S_OK, "Could not get IDispatchEx iface: %08x\n", hres);
175 str = a2bstr("publicProp");
176 hres = IDispatchEx_GetDispID(dispex, str, fdexNameCaseInsensitive, &public_prop_id);
178 ok(hres == S_OK, "GetDispID(publicProp) failed: %08x\n", hres);
180 str = a2bstr("PUBLICPROP");
181 hres = IDispatchEx_GetDispID(dispex, str, 0, &id);
183 ok(hres == S_OK, "GetDispID(PUBLICPROP) failed: %08x\n", hres);
184 ok(public_prop_id == id, "id = %d\n", public_prop_id);
186 str = a2bstr("publicPROP2");
187 hres = IDispatchEx_GetDispID(dispex, str, fdexNameCaseInsensitive, &public_prop2_id);
189 ok(hres == S_OK, "GetDispID(publicProp2) failed: %08x\n", hres);
191 hres = IDispatchEx_InvokeEx(dispex, public_prop_id, 0, DISPATCH_PROPERTYGET|DISPATCH_METHOD, &dp, &v, &ei, NULL);
192 ok(hres == S_OK, "InvokeEx failed: %08x\n", hres);
193 ok(V_VT(&v) == VT_EMPTY, "V_VT(v) = %d\n", V_VT(&v));
195 V_VT(args) = VT_BOOL;
196 V_BOOL(args) = VARIANT_TRUE;
197 dp.cArgs = dp.cNamedArgs = 1;
199 hres = IDispatchEx_InvokeEx(dispex, public_prop_id, 0, DISPATCH_PROPERTYPUT, &dp, &v, &ei, NULL);
200 ok(hres == S_OK, "InvokeEx failed: %08x\n", hres);
201 ok(V_VT(&v) == VT_EMPTY, "V_VT(v) = %d\n", V_VT(&v));
203 dp.cArgs = dp.cNamedArgs = 0;
204 hres = IDispatchEx_InvokeEx(dispex, public_prop_id, 0, DISPATCH_PROPERTYGET|DISPATCH_METHOD, &dp, &v, &ei, NULL);
205 ok(hres == S_OK, "InvokeEx failed: %08x\n", hres);
206 ok(V_VT(&v) == VT_BOOL, "V_VT(v) = %d\n", V_VT(&v));
207 ok(V_BOOL(&v), "V_BOOL(v) = %x\n", V_BOOL(&v));
210 hres = IDispatchEx_InvokeEx(dispex, public_prop_id, 0, DISPATCH_PROPERTYGET|DISPATCH_METHOD, &dp, &v, &ei, NULL);
211 ok(hres == DISP_E_MEMBERNOTFOUND, "InvokeEx failed: %08x, expected DISP_E_MEMBERNOTFOUND\n", hres);
212 ok(V_VT(&v) == VT_EMPTY, "V_VT(v) = %d\n", V_VT(&v));
214 V_VT(args) = VT_BOOL;
215 V_BOOL(args) = VARIANT_FALSE;
218 hres = IDispatchEx_InvokeEx(dispex, public_prop_id, 0, DISPATCH_PROPERTYPUT, &dp, NULL, &ei, NULL);
219 ok(hres == DISP_E_PARAMNOTOPTIONAL, "InvokeEx failed: %08x, expected DISP_E_PARAMNOTOPTIONAL\n", hres);
221 str = a2bstr("publicFunction");
222 hres = IDispatchEx_GetDispID(dispex, str, fdexNameCaseInsensitive, &public_func_id);
224 ok(hres == S_OK, "GetDispID(publicFunction) failed: %08x\n", hres);
225 ok(public_func_id != -1, "public_func_id = -1\n");
227 str = a2bstr("publicSub");
228 hres = IDispatchEx_GetDispID(dispex, str, fdexNameCaseInsensitive, &public_sub_id);
230 ok(hres == S_OK, "GetDispID(publicSub) failed: %08x\n", hres);
231 ok(public_sub_id != -1, "public_func_id = -1\n");
233 dp.cArgs = dp.cNamedArgs = 0;
234 hres = IDispatchEx_InvokeEx(dispex, public_func_id, 0, DISPATCH_PROPERTYGET|DISPATCH_METHOD, &dp, &v, &ei, NULL);
235 ok(hres == S_OK, "InvokeEx failed: %08x\n", hres);
236 ok(V_VT(&v) == VT_I2, "V_VT(v) = %d\n", V_VT(&v));
237 ok(V_I2(&v) == 4, "V_I2(v) = %d\n", V_I2(&v));
239 dp.cArgs = dp.cNamedArgs = 0;
240 hres = IDispatchEx_InvokeEx(dispex, public_func_id, 0, DISPATCH_METHOD, &dp, &v, &ei, NULL);
241 ok(hres == S_OK, "InvokeEx failed: %08x\n", hres);
242 ok(V_VT(&v) == VT_I2, "V_VT(v) = %d\n", V_VT(&v));
243 ok(V_I2(&v) == 4, "V_I2(v) = %d\n", V_I2(&v));
245 dp.cArgs = dp.cNamedArgs = 0;
246 hres = IDispatchEx_InvokeEx(dispex, public_sub_id, 0, DISPATCH_PROPERTYGET|DISPATCH_METHOD, &dp, &v, &ei, NULL);
247 ok(hres == S_OK, "InvokeEx failed: %08x\n", hres);
248 ok(V_VT(&v) == VT_EMPTY, "V_VT(v) = %d\n", V_VT(&v));
250 dp.cArgs = dp.cNamedArgs = 0;
251 hres = IDispatchEx_InvokeEx(dispex, public_sub_id, 0, DISPATCH_METHOD, &dp, &v, &ei, NULL);
252 ok(hres == S_OK, "InvokeEx failed: %08x\n", hres);
253 ok(V_VT(&v) == VT_EMPTY, "V_VT(v) = %d\n", V_VT(&v));
255 str = a2bstr("privateSub");
256 hres = IDispatchEx_GetDispID(dispex, str, fdexNameCaseInsensitive, &id);
258 ok(hres == DISP_E_UNKNOWNNAME, "GetDispID(privateSub) failed: %08x, expected DISP_E_UNKNOWNNAME\n", hres);
259 ok(id == -1, "id = %d\n", id);
261 str = a2bstr("dynprop");
262 hres = IDispatchEx_GetDispID(dispex, str, fdexNameCaseInsensitive|fdexNameEnsure, &id);
263 ok(hres == DISP_E_UNKNOWNNAME, "GetDispID(privateProp) failed: %08x, expected DISP_E_UNKNOWNNAME\n", hres);
264 ok(id == -1, "id = %d\n", id);
265 hres = IDispatchEx_GetDispID(dispex, str, fdexNameEnsure, &id);
266 ok(hres == DISP_E_UNKNOWNNAME, "GetDispID(privateProp) failed: %08x, expected DISP_E_UNKNOWNNAME\n", hres);
267 ok(id == -1, "id = %d\n", id);
270 IDispatchEx_Release(dispex);
273 #define test_grfdex(a,b) _test_grfdex(__LINE__,a,b)
274 static void _test_grfdex(unsigned line, DWORD grfdex, DWORD expect)
276 ok_(__FILE__,line)(grfdex == expect, "grfdex = %x, expected %x\n", grfdex, expect);
279 static HRESULT WINAPI DispatchEx_QueryInterface(IDispatchEx *iface, REFIID riid, void **ppv)
283 if(IsEqualGUID(riid, &IID_IUnknown)
284 || IsEqualGUID(riid, &IID_IDispatch)
285 || IsEqualGUID(riid, &IID_IDispatchEx))
288 return E_NOINTERFACE;
293 static ULONG WINAPI DispatchEx_AddRef(IDispatchEx *iface)
298 static ULONG WINAPI DispatchEx_Release(IDispatchEx *iface)
303 static HRESULT WINAPI DispatchEx_GetTypeInfoCount(IDispatchEx *iface, UINT *pctinfo)
305 ok(0, "unexpected call\n");
309 static HRESULT WINAPI DispatchEx_GetTypeInfo(IDispatchEx *iface, UINT iTInfo,
310 LCID lcid, ITypeInfo **ppTInfo)
312 ok(0, "unexpected call\n");
316 static HRESULT WINAPI DispatchEx_GetIDsOfNames(IDispatchEx *iface, REFIID riid,
317 LPOLESTR *rgszNames, UINT cNames,
318 LCID lcid, DISPID *rgDispId)
320 ok(0, "unexpected call\n");
324 static HRESULT WINAPI DispatchEx_Invoke(IDispatchEx *iface, DISPID dispIdMember,
325 REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS *pDispParams,
326 VARIANT *pVarResult, EXCEPINFO *pExcepInfo, UINT *puArgErr)
328 ok(0, "unexpected call\n");
332 static HRESULT WINAPI DispatchEx_DeleteMemberByName(IDispatchEx *iface, BSTR bstrName, DWORD grfdex)
334 ok(0, "unexpected call %s %x\n", wine_dbgstr_w(bstrName), grfdex);
338 static HRESULT WINAPI DispatchEx_DeleteMemberByDispID(IDispatchEx *iface, DISPID id)
340 ok(0, "unexpected call\n");
344 static HRESULT WINAPI DispatchEx_GetMemberProperties(IDispatchEx *iface, DISPID id, DWORD grfdexFetch, DWORD *pgrfdex)
346 ok(0, "unexpected call\n");
350 static HRESULT WINAPI DispatchEx_GetMemberName(IDispatchEx *iface, DISPID id, BSTR *pbstrName)
352 ok(0, "unexpected call\n");
356 static HRESULT WINAPI DispatchEx_GetNextDispID(IDispatchEx *iface, DWORD grfdex, DISPID id, DISPID *pid)
358 ok(0, "unexpected call\n");
362 static HRESULT WINAPI DispatchEx_GetNameSpaceParent(IDispatchEx *iface, IUnknown **ppunk)
364 ok(0, "unexpected call\n");
368 static HRESULT WINAPI testObj_GetDispID(IDispatchEx *iface, BSTR bstrName, DWORD grfdex, DISPID *pid)
370 if(!strcmp_wa(bstrName, "propget")) {
371 CHECK_EXPECT(testobj_propget_d);
372 test_grfdex(grfdex, fdexNameCaseInsensitive);
373 *pid = DISPID_TESTOBJ_PROPGET;
376 if(!strcmp_wa(bstrName, "propput")) {
377 CHECK_EXPECT(testobj_propput_d);
378 test_grfdex(grfdex, fdexNameCaseInsensitive);
379 *pid = DISPID_TESTOBJ_PROPPUT;
383 ok(0, "unexpected call %s\n", wine_dbgstr_w(bstrName));
384 return DISP_E_UNKNOWNNAME;
387 static HRESULT WINAPI testObj_InvokeEx(IDispatchEx *iface, DISPID id, LCID lcid, WORD wFlags, DISPPARAMS *pdp,
388 VARIANT *pvarRes, EXCEPINFO *pei, IServiceProvider *pspCaller)
391 case DISPID_TESTOBJ_PROPGET:
392 CHECK_EXPECT(testobj_propget_i);
394 ok(wFlags == (DISPATCH_PROPERTYGET|DISPATCH_METHOD), "wFlags = %x\n", wFlags);
395 ok(pdp != NULL, "pdp == NULL\n");
396 ok(!pdp->rgvarg, "rgvarg == NULL\n");
397 ok(!pdp->rgdispidNamedArgs, "rgdispidNamedArgs != NULL\n");
398 ok(!pdp->cArgs, "cArgs = %d\n", pdp->cArgs);
399 ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
400 ok(pvarRes != NULL, "pvarRes == NULL\n");
401 ok(pei != NULL, "pei == NULL\n");
403 V_VT(pvarRes) = VT_I2;
406 case DISPID_TESTOBJ_PROPPUT:
407 CHECK_EXPECT(testobj_propput_i);
409 ok(wFlags == DISPATCH_PROPERTYPUT, "wFlags = %x\n", wFlags);
410 ok(pdp != NULL, "pdp == NULL\n");
411 ok(pdp->rgvarg != NULL, "rgvarg == NULL\n");
412 ok(pdp->rgdispidNamedArgs != NULL, "rgdispidNamedArgs == NULL\n");
413 ok(pdp->cArgs == 1, "cArgs = %d\n", pdp->cArgs);
414 ok(pdp->cNamedArgs == 1, "cNamedArgs = %d\n", pdp->cNamedArgs);
415 ok(pdp->rgdispidNamedArgs[0] == DISPID_PROPERTYPUT, "pdp->rgdispidNamedArgs[0] = %d\n", pdp->rgdispidNamedArgs[0]);
416 ok(!pvarRes, "pvarRes != NULL\n");
417 ok(pei != NULL, "pei == NULL\n");
419 ok(V_VT(pdp->rgvarg) == VT_I2, "V_VT(psp->rgvargs) = %d\n", V_VT(pdp->rgvarg));
420 ok(V_I2(pdp->rgvarg) == 1, "V_I2(psp->rgvargs) = %d\n", V_I2(pdp->rgvarg));
424 ok(0, "unexpected call %d\n", id);
428 static IDispatchExVtbl testObjVtbl = {
429 DispatchEx_QueryInterface,
432 DispatchEx_GetTypeInfoCount,
433 DispatchEx_GetTypeInfo,
434 DispatchEx_GetIDsOfNames,
438 DispatchEx_DeleteMemberByName,
439 DispatchEx_DeleteMemberByDispID,
440 DispatchEx_GetMemberProperties,
441 DispatchEx_GetMemberName,
442 DispatchEx_GetNextDispID,
443 DispatchEx_GetNameSpaceParent
446 static IDispatchEx testObj = { &testObjVtbl };
448 static HRESULT WINAPI Global_GetDispID(IDispatchEx *iface, BSTR bstrName, DWORD grfdex, DISPID *pid)
450 if(!strcmp_wa(bstrName, "ok")) {
451 test_grfdex(grfdex, fdexNameCaseInsensitive);
452 *pid = DISPID_GLOBAL_OK;
455 if(!strcmp_wa(bstrName, "trace")) {
456 test_grfdex(grfdex, fdexNameCaseInsensitive);
457 *pid = DISPID_GLOBAL_TRACE;
460 if(!strcmp_wa(bstrName, "reportSuccess")) {
461 CHECK_EXPECT(global_success_d);
462 test_grfdex(grfdex, fdexNameCaseInsensitive);
463 *pid = DISPID_GLOBAL_REPORTSUCCESS;
466 if(!strcmp_wa(bstrName, "getVT")) {
467 test_grfdex(grfdex, fdexNameCaseInsensitive);
468 *pid = DISPID_GLOBAL_GETVT;
471 if(!strcmp_wa(bstrName, "isEnglishLang")) {
472 test_grfdex(grfdex, fdexNameCaseInsensitive);
473 *pid = DISPID_GLOBAL_ISENGLANG;
476 if(!strcmp_wa(bstrName, "testObj")) {
477 test_grfdex(grfdex, fdexNameCaseInsensitive);
478 *pid = DISPID_GLOBAL_TESTOBJ;
481 if(!strcmp_wa(bstrName, "vbvar")) {
482 CHECK_EXPECT(global_vbvar_d);
483 test_grfdex(grfdex, fdexNameCaseInsensitive);
484 *pid = DISPID_GLOBAL_VBVAR;
487 if(!strcmp_wa(bstrName, "isNullDisp")) {
488 test_grfdex(grfdex, fdexNameCaseInsensitive);
489 *pid = DISPID_GLOBAL_ISNULLDISP;
492 if(!strcmp_wa(bstrName, "testDisp")) {
493 test_grfdex(grfdex, fdexNameCaseInsensitive);
494 *pid = DISPID_GLOBAL_TESTDISP;
498 if(strict_dispid_check && strcmp_wa(bstrName, "x"))
499 ok(0, "unexpected call %s %x\n", wine_dbgstr_w(bstrName), grfdex);
500 return DISP_E_UNKNOWNNAME;
503 static HRESULT WINAPI Global_InvokeEx(IDispatchEx *iface, DISPID id, LCID lcid, WORD wFlags, DISPPARAMS *pdp,
504 VARIANT *pvarRes, EXCEPINFO *pei, IServiceProvider *pspCaller)
507 case DISPID_GLOBAL_OK: {
510 ok(wFlags == INVOKE_FUNC || wFlags == (INVOKE_FUNC|INVOKE_PROPERTYGET), "wFlags = %x\n", wFlags);
511 ok(pdp != NULL, "pdp == NULL\n");
512 ok(pdp->rgvarg != NULL, "rgvarg == NULL\n");
513 ok(!pdp->rgdispidNamedArgs, "rgdispidNamedArgs != NULL\n");
514 ok(pdp->cArgs == 2, "cArgs = %d\n", pdp->cArgs);
515 ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
516 if(wFlags & INVOKE_PROPERTYGET)
517 ok(pvarRes != NULL, "pvarRes == NULL\n");
519 ok(!pvarRes, "pvarRes != NULL\n");
520 ok(pei != NULL, "pei == NULL\n");
522 ok(V_VT(pdp->rgvarg) == VT_BSTR, "V_VT(psp->rgvargs) = %d\n", V_VT(pdp->rgvarg));
525 if(V_VT(b) == (VT_BYREF|VT_VARIANT))
528 ok(V_VT(b) == VT_BOOL, "V_VT(b) = %d\n", V_VT(b));
530 ok(V_BOOL(b), "%s: %s\n", test_name, wine_dbgstr_w(V_BSTR(pdp->rgvarg)));
534 case DISPID_GLOBAL_TRACE:
535 ok(wFlags == INVOKE_FUNC, "wFlags = %x\n", wFlags);
536 ok(pdp != NULL, "pdp == NULL\n");
537 ok(pdp->rgvarg != NULL, "rgvarg == NULL\n");
538 ok(!pdp->rgdispidNamedArgs, "rgdispidNamedArgs != NULL\n");
539 ok(pdp->cArgs == 1, "cArgs = %d\n", pdp->cArgs);
540 ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
541 ok(!pvarRes, "pvarRes != NULL\n");
542 ok(pei != NULL, "pei == NULL\n");
544 ok(V_VT(pdp->rgvarg) == VT_BSTR, "V_VT(psp->rgvargs) = %d\n", V_VT(pdp->rgvarg));
545 if(V_VT(pdp->rgvarg) == VT_BSTR)
546 trace("%s: %s\n", test_name, wine_dbgstr_w(V_BSTR(pdp->rgvarg)));
550 case DISPID_GLOBAL_REPORTSUCCESS:
551 CHECK_EXPECT(global_success_i);
553 ok(wFlags == INVOKE_FUNC, "wFlags = %x\n", wFlags);
554 ok(pdp != NULL, "pdp == NULL\n");
555 ok(!pdp->rgdispidNamedArgs, "rgdispidNamedArgs != NULL\n");
556 ok(pdp->cArgs == 0, "cArgs = %d\n", pdp->cArgs);
557 ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
558 ok(!pvarRes, "pvarRes != NULL\n");
559 ok(pei != NULL, "pei == NULL\n");
563 case DISPID_GLOBAL_GETVT:
564 ok(pdp != NULL, "pdp == NULL\n");
565 ok(pdp->rgvarg != NULL, "rgvarg == NULL\n");
566 ok(!pdp->rgdispidNamedArgs, "rgdispidNamedArgs != NULL\n");
567 ok(pdp->cArgs == 1, "cArgs = %d\n", pdp->cArgs);
568 ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
569 ok(pvarRes != NULL, "pvarRes == NULL\n");
570 ok(V_VT(pvarRes) == VT_EMPTY, "V_VT(pvarRes) = %d\n", V_VT(pvarRes));
571 ok(pei != NULL, "pei == NULL\n");
573 V_VT(pvarRes) = VT_BSTR;
574 V_BSTR(pvarRes) = a2bstr(vt2a(pdp->rgvarg));
577 case DISPID_GLOBAL_ISENGLANG:
578 ok(wFlags == (INVOKE_FUNC|INVOKE_PROPERTYGET), "wFlags = %x\n", wFlags);
579 ok(pdp != NULL, "pdp == NULL\n");
580 ok(!pdp->rgdispidNamedArgs, "rgdispidNamedArgs != NULL\n");
581 ok(pdp->cArgs == 0, "cArgs = %d\n", pdp->cArgs);
582 ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
583 ok(pvarRes != NULL, "pvarRes == NULL\n");
584 ok(pei != NULL, "pei == NULL\n");
586 V_VT(pvarRes) = VT_BOOL;
587 if(is_lang_english()) {
588 V_BOOL(pvarRes) = VARIANT_TRUE;
590 skip("Skipping some tests in non-English UIs\n");
591 V_BOOL(pvarRes) = VARIANT_FALSE;
595 case DISPID_GLOBAL_VBVAR:
596 CHECK_EXPECT(global_vbvar_i);
598 ok(wFlags == DISPATCH_PROPERTYPUT, "wFlags = %x\n", wFlags);
599 ok(pdp != NULL, "pdp == NULL\n");
600 ok(pdp->rgvarg != NULL, "rgvarg == NULL\n");
601 ok(pdp->rgdispidNamedArgs != NULL, "rgdispidNamedArgs == NULL\n");
602 ok(pdp->cArgs == 1, "cArgs = %d\n", pdp->cArgs);
603 ok(pdp->cNamedArgs == 1, "cNamedArgs = %d\n", pdp->cNamedArgs);
604 ok(pdp->rgdispidNamedArgs[0] == DISPID_PROPERTYPUT, "pdp->rgdispidNamedArgs[0] = %d\n", pdp->rgdispidNamedArgs[0]);
605 ok(!pvarRes, "pvarRes != NULL\n");
606 ok(pei != NULL, "pei == NULL\n");
608 ok(V_VT(pdp->rgvarg) == VT_I2, "V_VT(psp->rgvargs) = %d\n", V_VT(pdp->rgvarg));
609 ok(V_I2(pdp->rgvarg) == 3, "V_I2(psp->rgvargs) = %d\n", V_I2(pdp->rgvarg));
612 case DISPID_GLOBAL_TESTOBJ:
613 ok(wFlags == (DISPATCH_PROPERTYGET|DISPATCH_METHOD), "wFlags = %x\n", wFlags);
615 ok(pdp != NULL, "pdp == NULL\n");
616 ok(!pdp->rgvarg, "rgvarg == NULL\n");
617 ok(!pdp->rgdispidNamedArgs, "rgdispidNamedArgs != NULL\n");
618 ok(!pdp->cArgs, "cArgs = %d\n", pdp->cArgs);
619 ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
620 ok(pvarRes != NULL, "pvarRes == NULL\n");
621 ok(pei != NULL, "pei == NULL\n");
623 V_VT(pvarRes) = VT_DISPATCH;
624 V_DISPATCH(pvarRes) = (IDispatch*)&testObj;
627 case DISPID_GLOBAL_ISNULLDISP: {
630 ok(wFlags == (INVOKE_FUNC|INVOKE_PROPERTYGET), "wFlags = %x\n", wFlags);
631 ok(pdp != NULL, "pdp == NULL\n");
632 ok(pdp->rgvarg != NULL, "rgvarg == NULL\n");
633 ok(!pdp->rgdispidNamedArgs, "rgdispidNamedArgs != NULL\n");
634 ok(pdp->cArgs == 1, "cArgs = %d\n", pdp->cArgs);
635 ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
636 ok(pvarRes != NULL, "pvarRes == NULL\n");
637 ok(pei != NULL, "pei == NULL\n");
640 if(V_VT(v) == (VT_VARIANT|VT_BYREF))
643 ok(V_VT(v) == VT_DISPATCH, "V_VT(psp->rgvargs) = %d\n", V_VT(pdp->rgvarg));
644 V_VT(pvarRes) = VT_BOOL;
645 V_BOOL(pvarRes) = V_DISPATCH(v) ? VARIANT_FALSE : VARIANT_TRUE;
649 case DISPID_GLOBAL_TESTDISP:
650 ok(wFlags == INVOKE_FUNC, "wFlags = %x\n", wFlags);
651 ok(pdp != NULL, "pdp == NULL\n");
652 ok(pdp->rgvarg != NULL, "rgvarg == NULL\n");
653 ok(!pdp->rgdispidNamedArgs, "rgdispidNamedArgs != NULL\n");
654 ok(pdp->cArgs == 1, "cArgs = %d\n", pdp->cArgs);
655 ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
656 ok(!pvarRes, "pvarRes != NULL\n");
657 ok(pei != NULL, "pei == NULL\n");
659 ok(V_VT(pdp->rgvarg) == VT_DISPATCH, "V_VT(psp->rgvargs) = %d\n", V_VT(pdp->rgvarg));
660 test_disp(V_DISPATCH(pdp->rgvarg));
664 ok(0, "unexpected call %d\n", id);
665 return DISP_E_MEMBERNOTFOUND;
668 static IDispatchExVtbl GlobalVtbl = {
669 DispatchEx_QueryInterface,
672 DispatchEx_GetTypeInfoCount,
673 DispatchEx_GetTypeInfo,
674 DispatchEx_GetIDsOfNames,
678 DispatchEx_DeleteMemberByName,
679 DispatchEx_DeleteMemberByDispID,
680 DispatchEx_GetMemberProperties,
681 DispatchEx_GetMemberName,
682 DispatchEx_GetNextDispID,
683 DispatchEx_GetNameSpaceParent
686 static IDispatchEx Global = { &GlobalVtbl };
688 static HRESULT WINAPI ActiveScriptSite_QueryInterface(IActiveScriptSite *iface, REFIID riid, void **ppv)
692 if(IsEqualGUID(&IID_IUnknown, riid))
694 else if(IsEqualGUID(&IID_IActiveScriptSite, riid))
697 return E_NOINTERFACE;
699 IUnknown_AddRef((IUnknown*)*ppv);
703 static ULONG WINAPI ActiveScriptSite_AddRef(IActiveScriptSite *iface)
708 static ULONG WINAPI ActiveScriptSite_Release(IActiveScriptSite *iface)
713 static HRESULT WINAPI ActiveScriptSite_GetLCID(IActiveScriptSite *iface, LCID *plcid)
715 *plcid = GetUserDefaultLCID();
719 static HRESULT WINAPI ActiveScriptSite_GetItemInfo(IActiveScriptSite *iface, LPCOLESTR pstrName,
720 DWORD dwReturnMask, IUnknown **ppiunkItem, ITypeInfo **ppti)
722 ok(dwReturnMask == SCRIPTINFO_IUNKNOWN, "unexpected dwReturnMask %x\n", dwReturnMask);
723 ok(!ppti, "ppti != NULL\n");
725 if(strcmp_wa(pstrName, "test"))
726 ok(0, "unexpected pstrName %s\n", wine_dbgstr_w(pstrName));
728 *ppiunkItem = (IUnknown*)&Global;
732 static HRESULT WINAPI ActiveScriptSite_GetDocVersionString(IActiveScriptSite *iface, BSTR *pbstrVersion)
737 static HRESULT WINAPI ActiveScriptSite_OnScriptTerminate(IActiveScriptSite *iface,
738 const VARIANT *pvarResult, const EXCEPINFO *pexcepinfo)
743 static HRESULT WINAPI ActiveScriptSite_OnStateChange(IActiveScriptSite *iface, SCRIPTSTATE ssScriptState)
748 static HRESULT WINAPI ActiveScriptSite_OnScriptError(IActiveScriptSite *iface, IActiveScriptError *pscripterror)
753 static HRESULT WINAPI ActiveScriptSite_OnEnterScript(IActiveScriptSite *iface)
758 static HRESULT WINAPI ActiveScriptSite_OnLeaveScript(IActiveScriptSite *iface)
763 #undef ACTSCPSITE_THIS
765 static const IActiveScriptSiteVtbl ActiveScriptSiteVtbl = {
766 ActiveScriptSite_QueryInterface,
767 ActiveScriptSite_AddRef,
768 ActiveScriptSite_Release,
769 ActiveScriptSite_GetLCID,
770 ActiveScriptSite_GetItemInfo,
771 ActiveScriptSite_GetDocVersionString,
772 ActiveScriptSite_OnScriptTerminate,
773 ActiveScriptSite_OnStateChange,
774 ActiveScriptSite_OnScriptError,
775 ActiveScriptSite_OnEnterScript,
776 ActiveScriptSite_OnLeaveScript
779 static IActiveScriptSite ActiveScriptSite = { &ActiveScriptSiteVtbl };
781 static IActiveScript *create_script(void)
783 IActiveScript *script;
786 hres = CoCreateInstance(&CLSID_VBScript, NULL, CLSCTX_INPROC_SERVER|CLSCTX_INPROC_HANDLER,
787 &IID_IActiveScript, (void**)&script);
788 ok(hres == S_OK, "CoCreateInstance failed: %08x\n", hres);
793 static HRESULT parse_script(DWORD flags, BSTR script_str)
795 IActiveScriptParse *parser;
796 IActiveScript *engine;
797 IDispatch *script_disp;
800 engine = create_script();
804 hres = IActiveScript_QueryInterface(engine, &IID_IActiveScriptParse, (void**)&parser);
805 ok(hres == S_OK, "Could not get IActiveScriptParse: %08x\n", hres);
808 IActiveScript_Release(engine);
812 hres = IActiveScriptParse64_InitNew(parser);
813 ok(hres == S_OK, "InitNew failed: %08x\n", hres);
815 hres = IActiveScript_SetScriptSite(engine, &ActiveScriptSite);
816 ok(hres == S_OK, "SetScriptSite failed: %08x\n", hres);
818 hres = IActiveScript_AddNamedItem(engine, testW,
819 SCRIPTITEM_ISVISIBLE|SCRIPTITEM_ISSOURCE|flags);
820 ok(hres == S_OK, "AddNamedItem failed: %08x\n", hres);
822 hres = IActiveScript_SetScriptState(engine, SCRIPTSTATE_STARTED);
823 ok(hres == S_OK, "SetScriptState(SCRIPTSTATE_STARTED) failed: %08x\n", hres);
825 hres = IActiveScript_GetScriptDispatch(engine, NULL, &script_disp);
826 ok(hres == S_OK, "GetScriptDispatch failed: %08x\n", hres);
827 ok(script_disp != NULL, "script_disp == NULL\n");
828 ok(script_disp != (IDispatch*)&Global, "script_disp == Global\n");
830 hres = IActiveScriptParse64_ParseScriptText(parser, script_str, NULL, NULL, NULL, 0, 0, 0, NULL, NULL);
832 IDispatch_Release(script_disp);
833 IActiveScript_Release(engine);
834 IUnknown_Release(parser);
839 static void parse_script_af(DWORD flags, const char *src)
845 hres = parse_script(flags, tmp);
847 ok(hres == S_OK, "parse_script failed: %08x\n", hres);
850 static void parse_script_a(const char *src)
852 parse_script_af(SCRIPTITEM_GLOBALMEMBERS, src);
855 static BSTR get_script_from_file(const char *filename)
859 const char *file_map;
862 file = CreateFileA(filename, GENERIC_READ, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_READONLY, NULL);
863 if(file == INVALID_HANDLE_VALUE) {
864 trace("Could not open file: %u\n", GetLastError());
868 size = GetFileSize(file, NULL);
870 map = CreateFileMapping(file, NULL, PAGE_READONLY, 0, 0, NULL);
872 if(map == INVALID_HANDLE_VALUE) {
873 trace("Could not create file mapping: %u\n", GetLastError());
877 file_map = MapViewOfFile(map, FILE_MAP_READ, 0, 0, 0);
880 trace("MapViewOfFile failed: %u\n", GetLastError());
884 len = MultiByteToWideChar(CP_ACP, 0, file_map, size, NULL, 0);
885 ret = SysAllocStringLen(NULL, len);
886 MultiByteToWideChar(CP_ACP, 0, file_map, size, ret, len);
888 UnmapViewOfFile(file_map);
893 static void run_from_file(const char *filename)
898 script_str = get_script_from_file(filename);
902 strict_dispid_check = FALSE;
903 hres = parse_script(SCRIPTITEM_GLOBALMEMBERS, script_str);
904 SysFreeString(script_str);
905 ok(hres == S_OK, "parse_script failed: %08x\n", hres);
908 static void run_from_res(const char *name)
916 strict_dispid_check = FALSE;
919 src = FindResourceA(NULL, name, (LPCSTR)40);
920 ok(src != NULL, "Could not find resource %s\n", name);
922 size = SizeofResource(NULL, src);
923 data = LoadResource(NULL, src);
925 len = MultiByteToWideChar(CP_ACP, 0, data, size, NULL, 0);
926 str = SysAllocStringLen(NULL, len);
927 MultiByteToWideChar(CP_ACP, 0, data, size, str, len);
929 SET_EXPECT(global_success_d);
930 SET_EXPECT(global_success_i);
931 hres = parse_script(SCRIPTITEM_GLOBALMEMBERS, str);
932 CHECK_CALLED(global_success_d);
933 CHECK_CALLED(global_success_i);
935 ok(hres == S_OK, "parse_script failed: %08x\n", hres);
939 static void run_tests(void)
941 strict_dispid_check = TRUE;
944 parse_script_a("' empty ;");
946 SET_EXPECT(global_success_d);
947 SET_EXPECT(global_success_i);
948 parse_script_a("reportSuccess");
949 CHECK_CALLED(global_success_d);
950 CHECK_CALLED(global_success_i);
952 SET_EXPECT(global_success_d);
953 SET_EXPECT(global_success_i);
954 parse_script_a("reportSuccess()");
955 CHECK_CALLED(global_success_d);
956 CHECK_CALLED(global_success_i);
958 SET_EXPECT(global_success_d);
959 SET_EXPECT(global_success_i);
960 parse_script_a("Call reportSuccess");
961 CHECK_CALLED(global_success_d);
962 CHECK_CALLED(global_success_i);
964 SET_EXPECT(global_vbvar_d);
965 SET_EXPECT(global_vbvar_i);
966 parse_script_a("Option Explicit\nvbvar = 3");
967 CHECK_CALLED(global_vbvar_d);
968 CHECK_CALLED(global_vbvar_i);
970 SET_EXPECT(global_vbvar_d);
971 SET_EXPECT(global_vbvar_i);
972 parse_script_a("Option Explicit\nvbvar() = 3");
973 CHECK_CALLED(global_vbvar_d);
974 CHECK_CALLED(global_vbvar_i);
976 SET_EXPECT(testobj_propget_d);
977 SET_EXPECT(testobj_propget_i);
978 parse_script_a("dim x\nx = testObj.propget");
979 CHECK_CALLED(testobj_propget_d);
980 CHECK_CALLED(testobj_propget_i);
982 SET_EXPECT(testobj_propput_d);
983 SET_EXPECT(testobj_propput_i);
984 parse_script_a("testObj.propput = 1");
985 CHECK_CALLED(testobj_propput_d);
986 CHECK_CALLED(testobj_propput_i);
988 run_from_res("lang.vbs");
991 static BOOL check_vbscript(void)
993 IActiveScript *vbscript;
996 hres = CoCreateInstance(&CLSID_VBScript, NULL, CLSCTX_INPROC_SERVER|CLSCTX_INPROC_HANDLER,
997 &IID_IActiveScript, (void**)&vbscript);
999 IActiveScript_Release(vbscript);
1001 return hres == S_OK;
1009 argc = winetest_get_mainargs(&argv);
1013 if(!check_vbscript()) {
1014 win_skip("Broken engine, probably too old\n");
1015 }else if(argc > 2) {
1016 run_from_file(argv[2]);