jscript: Added Object_valueOf implementation.
[wine] / dlls / jscript / object.c
1 /*
2  * Copyright 2008 Jacek Caban for CodeWeavers
3  *
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.
8  *
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.
13  *
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
17  */
18
19 #include "jscript.h"
20
21 #include "wine/debug.h"
22
23 WINE_DEFAULT_DEBUG_CHANNEL(jscript);
24
25 static const WCHAR toStringW[] = {'t','o','S','t','r','i','n','g',0};
26 static const WCHAR toLocaleStringW[] = {'t','o','L','o','c','a','l','e','S','t','r','i','n','g',0};
27 static const WCHAR valueOfW[] = {'v','a','l','u','e','O','f',0};
28 static const WCHAR hasOwnPropertyW[] = {'h','a','s','O','w','n','P','r','o','p','e','r','t','y',0};
29 static const WCHAR propertyIsEnumerableW[] =
30     {'p','r','o','p','e','r','t','y','I','s','E','n','u','m','e','r','a','b','l','e',0};
31 static const WCHAR isPrototypeOfW[] = {'i','s','P','r','o','t','o','t','y','p','e','O','f',0};
32
33 static const WCHAR default_valueW[] = {'[','o','b','j','e','c','t',' ','O','b','j','e','c','t',']',0};
34
35 static HRESULT Object_toString(DispatchEx *dispex, LCID lcid, WORD flags, DISPPARAMS *dp,
36         VARIANT *retv, jsexcept_t *ei, IServiceProvider *sp)
37 {
38     FIXME("\n");
39     return E_NOTIMPL;
40 }
41
42 static HRESULT Object_toLocaleString(DispatchEx *dispex, LCID lcid, WORD flags, DISPPARAMS *dp,
43         VARIANT *retv, jsexcept_t *ei, IServiceProvider *sp)
44 {
45     FIXME("\n");
46     return E_NOTIMPL;
47 }
48
49 static HRESULT Object_valueOf(DispatchEx *dispex, LCID lcid, WORD flags, DISPPARAMS *dp,
50         VARIANT *retv, jsexcept_t *ei, IServiceProvider *sp)
51 {
52     TRACE("\n");
53
54     if(retv) {
55         IDispatchEx_AddRef(_IDispatchEx_(dispex));
56
57         V_VT(retv) = VT_DISPATCH;
58         V_DISPATCH(retv) = (IDispatch*)_IDispatchEx_(dispex);
59     }
60
61     return S_OK;
62 }
63
64 static HRESULT Object_hasOwnProperty(DispatchEx *dispex, LCID lcid, WORD flags, DISPPARAMS *dp,
65         VARIANT *retv, jsexcept_t *ei, IServiceProvider *sp)
66 {
67     FIXME("\n");
68     return E_NOTIMPL;
69 }
70
71 static HRESULT Object_propertyIsEnumerable(DispatchEx *dispex, LCID lcid, WORD flags, DISPPARAMS *dp,
72         VARIANT *retv, jsexcept_t *ei, IServiceProvider *sp)
73 {
74     FIXME("\n");
75     return E_NOTIMPL;
76 }
77
78 static HRESULT Object_isPrototypeOf(DispatchEx *dispex, LCID lcid, WORD flags, DISPPARAMS *dp,
79         VARIANT *retv, jsexcept_t *ei, IServiceProvider *sp)
80 {
81     FIXME("\n");
82     return E_NOTIMPL;
83 }
84
85 static HRESULT Object_value(DispatchEx *dispex, LCID lcid, WORD flags, DISPPARAMS *dp,
86         VARIANT *retv, jsexcept_t *ei, IServiceProvider *sp)
87 {
88     TRACE("\n");
89
90     switch(flags) {
91     case DISPATCH_PROPERTYGET:
92         V_VT(retv) = VT_BSTR;
93         V_BSTR(retv) = SysAllocString(default_valueW);
94         if(!V_BSTR(retv))
95             return E_OUTOFMEMORY;
96         break;
97     default:
98         FIXME("unimplemented flags %x\n", flags);
99         return E_NOTIMPL;
100     }
101
102     return S_OK;
103 }
104
105 static void Object_destructor(DispatchEx *dispex)
106 {
107     heap_free(dispex);
108 }
109
110 static const builtin_prop_t Object_props[] = {
111     {hasOwnPropertyW,        Object_hasOwnProperty,        PROPF_METHOD},
112     {isPrototypeOfW,         Object_isPrototypeOf,         PROPF_METHOD},
113     {propertyIsEnumerableW,  Object_propertyIsEnumerable,  PROPF_METHOD},
114     {toLocaleStringW,        Object_toLocaleString,        PROPF_METHOD},
115     {toStringW,              Object_toString,              PROPF_METHOD},
116     {valueOfW,               Object_valueOf,               PROPF_METHOD}
117 };
118
119 static const builtin_info_t Object_info = {
120     JSCLASS_OBJECT,
121     {NULL, Object_value, 0},
122     sizeof(Object_props)/sizeof(*Object_props),
123     Object_props,
124     Object_destructor,
125     NULL
126 };
127
128 static HRESULT ObjectConstr_value(DispatchEx *dispex, LCID lcid, WORD flags, DISPPARAMS *dp,
129         VARIANT *retv, jsexcept_t *ei, IServiceProvider *sp)
130 {
131     HRESULT hres;
132
133     TRACE("\n");
134
135     switch(flags) {
136     case DISPATCH_CONSTRUCT: {
137         DispatchEx *obj;
138
139         hres = create_object(dispex->ctx, NULL, &obj);
140         if(FAILED(hres))
141             return hres;
142
143         V_VT(retv) = VT_DISPATCH;
144         V_DISPATCH(retv) = (IDispatch*)_IDispatchEx_(obj);
145         break;
146     }
147
148     default:
149         FIXME("unimplemented flags: %x\n", flags);
150         return E_NOTIMPL;
151     }
152
153     return S_OK;
154 }
155
156 HRESULT create_object_constr(script_ctx_t *ctx, DispatchEx **ret)
157 {
158     DispatchEx *object;
159     HRESULT hres;
160
161     hres = create_dispex(ctx, &Object_info, NULL, &object);
162     if(FAILED(hres))
163         return hres;
164
165     hres = create_builtin_function(ctx, ObjectConstr_value, NULL, PROPF_CONSTR, object, ret);
166
167     jsdisp_release(object);
168     return hres;
169 }
170
171 HRESULT create_object(script_ctx_t *ctx, DispatchEx *constr, DispatchEx **ret)
172 {
173     DispatchEx *object;
174     HRESULT hres;
175
176     object = heap_alloc_zero(sizeof(DispatchEx));
177     if(!object)
178         return E_OUTOFMEMORY;
179
180     hres = init_dispex_from_constr(object, ctx, &Object_info, constr ? constr : ctx->object_constr);
181     if(FAILED(hres)) {
182         heap_free(object);
183         return hres;
184     }
185
186     *ret = object;
187     return S_OK;
188 }