dbghelp: Remove an "#if 1" preprocessor directive.
[wine/multimedia.git] / dlls / vbscript / global.c
blob9d8f0210957b3b805a3ef517166d96dff5a46dd6
1 /*
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
19 #include <assert.h>
20 #include <math.h>
22 #include "vbscript.h"
23 #include "vbscript_defs.h"
25 #include "mshtmhst.h"
26 #include "objsafe.h"
28 #include "wine/debug.h"
30 WINE_DEFAULT_DEBUG_CHANNEL(vbscript);
32 #define VB_E_CANNOT_CREATE_OBJ 0x800a01ad
33 #define VB_E_MK_PARSE_ERROR 0x800a01b0
35 /* Defined as extern in urlmon.idl, but not exported by uuid.lib */
36 const GUID GUID_CUSTOM_CONFIRMOBJECTSAFETY =
37 {0x10200490,0xfa38,0x11d0,{0xac,0x0e,0x00,0xa0,0xc9,0xf,0xff,0xc0}};
39 static const WCHAR emptyW[] = {0};
40 static const WCHAR vbscriptW[] = {'V','B','S','c','r','i','p','t',0};
42 static IInternetHostSecurityManager *get_sec_mgr(script_ctx_t *ctx)
44 IInternetHostSecurityManager *secmgr;
45 IServiceProvider *sp;
46 HRESULT hres;
48 if(!ctx->site)
49 return NULL;
51 if(ctx->secmgr)
52 return ctx->secmgr;
54 hres = IActiveScriptSite_QueryInterface(ctx->site, &IID_IServiceProvider, (void**)&sp);
55 if(FAILED(hres))
56 return NULL;
58 hres = IServiceProvider_QueryService(sp, &SID_SInternetHostSecurityManager, &IID_IInternetHostSecurityManager,
59 (void**)&secmgr);
60 IServiceProvider_Release(sp);
61 if(FAILED(hres))
62 return NULL;
64 return ctx->secmgr = secmgr;
67 static HRESULT return_string(VARIANT *res, const WCHAR *str)
69 BSTR ret;
71 if(!res)
72 return S_OK;
74 ret = SysAllocString(str);
75 if(!ret)
76 return E_OUTOFMEMORY;
78 V_VT(res) = VT_BSTR;
79 V_BSTR(res) = ret;
80 return S_OK;
83 static HRESULT return_bstr(VARIANT *res, BSTR str)
85 if(res) {
86 V_VT(res) = VT_BSTR;
87 V_BSTR(res) = str;
88 }else {
89 SysFreeString(str);
91 return S_OK;
94 static HRESULT return_short(VARIANT *res, short val)
96 if(res) {
97 V_VT(res) = VT_I2;
98 V_I2(res) = val;
101 return S_OK;
104 static HRESULT return_int(VARIANT *res, int val)
106 if((short)val == val)
107 return return_short(res, val);
109 if(res) {
110 V_VT(res) = VT_I4;
111 V_I4(res) = val;
114 return S_OK;
117 static HRESULT return_bool(VARIANT *res, int val)
119 if(res) {
120 V_VT(res) = VT_BOOL;
121 V_BOOL(res) = val != 0 ? VARIANT_TRUE : VARIANT_FALSE;
124 return S_OK;
127 static inline HRESULT return_double(VARIANT *res, double val)
129 if(res) {
130 V_VT(res) = VT_R8;
131 V_R8(res) = val;
134 return S_OK;
137 static inline HRESULT return_null(VARIANT *res)
139 if(res)
140 V_VT(res) = VT_NULL;
141 return S_OK;
144 static inline HRESULT return_date(VARIANT *res, double date)
146 if(res) {
147 V_VT(res) = VT_DATE;
148 V_DATE(res) = date;
150 return S_OK;
153 HRESULT to_int(VARIANT *v, int *ret)
155 if(V_VT(v) == (VT_BYREF|VT_VARIANT))
156 v = V_VARIANTREF(v);
158 switch(V_VT(v)) {
159 case VT_I2:
160 *ret = V_I2(v);
161 break;
162 case VT_I4:
163 *ret = V_I4(v);
164 break;
165 case VT_R8: {
166 double n = floor(V_R8(v)+0.5);
167 INT32 i;
169 if(!is_int32(n)) {
170 FIXME("%lf is out of int range\n", n);
171 return E_FAIL;
174 /* Round half to even */
175 i = n;
176 if(i%2 && n-V_R8(v) == 0.5)
177 i--;
179 *ret = i;
180 break;
182 case VT_BOOL:
183 *ret = V_BOOL(v) ? -1 : 0;
184 break;
185 default:
186 FIXME("not supported %s\n", debugstr_variant(v));
187 return E_NOTIMPL;
190 return S_OK;
193 static HRESULT to_double(VARIANT *v, double *ret)
195 switch(V_VT(v)) {
196 case VT_I2:
197 *ret = V_I2(v);
198 break;
199 case VT_I4:
200 *ret = V_I4(v);
201 break;
202 case VT_R4:
203 *ret = V_R4(v);
204 break;
205 case VT_R8:
206 *ret = V_R8(v);
207 break;
208 case VT_BSTR: {
209 VARIANT dst;
210 HRESULT hres;
212 V_VT(&dst) = VT_EMPTY;
213 hres = VariantChangeType(&dst, v, VARIANT_LOCALBOOL, VT_R8);
214 if(FAILED(hres))
215 return hres;
216 *ret = V_R8(&dst);
217 break;
219 default:
220 FIXME("arg %s not supported\n", debugstr_variant(v));
221 return E_NOTIMPL;
224 return S_OK;
227 static HRESULT to_string(VARIANT *v, BSTR *ret)
229 VARIANT dst;
230 HRESULT hres;
232 V_VT(&dst) = VT_EMPTY;
233 hres = VariantChangeType(&dst, v, VARIANT_LOCALBOOL, VT_BSTR);
234 if(FAILED(hres))
235 return hres;
237 *ret = V_BSTR(&dst);
238 return S_OK;
241 static HRESULT set_object_site(script_ctx_t *ctx, IUnknown *obj)
243 IObjectWithSite *obj_site;
244 IUnknown *ax_site;
245 HRESULT hres;
247 hres = IUnknown_QueryInterface(obj, &IID_IObjectWithSite, (void**)&obj_site);
248 if(FAILED(hres))
249 return S_OK;
251 ax_site = create_ax_site(ctx);
252 if(ax_site)
253 hres = IObjectWithSite_SetSite(obj_site, ax_site);
254 else
255 hres = E_OUTOFMEMORY;
256 IUnknown_Release(ax_site);
257 IObjectWithSite_Release(obj_site);
258 return hres;
261 static IUnknown *create_object(script_ctx_t *ctx, const WCHAR *progid)
263 IInternetHostSecurityManager *secmgr = NULL;
264 struct CONFIRMSAFETY cs;
265 IClassFactoryEx *cfex;
266 IClassFactory *cf;
267 DWORD policy_size;
268 BYTE *bpolicy;
269 IUnknown *obj;
270 DWORD policy;
271 GUID guid;
272 HRESULT hres;
274 hres = CLSIDFromProgID(progid, &guid);
275 if(FAILED(hres))
276 return NULL;
278 TRACE("GUID %s\n", debugstr_guid(&guid));
280 if(ctx->safeopt & INTERFACE_USES_SECURITY_MANAGER) {
281 secmgr = get_sec_mgr(ctx);
282 if(!secmgr)
283 return NULL;
285 policy = 0;
286 hres = IInternetHostSecurityManager_ProcessUrlAction(secmgr, URLACTION_ACTIVEX_RUN,
287 (BYTE*)&policy, sizeof(policy), (BYTE*)&guid, sizeof(GUID), 0, 0);
288 if(FAILED(hres) || policy != URLPOLICY_ALLOW)
289 return NULL;
292 hres = CoGetClassObject(&guid, CLSCTX_INPROC_SERVER|CLSCTX_LOCAL_SERVER, NULL, &IID_IClassFactory, (void**)&cf);
293 if(FAILED(hres))
294 return NULL;
296 hres = IClassFactory_QueryInterface(cf, &IID_IClassFactoryEx, (void**)&cfex);
297 if(SUCCEEDED(hres)) {
298 FIXME("Use IClassFactoryEx\n");
299 IClassFactoryEx_Release(cfex);
302 hres = IClassFactory_CreateInstance(cf, NULL, &IID_IUnknown, (void**)&obj);
303 if(FAILED(hres))
304 return NULL;
306 if(secmgr) {
307 cs.clsid = guid;
308 cs.pUnk = obj;
309 cs.dwFlags = 0;
310 hres = IInternetHostSecurityManager_QueryCustomPolicy(secmgr, &GUID_CUSTOM_CONFIRMOBJECTSAFETY,
311 &bpolicy, &policy_size, (BYTE*)&cs, sizeof(cs), 0);
312 if(SUCCEEDED(hres)) {
313 policy = policy_size >= sizeof(DWORD) ? *(DWORD*)bpolicy : URLPOLICY_DISALLOW;
314 CoTaskMemFree(bpolicy);
317 if(FAILED(hres) || policy != URLPOLICY_ALLOW) {
318 IUnknown_Release(obj);
319 return NULL;
323 hres = set_object_site(ctx, obj);
324 if(FAILED(hres)) {
325 IUnknown_Release(obj);
326 return NULL;
329 return obj;
332 static HRESULT show_msgbox(script_ctx_t *ctx, BSTR prompt, VARIANT *res)
334 SCRIPTUICHANDLING uic_handling = SCRIPTUICHANDLING_ALLOW;
335 IActiveScriptSiteUIControl *ui_control;
336 IActiveScriptSiteWindow *acts_window;
337 const WCHAR *title;
338 HWND hwnd = NULL;
339 int ret;
340 HRESULT hres;
342 hres = IActiveScriptSite_QueryInterface(ctx->site, &IID_IActiveScriptSiteUIControl, (void**)&ui_control);
343 if(SUCCEEDED(hres)) {
344 hres = IActiveScriptSiteUIControl_GetUIBehavior(ui_control, SCRIPTUICITEM_MSGBOX, &uic_handling);
345 IActiveScriptSiteUIControl_Release(ui_control);
346 if(FAILED(hres))
347 uic_handling = SCRIPTUICHANDLING_ALLOW;
350 switch(uic_handling) {
351 case SCRIPTUICHANDLING_ALLOW:
352 break;
353 case SCRIPTUICHANDLING_NOUIDEFAULT:
354 return return_short(res, 0);
355 default:
356 FIXME("blocked\n");
357 return E_FAIL;
360 title = (ctx->safeopt & INTERFACE_USES_SECURITY_MANAGER) ? vbscriptW : emptyW;
362 hres = IActiveScriptSite_QueryInterface(ctx->site, &IID_IActiveScriptSiteWindow, (void**)&acts_window);
363 if(FAILED(hres)) {
364 FIXME("No IActiveScriptSiteWindow\n");
365 return hres;
368 hres = IActiveScriptSiteWindow_GetWindow(acts_window, &hwnd);
369 if(SUCCEEDED(hres)) {
370 hres = IActiveScriptSiteWindow_EnableModeless(acts_window, FALSE);
371 if(SUCCEEDED(hres)) {
372 ret = MessageBoxW(hwnd, prompt, title, MB_OK);
373 hres = IActiveScriptSiteWindow_EnableModeless(acts_window, TRUE);
377 IActiveScriptSiteWindow_Release(acts_window);
378 if(FAILED(hres)) {
379 FIXME("failed: %08x\n", hres);
380 return hres;
383 return return_short(res, ret);
386 static HRESULT Global_CCur(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
388 FIXME("\n");
389 return E_NOTIMPL;
392 static HRESULT Global_CInt(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
394 int val;
395 HRESULT hres;
397 TRACE("%s\n", debugstr_variant(arg));
399 assert(args_cnt == 1);
401 hres = to_int(arg, &val);
402 if(FAILED(hres))
403 return hres;
405 return return_int(res, val);
408 static HRESULT Global_CLng(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
410 FIXME("\n");
411 return E_NOTIMPL;
414 static HRESULT Global_CBool(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
416 int val;
417 TRACE("%s\n", debugstr_variant(arg));
419 assert(args_cnt == 1);
421 switch(V_VT(arg)) {
422 case VT_I2:
423 val = V_I2(arg);
424 break;
425 case VT_I4:
426 val = V_I4(arg);
427 break;
428 case VT_R4:
429 val = V_R4(arg) > 0.0 || V_R4(arg) < 0.0;
430 break;
431 case VT_R8:
432 val = V_R8(arg) > 0.0 || V_R8(arg) < 0.0;
433 break;
434 default:
435 ERR("Not a numeric value: %s\n", debugstr_variant(arg));
436 return E_FAIL;
439 return return_bool(res, val);
442 static HRESULT Global_CByte(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
444 FIXME("\n");
445 return E_NOTIMPL;
448 static HRESULT Global_CDate(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
450 FIXME("\n");
451 return E_NOTIMPL;
454 static HRESULT Global_CDbl(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
456 FIXME("\n");
457 return E_NOTIMPL;
460 static HRESULT Global_CSng(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
462 FIXME("\n");
463 return E_NOTIMPL;
466 static HRESULT Global_CStr(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
468 BSTR str;
469 HRESULT hres;
471 TRACE("%s\n", debugstr_variant(arg));
473 hres = to_string(arg, &str);
474 if(FAILED(hres))
475 return hres;
477 return return_bstr(res, str);
480 static inline WCHAR hex_char(unsigned n)
482 return n < 10 ? '0'+n : 'A'+n-10;
485 static HRESULT Global_Hex(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
487 WCHAR buf[17], *ptr;
488 DWORD n;
490 TRACE("%s\n", debugstr_variant(arg));
492 switch(V_VT(arg)) {
493 case VT_I2:
494 n = (WORD)V_I2(arg);
495 break;
496 case VT_I4:
497 n = V_I4(arg);
498 break;
499 case VT_EMPTY:
500 n = 0;
501 break;
502 case VT_NULL:
503 if(res)
504 V_VT(res) = VT_NULL;
505 return S_OK;
506 default:
507 FIXME("unsupported type %s\n", debugstr_variant(arg));
508 return E_NOTIMPL;
511 buf[16] = 0;
512 ptr = buf+15;
514 if(n) {
515 do {
516 *ptr-- = hex_char(n & 0xf);
517 n >>= 4;
518 }while(n);
519 ptr++;
520 }else {
521 *ptr = '0';
524 return return_string(res, ptr);
527 static HRESULT Global_Oct(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
529 FIXME("\n");
530 return E_NOTIMPL;
533 static HRESULT Global_VarType(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
535 FIXME("\n");
536 return E_NOTIMPL;
539 static HRESULT Global_IsDate(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
541 FIXME("\n");
542 return E_NOTIMPL;
545 static HRESULT Global_IsEmpty(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
547 TRACE("(%s)\n", debugstr_variant(arg));
549 assert(args_cnt == 1);
551 if(res) {
552 V_VT(res) = VT_BOOL;
553 V_BOOL(res) = V_VT(arg) == VT_EMPTY ? VARIANT_TRUE : VARIANT_FALSE;
555 return S_OK;
558 static HRESULT Global_IsNull(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
560 TRACE("(%s)\n", debugstr_variant(arg));
562 assert(args_cnt == 1);
564 if(res) {
565 V_VT(res) = VT_BOOL;
566 V_BOOL(res) = V_VT(arg) == VT_NULL ? VARIANT_TRUE : VARIANT_FALSE;
568 return S_OK;
571 static HRESULT Global_IsNumeric(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
573 FIXME("\n");
574 return E_NOTIMPL;
577 static HRESULT Global_IsArray(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
579 FIXME("\n");
580 return E_NOTIMPL;
583 static HRESULT Global_IsObject(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
585 TRACE("(%s)\n", debugstr_variant(arg));
587 assert(args_cnt == 1);
589 if(res) {
590 V_VT(res) = VT_BOOL;
591 V_BOOL(res) = V_VT(arg) == VT_DISPATCH ? VARIANT_TRUE : VARIANT_FALSE;
593 return S_OK;
596 static HRESULT Global_Ant(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
598 FIXME("\n");
599 return E_NOTIMPL;
602 static HRESULT Global_Cos(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
604 FIXME("\n");
605 return E_NOTIMPL;
608 static HRESULT Global_Sin(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
610 FIXME("\n");
611 return E_NOTIMPL;
614 static HRESULT Global_Tan(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
616 FIXME("\n");
617 return E_NOTIMPL;
620 static HRESULT Global_Exp(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
622 FIXME("\n");
623 return E_NOTIMPL;
626 static HRESULT Global_Log(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
628 FIXME("\n");
629 return E_NOTIMPL;
632 static HRESULT Global_Sqr(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
634 FIXME("\n");
635 return E_NOTIMPL;
638 static HRESULT Global_Randomize(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
640 FIXME("\n");
641 return E_NOTIMPL;
644 static HRESULT Global_Rnd(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
646 FIXME("\n");
647 return E_NOTIMPL;
650 static HRESULT Global_Timer(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
652 FIXME("\n");
653 return E_NOTIMPL;
656 static HRESULT Global_LBound(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
658 FIXME("\n");
659 return E_NOTIMPL;
662 static HRESULT Global_UBound(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
664 FIXME("\n");
665 return E_NOTIMPL;
668 static HRESULT Global_RGB(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
670 FIXME("\n");
671 return E_NOTIMPL;
674 static HRESULT Global_Len(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
676 DWORD len;
677 HRESULT hres;
679 TRACE("%s\n", debugstr_variant(arg));
681 if(V_VT(arg) == VT_NULL)
682 return return_null(res);
684 if(V_VT(arg) != VT_BSTR) {
685 BSTR str;
687 hres = to_string(arg, &str);
688 if(FAILED(hres))
689 return hres;
691 len = SysStringLen(str);
692 SysFreeString(str);
693 }else {
694 len = SysStringLen(V_BSTR(arg));
697 return return_int(res, len);
700 static HRESULT Global_LenB(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
702 FIXME("\n");
703 return E_NOTIMPL;
706 static HRESULT Global_Left(vbdisp_t *This, VARIANT *args, unsigned args_cnt, VARIANT *res)
708 BSTR str, ret, conv_str = NULL;
709 int len, str_len;
710 HRESULT hres;
712 TRACE("(%s %s)\n", debugstr_variant(args+1), debugstr_variant(args));
714 if(V_VT(args) == VT_BSTR) {
715 str = V_BSTR(args);
716 }else {
717 hres = to_string(args, &conv_str);
718 if(FAILED(hres))
719 return hres;
720 str = conv_str;
723 hres = to_int(args+1, &len);
724 if(FAILED(hres))
725 return hres;
727 if(len < 0) {
728 FIXME("len = %d\n", len);
729 return E_FAIL;
732 str_len = SysStringLen(str);
733 if(len > str_len)
734 len = str_len;
736 ret = SysAllocStringLen(str, len);
737 SysFreeString(conv_str);
738 if(!ret)
739 return E_OUTOFMEMORY;
741 return return_bstr(res, ret);
744 static HRESULT Global_LeftB(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
746 FIXME("\n");
747 return E_NOTIMPL;
750 static HRESULT Global_Right(vbdisp_t *This, VARIANT *args, unsigned args_cnt, VARIANT *res)
752 BSTR str, ret, conv_str = NULL;
753 int len, str_len;
754 HRESULT hres;
756 TRACE("(%s %s)\n", debugstr_variant(args), debugstr_variant(args+1));
758 if(V_VT(args+1) == VT_BSTR) {
759 str = V_BSTR(args);
760 }else {
761 hres = to_string(args, &conv_str);
762 if(FAILED(hres))
763 return hres;
764 str = conv_str;
767 hres = to_int(args+1, &len);
768 if(FAILED(hres))
769 return hres;
771 if(len < 0) {
772 FIXME("len = %d\n", len);
773 return E_FAIL;
776 str_len = SysStringLen(str);
777 if(len > str_len)
778 len = str_len;
780 ret = SysAllocStringLen(str+str_len-len, len);
781 SysFreeString(conv_str);
782 if(!ret)
783 return E_OUTOFMEMORY;
785 return return_bstr(res, ret);
788 static HRESULT Global_RightB(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
790 FIXME("\n");
791 return E_NOTIMPL;
794 static HRESULT Global_Mid(vbdisp_t *This, VARIANT *args, unsigned args_cnt, VARIANT *res)
796 int len = -1, start, str_len;
797 BSTR str;
798 HRESULT hres;
800 TRACE("(%s %s ...)\n", debugstr_variant(args), debugstr_variant(args+1));
802 assert(args_cnt == 2 || args_cnt == 3);
804 if(V_VT(args) != VT_BSTR) {
805 FIXME("args[0] = %s\n", debugstr_variant(args));
806 return E_NOTIMPL;
809 str = V_BSTR(args);
811 hres = to_int(args+1, &start);
812 if(FAILED(hres))
813 return hres;
815 if(args_cnt == 3) {
816 hres = to_int(args+2, &len);
817 if(FAILED(hres))
818 return hres;
820 if(len < 0) {
821 FIXME("len = %d\n", len);
822 return E_FAIL;
827 str_len = SysStringLen(str);
828 start--;
829 if(start > str_len)
830 start = str_len;
832 if(len == -1)
833 len = str_len-start;
834 else if(len > str_len-start)
835 len = str_len-start;
837 if(res) {
838 V_VT(res) = VT_BSTR;
839 V_BSTR(res) = SysAllocStringLen(str+start, len);
840 if(!V_BSTR(res))
841 return E_OUTOFMEMORY;
844 return S_OK;
847 static HRESULT Global_MidB(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
849 FIXME("\n");
850 return E_NOTIMPL;
853 static HRESULT Global_StrComp(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
855 FIXME("\n");
856 return E_NOTIMPL;
859 static HRESULT Global_LCase(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
861 BSTR str;
862 HRESULT hres;
864 TRACE("%s\n", debugstr_variant(arg));
866 if(V_VT(arg) == VT_NULL) {
867 if(res)
868 V_VT(res) = VT_NULL;
869 return S_OK;
872 hres = to_string(arg, &str);
873 if(FAILED(hres))
874 return hres;
876 if(res) {
877 WCHAR *ptr;
879 for(ptr = str; *ptr; ptr++)
880 *ptr = tolowerW(*ptr);
882 V_VT(res) = VT_BSTR;
883 V_BSTR(res) = str;
884 }else {
885 SysFreeString(str);
887 return S_OK;
890 static HRESULT Global_UCase(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
892 BSTR str;
893 HRESULT hres;
895 TRACE("%s\n", debugstr_variant(arg));
897 if(V_VT(arg) == VT_NULL) {
898 if(res)
899 V_VT(res) = VT_NULL;
900 return S_OK;
903 hres = to_string(arg, &str);
904 if(FAILED(hres))
905 return hres;
907 if(res) {
908 WCHAR *ptr;
910 for(ptr = str; *ptr; ptr++)
911 *ptr = toupperW(*ptr);
913 V_VT(res) = VT_BSTR;
914 V_BSTR(res) = str;
915 }else {
916 SysFreeString(str);
918 return S_OK;
921 static HRESULT Global_LTrim(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
923 BSTR str, conv_str = NULL;
924 WCHAR *ptr;
925 HRESULT hres;
927 TRACE("%s\n", debugstr_variant(arg));
929 if(V_VT(arg) == VT_BSTR) {
930 str = V_BSTR(arg);
931 }else {
932 hres = to_string(arg, &conv_str);
933 if(FAILED(hres))
934 return hres;
935 str = conv_str;
938 for(ptr = str; *ptr && isspaceW(*ptr); ptr++);
940 str = SysAllocString(ptr);
941 SysFreeString(conv_str);
942 if(!str)
943 return E_OUTOFMEMORY;
945 return return_bstr(res, str);
948 static HRESULT Global_RTrim(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
950 BSTR str, conv_str = NULL;
951 WCHAR *ptr;
952 HRESULT hres;
954 TRACE("%s\n", debugstr_variant(arg));
956 if(V_VT(arg) == VT_BSTR) {
957 str = V_BSTR(arg);
958 }else {
959 hres = to_string(arg, &conv_str);
960 if(FAILED(hres))
961 return hres;
962 str = conv_str;
965 for(ptr = str+SysStringLen(str); ptr-1 > str && isspaceW(*(ptr-1)); ptr--);
967 str = SysAllocStringLen(str, ptr-str);
968 SysFreeString(conv_str);
969 if(!str)
970 return E_OUTOFMEMORY;
972 return return_bstr(res, str);
975 static HRESULT Global_Trim(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
977 BSTR str, conv_str = NULL;
978 WCHAR *begin_ptr, *end_ptr;
979 HRESULT hres;
981 TRACE("%s\n", debugstr_variant(arg));
983 if(V_VT(arg) == VT_BSTR) {
984 str = V_BSTR(arg);
985 }else {
986 hres = to_string(arg, &conv_str);
987 if(FAILED(hres))
988 return hres;
989 str = conv_str;
992 for(begin_ptr = str; *begin_ptr && isspaceW(*begin_ptr); begin_ptr++);
993 for(end_ptr = str+SysStringLen(str); end_ptr-1 > begin_ptr && isspaceW(*(end_ptr-1)); end_ptr--);
995 str = SysAllocStringLen(begin_ptr, end_ptr-begin_ptr);
996 SysFreeString(conv_str);
997 if(!str)
998 return E_OUTOFMEMORY;
1000 return return_bstr(res, str);
1003 static HRESULT Global_Space(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1005 BSTR str;
1006 int n, i;
1007 HRESULT hres;
1009 TRACE("%s\n", debugstr_variant(arg));
1011 hres = to_int(arg, &n);
1012 if(FAILED(hres))
1013 return hres;
1015 if(n < 0) {
1016 FIXME("n = %d\n", n);
1017 return E_NOTIMPL;
1020 if(!res)
1021 return S_OK;
1023 str = SysAllocStringLen(NULL, n);
1024 if(!str)
1025 return E_OUTOFMEMORY;
1027 for(i=0; i<n; i++)
1028 str[i] = ' ';
1030 V_VT(res) = VT_BSTR;
1031 V_BSTR(res) = str;
1032 return S_OK;
1035 static HRESULT Global_String(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1037 FIXME("\n");
1038 return E_NOTIMPL;
1041 static HRESULT Global_InStr(vbdisp_t *This, VARIANT *args, unsigned args_cnt, VARIANT *res)
1043 VARIANT *startv, *str1v, *str2v;
1044 BSTR str1, str2;
1045 int start, ret;
1046 HRESULT hres;
1048 TRACE("\n");
1050 assert(2 <= args_cnt && args_cnt <= 4);
1052 switch(args_cnt) {
1053 case 2:
1054 startv = NULL;
1055 str1v = args;
1056 str2v = args+1;
1057 break;
1058 case 3:
1059 startv = args;
1060 str1v = args+1;
1061 str2v = args+2;
1062 break;
1063 case 4:
1064 FIXME("unsupported compare argument %s\n", debugstr_variant(args));
1065 return E_NOTIMPL;
1066 DEFAULT_UNREACHABLE;
1069 if(startv) {
1070 hres = to_int(startv, &start);
1071 if(FAILED(hres))
1072 return hres;
1073 if(--start < 0) {
1074 FIXME("start %d\n", start);
1075 return E_FAIL;
1077 }else {
1078 start = 0;
1081 if(V_VT(str1v) == VT_NULL || V_VT(str2v) == VT_NULL)
1082 return return_null(res);
1084 if(V_VT(str1v) != VT_BSTR) {
1085 FIXME("Unsupported str1 type %s\n", debugstr_variant(str1v));
1086 return E_NOTIMPL;
1088 str1 = V_BSTR(str1v);
1090 if(V_VT(str2v) != VT_BSTR) {
1091 FIXME("Unsupported str2 type %s\n", debugstr_variant(str2v));
1092 return E_NOTIMPL;
1094 str2 = V_BSTR(str2v);
1096 if(start < SysStringLen(str1)) {
1097 WCHAR *ptr;
1099 ptr = strstrW(str1+start, str2);
1100 ret = ptr ? ptr-str1+1 : 0;
1101 }else {
1102 ret = 0;
1105 return return_int(res, ret);
1108 static HRESULT Global_InStrB(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1110 FIXME("\n");
1111 return E_NOTIMPL;
1114 static HRESULT Global_AscB(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1116 FIXME("\n");
1117 return E_NOTIMPL;
1120 static HRESULT Global_ChrB(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1122 FIXME("\n");
1123 return E_NOTIMPL;
1126 static HRESULT Global_Asc(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1128 FIXME("\n");
1129 return E_NOTIMPL;
1132 static HRESULT Global_Chr(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1134 int c;
1135 HRESULT hres;
1137 TRACE("%s\n", debugstr_variant(arg));
1139 hres = to_int(arg, &c);
1140 if(FAILED(hres))
1141 return hres;
1143 if(c < 0 || c >= 0x100) {
1144 FIXME("invalid arg\n");
1145 return E_FAIL;
1148 if(res) {
1149 WCHAR ch = c;
1151 V_VT(res) = VT_BSTR;
1152 V_BSTR(res) = SysAllocStringLen(&ch, 1);
1153 if(!V_BSTR(res))
1154 return E_OUTOFMEMORY;
1156 return S_OK;
1159 static HRESULT Global_AscW(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1161 FIXME("\n");
1162 return E_NOTIMPL;
1165 static HRESULT Global_ChrW(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1167 FIXME("\n");
1168 return E_NOTIMPL;
1171 static HRESULT Global_Abs(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1173 FIXME("\n");
1174 return E_NOTIMPL;
1177 static HRESULT Global_Fix(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1179 FIXME("\n");
1180 return E_NOTIMPL;
1183 static HRESULT Global_Int(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1185 FIXME("\n");
1186 return E_NOTIMPL;
1189 static HRESULT Global_Sgn(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1191 FIXME("\n");
1192 return E_NOTIMPL;
1195 static HRESULT Global_Now(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1197 SYSTEMTIME lt;
1198 double date;
1200 TRACE("\n");
1202 GetLocalTime(&lt);
1203 SystemTimeToVariantTime(&lt, &date);
1204 return return_date(res, date);
1207 static HRESULT Global_Date(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1209 FIXME("\n");
1210 return E_NOTIMPL;
1213 static HRESULT Global_Time(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1215 FIXME("\n");
1216 return E_NOTIMPL;
1219 static HRESULT Global_Day(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1221 FIXME("\n");
1222 return E_NOTIMPL;
1225 static HRESULT Global_Month(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1227 FIXME("\n");
1228 return E_NOTIMPL;
1231 static HRESULT Global_Weekday(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1233 FIXME("\n");
1234 return E_NOTIMPL;
1237 static HRESULT Global_Year(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1239 FIXME("\n");
1240 return E_NOTIMPL;
1243 static HRESULT Global_Hour(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1245 FIXME("\n");
1246 return E_NOTIMPL;
1249 static HRESULT Global_Minute(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1251 FIXME("\n");
1252 return E_NOTIMPL;
1255 static HRESULT Global_Second(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1257 FIXME("\n");
1258 return E_NOTIMPL;
1261 static HRESULT Global_DateValue(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1263 FIXME("\n");
1264 return E_NOTIMPL;
1267 static HRESULT Global_TimeValue(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1269 FIXME("\n");
1270 return E_NOTIMPL;
1273 static HRESULT Global_DateSerial(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1275 FIXME("\n");
1276 return E_NOTIMPL;
1279 static HRESULT Global_TimeSerial(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1281 FIXME("\n");
1282 return E_NOTIMPL;
1285 static HRESULT Global_InputBox(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1287 FIXME("\n");
1288 return E_NOTIMPL;
1291 static HRESULT Global_MsgBox(vbdisp_t *This, VARIANT *args, unsigned args_cnt, VARIANT *res)
1293 BSTR prompt;
1294 HRESULT hres;
1296 TRACE("\n");
1298 if(args_cnt != 1) {
1299 FIXME("unsupported arg_cnt %d\n", args_cnt);
1300 return E_NOTIMPL;
1303 hres = to_string(args, &prompt);
1304 if(FAILED(hres))
1305 return hres;
1307 hres = show_msgbox(This->desc->ctx, prompt, res);
1308 SysFreeString(prompt);
1309 return hres;
1312 static HRESULT Global_CreateObject(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1314 IUnknown *obj;
1315 HRESULT hres;
1317 TRACE("(%s)\n", debugstr_variant(arg));
1319 if(V_VT(arg) != VT_BSTR) {
1320 FIXME("non-bstr arg\n");
1321 return E_INVALIDARG;
1324 obj = create_object(This->desc->ctx, V_BSTR(arg));
1325 if(!obj)
1326 return VB_E_CANNOT_CREATE_OBJ;
1328 if(res) {
1329 hres = IUnknown_QueryInterface(obj, &IID_IDispatch, (void**)&V_DISPATCH(res));
1330 if(FAILED(hres))
1331 return hres;
1333 V_VT(res) = VT_DISPATCH;
1336 IUnknown_Release(obj);
1337 return S_OK;
1340 static HRESULT Global_GetObject(vbdisp_t *This, VARIANT *args, unsigned args_cnt, VARIANT *res)
1342 IBindCtx *bind_ctx;
1343 IUnknown *obj_unk;
1344 IDispatch *disp;
1345 ULONG eaten = 0;
1346 IMoniker *mon;
1347 HRESULT hres;
1349 TRACE("%s %s\n", args_cnt ? debugstr_variant(args) : "", args_cnt > 1 ? debugstr_variant(args+1) : "");
1351 if(args_cnt != 1 || V_VT(args) != VT_BSTR) {
1352 FIXME("unsupported args\n");
1353 return E_NOTIMPL;
1356 if(This->desc->ctx->safeopt & (INTERFACE_USES_SECURITY_MANAGER|INTERFACESAFE_FOR_UNTRUSTED_DATA)) {
1357 WARN("blocked in current safety mode\n");
1358 return VB_E_CANNOT_CREATE_OBJ;
1361 hres = CreateBindCtx(0, &bind_ctx);
1362 if(FAILED(hres))
1363 return hres;
1365 hres = MkParseDisplayName(bind_ctx, V_BSTR(args), &eaten, &mon);
1366 if(SUCCEEDED(hres)) {
1367 hres = IMoniker_BindToObject(mon, bind_ctx, NULL, &IID_IUnknown, (void**)&obj_unk);
1368 IMoniker_Release(mon);
1369 }else {
1370 hres = MK_E_SYNTAX;
1372 IBindCtx_Release(bind_ctx);
1373 if(FAILED(hres))
1374 return hres;
1376 hres = set_object_site(This->desc->ctx, obj_unk);
1377 if(FAILED(hres)) {
1378 IUnknown_Release(obj_unk);
1379 return hres;
1382 hres = IUnknown_QueryInterface(obj_unk, &IID_IDispatch, (void**)&disp);
1383 if(SUCCEEDED(hres)) {
1384 if(res) {
1385 V_VT(res) = VT_DISPATCH;
1386 V_DISPATCH(res) = disp;
1387 }else {
1388 IDispatch_Release(disp);
1390 }else {
1391 FIXME("object does not support IDispatch\n");
1394 return hres;
1397 static HRESULT Global_DateAdd(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1399 FIXME("\n");
1400 return E_NOTIMPL;
1403 static HRESULT Global_DateDiff(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1405 FIXME("\n");
1406 return E_NOTIMPL;
1409 static HRESULT Global_DatePart(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1411 FIXME("\n");
1412 return E_NOTIMPL;
1415 static HRESULT Global_TypeName(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1417 FIXME("\n");
1418 return E_NOTIMPL;
1421 static HRESULT Global_Array(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1423 FIXME("\n");
1424 return E_NOTIMPL;
1427 static HRESULT Global_Erase(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1429 FIXME("\n");
1430 return E_NOTIMPL;
1433 static HRESULT Global_Filter(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1435 FIXME("\n");
1436 return E_NOTIMPL;
1439 static HRESULT Global_Join(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1441 FIXME("\n");
1442 return E_NOTIMPL;
1445 static HRESULT Global_Split(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1447 FIXME("\n");
1448 return E_NOTIMPL;
1451 static HRESULT Global_Replace(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1453 FIXME("\n");
1454 return E_NOTIMPL;
1457 static HRESULT Global_StrReverse(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1459 WCHAR *ptr1, *ptr2, ch;
1460 BSTR ret;
1461 HRESULT hres;
1463 TRACE("%s\n", debugstr_variant(arg));
1465 hres = to_string(arg, &ret);
1466 if(FAILED(hres))
1467 return hres;
1469 ptr1 = ret;
1470 ptr2 = ret + SysStringLen(ret)-1;
1471 while(ptr1 < ptr2) {
1472 ch = *ptr1;
1473 *ptr1++ = *ptr2;
1474 *ptr2-- = ch;
1477 return return_bstr(res, ret);
1480 static HRESULT Global_InStrRev(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1482 FIXME("\n");
1483 return E_NOTIMPL;
1486 static HRESULT Global_LoadPicture(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1488 FIXME("\n");
1489 return E_NOTIMPL;
1492 static HRESULT Global_ScriptEngine(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1494 FIXME("\n");
1495 return E_NOTIMPL;
1498 static HRESULT Global_ScriptEngineMajorVersion(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1500 FIXME("\n");
1501 return E_NOTIMPL;
1504 static HRESULT Global_ScriptEngineMinorVersion(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1506 FIXME("\n");
1507 return E_NOTIMPL;
1510 static HRESULT Global_ScriptEngineBuildVersion(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1512 FIXME("\n");
1513 return E_NOTIMPL;
1516 static HRESULT Global_FormatNumber(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1518 FIXME("\n");
1519 return E_NOTIMPL;
1522 static HRESULT Global_FormatCurrency(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1524 FIXME("\n");
1525 return E_NOTIMPL;
1528 static HRESULT Global_FormatPercent(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1530 FIXME("\n");
1531 return E_NOTIMPL;
1534 static HRESULT Global_FormatDateTime(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1536 FIXME("\n");
1537 return E_NOTIMPL;
1540 static HRESULT Global_WeekdayName(vbdisp_t *This, VARIANT *args, unsigned args_cnt, VARIANT *res)
1542 int weekday, first_day = 1, abbrev = 0;
1543 BSTR ret;
1544 HRESULT hres;
1546 TRACE("\n");
1548 assert(1 <= args_cnt && args_cnt <= 3);
1550 hres = to_int(args, &weekday);
1551 if(FAILED(hres))
1552 return hres;
1554 if(args_cnt > 1) {
1555 hres = to_int(args+1, &abbrev);
1556 if(FAILED(hres))
1557 return hres;
1559 if(args_cnt == 3) {
1560 hres = to_int(args+2, &first_day);
1561 if(FAILED(hres))
1562 return hres;
1566 hres = VarWeekdayName(weekday, abbrev, first_day, 0, &ret);
1567 if(FAILED(hres))
1568 return hres;
1570 return return_bstr(res, ret);
1573 static HRESULT Global_MonthName(vbdisp_t *This, VARIANT *args, unsigned args_cnt, VARIANT *res)
1575 int month, abbrev = 0;
1576 BSTR ret;
1577 HRESULT hres;
1579 TRACE("\n");
1581 assert(args_cnt == 1 || args_cnt == 2);
1583 hres = to_int(args, &month);
1584 if(FAILED(hres))
1585 return hres;
1587 if(args_cnt == 2) {
1588 hres = to_int(args+1, &abbrev);
1589 if(FAILED(hres))
1590 return hres;
1593 hres = VarMonthName(month, abbrev, 0, &ret);
1594 if(FAILED(hres))
1595 return hres;
1597 return return_bstr(res, ret);
1600 static HRESULT Global_Round(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1602 double n;
1603 HRESULT hres;
1605 TRACE("%s\n", debugstr_variant(arg));
1607 if(!res)
1608 return S_OK;
1610 switch(V_VT(arg)) {
1611 case VT_I2:
1612 case VT_I4:
1613 case VT_BOOL:
1614 *res = *arg;
1615 return S_OK;
1616 case VT_R8:
1617 n = V_R8(arg);
1618 break;
1619 default:
1620 hres = to_double(arg, &n);
1621 if(FAILED(hres))
1622 return hres;
1625 return return_double(res, round(n));
1628 static HRESULT Global_Escape(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1630 FIXME("\n");
1631 return E_NOTIMPL;
1634 static HRESULT Global_Unescape(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1636 FIXME("\n");
1637 return E_NOTIMPL;
1640 static HRESULT Global_Eval(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1642 FIXME("\n");
1643 return E_NOTIMPL;
1646 static HRESULT Global_Execute(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1648 FIXME("\n");
1649 return E_NOTIMPL;
1652 static HRESULT Global_ExecuteGlobal(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1654 FIXME("\n");
1655 return E_NOTIMPL;
1658 static HRESULT Global_GetRef(vbdisp_t *This, VARIANT *arg, unsigned args_cnt, VARIANT *res)
1660 FIXME("\n");
1661 return E_NOTIMPL;
1664 static const string_constant_t vbCr = {1, {'\r'}};
1665 static const string_constant_t vbCrLf = {2, {'\r','\n'}};
1666 static const string_constant_t vbNewLine = {2, {'\r','\n'}};
1667 static const string_constant_t vbFormFeed = {1, {0xc}};
1668 static const string_constant_t vbLf = {1, {'\n'}};
1669 static const string_constant_t vbNullChar = {1};
1670 static const string_constant_t vbNullString = {0};
1671 static const string_constant_t vbTab = {1, {'\t'}};
1672 static const string_constant_t vbVerticalTab = {1, {0xb}};
1674 static const builtin_prop_t global_props[] = {
1675 {DISPID_GLOBAL_VBUSESYSTEM, NULL, BP_GET, VT_I2, 0},
1676 {DISPID_GLOBAL_USESYSTEMDAYOFWEEK, NULL, BP_GET, VT_I2, 0},
1677 {DISPID_GLOBAL_VBSUNDAY, NULL, BP_GET, VT_I2, 1},
1678 {DISPID_GLOBAL_VBMONDAY, NULL, BP_GET, VT_I2, 2},
1679 {DISPID_GLOBAL_VBTUESDAY, NULL, BP_GET, VT_I2, 3},
1680 {DISPID_GLOBAL_VBWEDNESDAY, NULL, BP_GET, VT_I2, 4},
1681 {DISPID_GLOBAL_VBTHURSDAY, NULL, BP_GET, VT_I2, 5},
1682 {DISPID_GLOBAL_VBFRIDAY, NULL, BP_GET, VT_I2, 6},
1683 {DISPID_GLOBAL_VBSATURDAY, NULL, BP_GET, VT_I2, 7},
1684 {DISPID_GLOBAL_VBFIRSTJAN1, NULL, BP_GET, VT_I2, 1},
1685 {DISPID_GLOBAL_VBFIRSTFOURDAYS, NULL, BP_GET, VT_I2, 2},
1686 {DISPID_GLOBAL_VBFIRSTFULLWEEK, NULL, BP_GET, VT_I2, 3},
1687 {DISPID_GLOBAL_VBOKONLY, NULL, BP_GET, VT_I2, MB_OK},
1688 {DISPID_GLOBAL_VBOKCANCEL, NULL, BP_GET, VT_I2, MB_OKCANCEL},
1689 {DISPID_GLOBAL_VBABORTRETRYIGNORE, NULL, BP_GET, VT_I2, MB_ABORTRETRYIGNORE},
1690 {DISPID_GLOBAL_VBYESNOCANCEL, NULL, BP_GET, VT_I2, MB_YESNOCANCEL},
1691 {DISPID_GLOBAL_VBYESNO, NULL, BP_GET, VT_I2, MB_YESNO},
1692 {DISPID_GLOBAL_VBRETRYCANCEL, NULL, BP_GET, VT_I2, MB_RETRYCANCEL},
1693 {DISPID_GLOBAL_VBCRITICAL, NULL, BP_GET, VT_I2, MB_ICONHAND},
1694 {DISPID_GLOBAL_VBQUESTION, NULL, BP_GET, VT_I2, MB_ICONQUESTION},
1695 {DISPID_GLOBAL_VBEXCLAMATION, NULL, BP_GET, VT_I2, MB_ICONEXCLAMATION},
1696 {DISPID_GLOBAL_VBINFORMATION, NULL, BP_GET, VT_I2, MB_ICONASTERISK},
1697 {DISPID_GLOBAL_VBDEFAULTBUTTON1, NULL, BP_GET, VT_I2, MB_DEFBUTTON1},
1698 {DISPID_GLOBAL_VBDEFAULTBUTTON2, NULL, BP_GET, VT_I2, MB_DEFBUTTON2},
1699 {DISPID_GLOBAL_VBDEFAULTBUTTON3, NULL, BP_GET, VT_I2, MB_DEFBUTTON3},
1700 {DISPID_GLOBAL_VBDEFAULTBUTTON4, NULL, BP_GET, VT_I2, MB_DEFBUTTON4},
1701 {DISPID_GLOBAL_VBAPPLICATIONMODAL, NULL, BP_GET, VT_I2, MB_APPLMODAL},
1702 {DISPID_GLOBAL_VBSYSTEMMODAL, NULL, BP_GET, VT_I2, MB_SYSTEMMODAL},
1703 {DISPID_GLOBAL_VBOK, NULL, BP_GET, VT_I2, IDOK},
1704 {DISPID_GLOBAL_VBCANCEL, NULL, BP_GET, VT_I2, IDCANCEL},
1705 {DISPID_GLOBAL_VBABORT, NULL, BP_GET, VT_I2, IDABORT},
1706 {DISPID_GLOBAL_VBRETRY, NULL, BP_GET, VT_I2, IDRETRY},
1707 {DISPID_GLOBAL_VBIGNORE, NULL, BP_GET, VT_I2, IDIGNORE},
1708 {DISPID_GLOBAL_VBYES, NULL, BP_GET, VT_I2, IDYES},
1709 {DISPID_GLOBAL_VBNO, NULL, BP_GET, VT_I2, IDNO},
1710 {DISPID_GLOBAL_VBEMPTY, NULL, BP_GET, VT_I2, VT_EMPTY},
1711 {DISPID_GLOBAL_VBNULL, NULL, BP_GET, VT_I2, VT_NULL},
1712 {DISPID_GLOBAL_VBINTEGER, NULL, BP_GET, VT_I2, VT_I2},
1713 {DISPID_GLOBAL_VBLONG, NULL, BP_GET, VT_I2, VT_I4},
1714 {DISPID_GLOBAL_VBSINGLE, NULL, BP_GET, VT_I2, VT_R4},
1715 {DISPID_GLOBAL_VBDOUBLE, NULL, BP_GET, VT_I2, VT_R8},
1716 {DISPID_GLOBAL_VBCURRENCY, NULL, BP_GET, VT_I2, VT_CY},
1717 {DISPID_GLOBAL_VBDATE, NULL, BP_GET, VT_I2, VT_DATE},
1718 {DISPID_GLOBAL_VBSTRING, NULL, BP_GET, VT_I2, VT_BSTR},
1719 {DISPID_GLOBAL_VBOBJECT, NULL, BP_GET, VT_I2, VT_DISPATCH},
1720 {DISPID_GLOBAL_VBERROR, NULL, BP_GET, VT_I2, VT_ERROR},
1721 {DISPID_GLOBAL_VBBOOLEAN, NULL, BP_GET, VT_I2, VT_BOOL},
1722 {DISPID_GLOBAL_VBVARIANT, NULL, BP_GET, VT_I2, VT_VARIANT},
1723 {DISPID_GLOBAL_VBDATAOBJECT, NULL, BP_GET, VT_I2, VT_UNKNOWN},
1724 {DISPID_GLOBAL_VBDECIMAL, NULL, BP_GET, VT_I2, VT_DECIMAL},
1725 {DISPID_GLOBAL_VBBYTE, NULL, BP_GET, VT_I2, VT_UI1},
1726 {DISPID_GLOBAL_VBARRAY, NULL, BP_GET, VT_I2, VT_ARRAY},
1727 {DISPID_GLOBAL_VBTRUE, NULL, BP_GET, VT_I2, VARIANT_TRUE},
1728 {DISPID_GLOBAL_VBFALSE, NULL, BP_GET, VT_I2, VARIANT_FALSE},
1729 {DISPID_GLOBAL_VBUSEDEFAULT, NULL, BP_GET, VT_I2, -2},
1730 {DISPID_GLOBAL_VBBINARYCOMPARE, NULL, BP_GET, VT_I2, 0},
1731 {DISPID_GLOBAL_VBTEXTCOMPARE, NULL, BP_GET, VT_I2, 1},
1732 {DISPID_GLOBAL_VBDATABASECOMPARE, NULL, BP_GET, VT_I2, 2},
1733 {DISPID_GLOBAL_VBGENERALDATE, NULL, BP_GET, VT_I2, 0},
1734 {DISPID_GLOBAL_VBLONGDATE, NULL, BP_GET, VT_I2, 1},
1735 {DISPID_GLOBAL_VBSHORTDATE, NULL, BP_GET, VT_I2, 2},
1736 {DISPID_GLOBAL_VBLONGTIME, NULL, BP_GET, VT_I2, 3},
1737 {DISPID_GLOBAL_VBSHORTTIME, NULL, BP_GET, VT_I2, 4},
1738 {DISPID_GLOBAL_VBOBJECTERROR, NULL, BP_GET, VT_I4, 0x80040000},
1739 {DISPID_GLOBAL_VBBLACK, NULL, BP_GET, VT_I4, 0x000000},
1740 {DISPID_GLOBAL_VBBLUE, NULL, BP_GET, VT_I4, 0xff0000},
1741 {DISPID_GLOBAL_VBCYAN, NULL, BP_GET, VT_I4, 0xffff00},
1742 {DISPID_GLOBAL_VBGREEN, NULL, BP_GET, VT_I4, 0x00ff00},
1743 {DISPID_GLOBAL_VBMAGENTA, NULL, BP_GET, VT_I4, 0xff00ff},
1744 {DISPID_GLOBAL_VBRED, NULL, BP_GET, VT_I4, 0x0000ff},
1745 {DISPID_GLOBAL_VBWHITE, NULL, BP_GET, VT_I4, 0xffffff},
1746 {DISPID_GLOBAL_VBYELLOW, NULL, BP_GET, VT_I4, 0x00ffff},
1747 {DISPID_GLOBAL_VBCR, NULL, BP_GET, VT_BSTR, (UINT_PTR)&vbCr},
1748 {DISPID_GLOBAL_VBCRLF, NULL, BP_GET, VT_BSTR, (UINT_PTR)&vbCrLf},
1749 {DISPID_GLOBAL_VBNEWLINE, NULL, BP_GET, VT_BSTR, (UINT_PTR)&vbNewLine},
1750 {DISPID_GLOBAL_VBFORMFEED, NULL, BP_GET, VT_BSTR, (UINT_PTR)&vbFormFeed},
1751 {DISPID_GLOBAL_VBLF, NULL, BP_GET, VT_BSTR, (UINT_PTR)&vbLf},
1752 {DISPID_GLOBAL_VBNULLCHAR, NULL, BP_GET, VT_BSTR, (UINT_PTR)&vbNullChar},
1753 {DISPID_GLOBAL_VBNULLSTRING, NULL, BP_GET, VT_BSTR, (UINT_PTR)&vbNullString},
1754 {DISPID_GLOBAL_VBTAB, NULL, BP_GET, VT_BSTR, (UINT_PTR)&vbTab},
1755 {DISPID_GLOBAL_VBVERTICALTAB, NULL, BP_GET, VT_BSTR, (UINT_PTR)&vbVerticalTab},
1756 {DISPID_GLOBAL_CCUR, Global_CCur, 0, 1},
1757 {DISPID_GLOBAL_CINT, Global_CInt, 0, 1},
1758 {DISPID_GLOBAL_CLNG, Global_CLng, 0, 1},
1759 {DISPID_GLOBAL_CBOOL, Global_CBool, 0, 1},
1760 {DISPID_GLOBAL_CBYTE, Global_CByte, 0, 1},
1761 {DISPID_GLOBAL_CDATE, Global_CDate, 0, 1},
1762 {DISPID_GLOBAL_CDBL, Global_CDbl, 0, 1},
1763 {DISPID_GLOBAL_CSNG, Global_CSng, 0, 1},
1764 {DISPID_GLOBAL_CSTR, Global_CStr, 0, 1},
1765 {DISPID_GLOBAL_HEX, Global_Hex, 0, 1},
1766 {DISPID_GLOBAL_OCT, Global_Oct, 0, 1},
1767 {DISPID_GLOBAL_VARTYPE, Global_VarType, 0, 1},
1768 {DISPID_GLOBAL_ISDATE, Global_IsDate, 0, 1},
1769 {DISPID_GLOBAL_ISEMPTY, Global_IsEmpty, 0, 1},
1770 {DISPID_GLOBAL_ISNULL, Global_IsNull, 0, 1},
1771 {DISPID_GLOBAL_ISNUMERIC, Global_IsNumeric, 0, 1},
1772 {DISPID_GLOBAL_ISARRAY, Global_IsArray, 0, 1},
1773 {DISPID_GLOBAL_ISOBJECT, Global_IsObject, 0, 1},
1774 {DISPID_GLOBAL_ATN, Global_Ant, 0, 1},
1775 {DISPID_GLOBAL_COS, Global_Cos, 0, 1},
1776 {DISPID_GLOBAL_SIN, Global_Sin, 0, 1},
1777 {DISPID_GLOBAL_TAN, Global_Tan, 0, 1},
1778 {DISPID_GLOBAL_EXP, Global_Exp, 0, 1},
1779 {DISPID_GLOBAL_LOG, Global_Log, 0, 1},
1780 {DISPID_GLOBAL_SQR, Global_Sqr, 0, 1},
1781 {DISPID_GLOBAL_RANDOMIZE, Global_Randomize, 0, 1},
1782 {DISPID_GLOBAL_RND, Global_Rnd, 0, 1},
1783 {DISPID_GLOBAL_TIMER, Global_Timer, 0, 0},
1784 {DISPID_GLOBAL_LBOUND, Global_LBound, 0, 1},
1785 {DISPID_GLOBAL_UBOUND, Global_UBound, 0, 1},
1786 {DISPID_GLOBAL_RGB, Global_RGB, 0, 3},
1787 {DISPID_GLOBAL_LEN, Global_Len, 0, 1},
1788 {DISPID_GLOBAL_LENB, Global_LenB, 0, 1},
1789 {DISPID_GLOBAL_LEFT, Global_Left, 0, 2},
1790 {DISPID_GLOBAL_LEFTB, Global_LeftB, 0, 2},
1791 {DISPID_GLOBAL_RIGHT, Global_Right, 0, 2},
1792 {DISPID_GLOBAL_RIGHTB, Global_RightB, 0, 2},
1793 {DISPID_GLOBAL_MID, Global_Mid, 0, 2, 3},
1794 {DISPID_GLOBAL_MIDB, Global_MidB, 0, 2, 3},
1795 {DISPID_GLOBAL_STRCOMP, Global_StrComp, 0, 2, 3},
1796 {DISPID_GLOBAL_LCASE, Global_LCase, 0, 1},
1797 {DISPID_GLOBAL_UCASE, Global_UCase, 0, 1},
1798 {DISPID_GLOBAL_LTRIM, Global_LTrim, 0, 1},
1799 {DISPID_GLOBAL_RTRIM, Global_RTrim, 0, 1},
1800 {DISPID_GLOBAL_TRIM, Global_Trim, 0, 1},
1801 {DISPID_GLOBAL_SPACE, Global_Space, 0, 1},
1802 {DISPID_GLOBAL_STRING, Global_String, 0, 0, 2},
1803 {DISPID_GLOBAL_INSTR, Global_InStr, 0, 2, 4},
1804 {DISPID_GLOBAL_INSTRB, Global_InStrB, 0, 3, 4},
1805 {DISPID_GLOBAL_ASCB, Global_AscB, 0, 1},
1806 {DISPID_GLOBAL_CHRB, Global_ChrB, 0, 1},
1807 {DISPID_GLOBAL_ASC, Global_Asc, 0, 1},
1808 {DISPID_GLOBAL_CHR, Global_Chr, 0, 1},
1809 {DISPID_GLOBAL_ASCW, Global_AscW, 0, 1},
1810 {DISPID_GLOBAL_CHRW, Global_ChrW, 0, 1},
1811 {DISPID_GLOBAL_ABS, Global_Abs, 0, 1},
1812 {DISPID_GLOBAL_FIX, Global_Fix, 0, 1},
1813 {DISPID_GLOBAL_INT, Global_Int, 0, 1},
1814 {DISPID_GLOBAL_SGN, Global_Sgn, 0, 1},
1815 {DISPID_GLOBAL_NOW, Global_Now, 0, 0},
1816 {DISPID_GLOBAL_DATE, Global_Date, 0, 0},
1817 {DISPID_GLOBAL_TIME, Global_Time, 0, 0},
1818 {DISPID_GLOBAL_DAY, Global_Day, 0, 1},
1819 {DISPID_GLOBAL_MONTH, Global_Month, 0, 1},
1820 {DISPID_GLOBAL_WEEKDAY, Global_Weekday, 0, 1, 2},
1821 {DISPID_GLOBAL_YEAR, Global_Year, 0, 1},
1822 {DISPID_GLOBAL_HOUR, Global_Hour, 0, 1},
1823 {DISPID_GLOBAL_MINUTE, Global_Minute, 0, 1},
1824 {DISPID_GLOBAL_SECOND, Global_Second, 0, 1},
1825 {DISPID_GLOBAL_DATEVALUE, Global_DateValue, 0, 1},
1826 {DISPID_GLOBAL_TIMEVALUE, Global_TimeValue, 0, 1},
1827 {DISPID_GLOBAL_DATESERIAL, Global_DateSerial, 0, 3},
1828 {DISPID_GLOBAL_TIMESERIAL, Global_TimeSerial, 0, 3},
1829 {DISPID_GLOBAL_INPUTBOX, Global_InputBox, 0, 1, 7},
1830 {DISPID_GLOBAL_MSGBOX, Global_MsgBox, 0, 1, 5},
1831 {DISPID_GLOBAL_CREATEOBJECT, Global_CreateObject, 0, 1},
1832 {DISPID_GLOBAL_GETOBJECT, Global_GetObject, 0, 0, 2},
1833 {DISPID_GLOBAL_DATEADD, Global_DateAdd, 0, 3},
1834 {DISPID_GLOBAL_DATEDIFF, Global_DateDiff, 0, 3, 5},
1835 {DISPID_GLOBAL_DATEPART, Global_DatePart, 0, 2, 4},
1836 {DISPID_GLOBAL_TYPENAME, Global_TypeName, 0, 1},
1837 {DISPID_GLOBAL_ARRAY, Global_Array, 0, 1},
1838 {DISPID_GLOBAL_ERASE, Global_Erase, 0, 1},
1839 {DISPID_GLOBAL_FILTER, Global_Filter, 0, 2, 4},
1840 {DISPID_GLOBAL_JOIN, Global_Join, 0, 1, 2},
1841 {DISPID_GLOBAL_SPLIT, Global_Split, 0, 1, 4},
1842 {DISPID_GLOBAL_REPLACE, Global_Replace, 0, 3, 6},
1843 {DISPID_GLOBAL_STRREVERSE, Global_StrReverse, 0, 1},
1844 {DISPID_GLOBAL_INSTRREV, Global_InStrRev, 0, 2, 4},
1845 {DISPID_GLOBAL_LOADPICTURE, Global_LoadPicture, 0, 1},
1846 {DISPID_GLOBAL_SCRIPTENGINE, Global_ScriptEngine, 0, 0},
1847 {DISPID_GLOBAL_SCRIPTENGINEMAJORVERSION, Global_ScriptEngineMajorVersion, 0, 0},
1848 {DISPID_GLOBAL_SCRIPTENGINEMINORVERSION, Global_ScriptEngineMinorVersion, 0, 0},
1849 {DISPID_GLOBAL_SCRIPTENGINEBUILDVERSION, Global_ScriptEngineBuildVersion, 0, 0},
1850 {DISPID_GLOBAL_FORMATNUMBER, Global_FormatNumber, 0, 1, 5},
1851 {DISPID_GLOBAL_FORMATCURRENCY, Global_FormatCurrency, 0, 1, 5},
1852 {DISPID_GLOBAL_FORMATPERCENT, Global_FormatPercent, 0, 1, 5},
1853 {DISPID_GLOBAL_FORMATDATETIME, Global_FormatDateTime, 0, 1, 2},
1854 {DISPID_GLOBAL_WEEKDAYNAME, Global_WeekdayName, 0, 1, 3},
1855 {DISPID_GLOBAL_MONTHNAME, Global_MonthName, 0, 1, 2},
1856 {DISPID_GLOBAL_ROUND, Global_Round, 0, 1, 2},
1857 {DISPID_GLOBAL_ESCAPE, Global_Escape, 0, 1},
1858 {DISPID_GLOBAL_UNESCAPE, Global_Unescape, 0, 1},
1859 {DISPID_GLOBAL_EVAL, Global_Eval, 0, 1},
1860 {DISPID_GLOBAL_EXECUTE, Global_Execute, 0, 1},
1861 {DISPID_GLOBAL_EXECUTEGLOBAL, Global_ExecuteGlobal, 0, 1},
1862 {DISPID_GLOBAL_GETREF, Global_GetRef, 0, 1},
1863 {DISPID_GLOBAL_VBMSGBOXHELPBUTTON, NULL, BP_GET, VT_I4, MB_HELP},
1864 {DISPID_GLOBAL_VBMSGBOXSETFOREGROUND, NULL, BP_GET, VT_I4, MB_SETFOREGROUND},
1865 {DISPID_GLOBAL_VBMSGBOXRIGHT, NULL, BP_GET, VT_I4, MB_RIGHT},
1866 {DISPID_GLOBAL_VBMSGBOXRTLREADING, NULL, BP_GET, VT_I4, MB_RTLREADING}
1869 HRESULT init_global(script_ctx_t *ctx)
1871 HRESULT hres;
1873 ctx->global_desc.ctx = ctx;
1874 ctx->global_desc.builtin_prop_cnt = sizeof(global_props)/sizeof(*global_props);
1875 ctx->global_desc.builtin_props = global_props;
1877 hres = get_typeinfo(GlobalObj_tid, &ctx->global_desc.typeinfo);
1878 if(FAILED(hres))
1879 return hres;
1881 hres = create_vbdisp(&ctx->global_desc, &ctx->global_obj);
1882 if(FAILED(hres))
1883 return hres;
1885 hres = create_script_disp(ctx, &ctx->script_obj);
1886 if(FAILED(hres))
1887 return hres;
1889 return init_err(ctx);