/* Exports struct view_state to the world of ECMAScript */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include #include "elinks.h" #include "ecmascript/spidermonkey/util.h" #include "protocol/uri.h" #include "scripting/smjs/view_state_object.h" #include "scripting/smjs/core.h" #include "util/error.h" #include "util/memory.h" #include "viewer/text/vs.h" static const JSClass view_state_class; /* defined below */ enum view_state_prop { VIEW_STATE_PLAIN, VIEW_STATE_URI, }; static const JSPropertySpec view_state_props[] = { { "plain", VIEW_STATE_PLAIN, JSPROP_ENUMERATE }, { "uri", VIEW_STATE_URI, JSPROP_ENUMERATE | JSPROP_READONLY }, { NULL } }; /* @view_state_class.getProperty */ static JSBool view_state_get_property(JSContext *ctx, JSObject *obj, jsval id, jsval *vp) { struct view_state *vs; /* This can be called if @obj if not itself an instance of the * appropriate class but has one in its prototype chain. Fail * such calls. */ if (!JS_InstanceOf(ctx, obj, (JSClass *) &view_state_class, NULL)) return JS_FALSE; vs = JS_GetInstancePrivate(ctx, obj, (JSClass *) &view_state_class, NULL); undef_to_jsval(ctx, vp); if (!JSVAL_IS_INT(id)) return JS_FALSE; switch (JSVAL_TO_INT(id)) { case VIEW_STATE_PLAIN: *vp = INT_TO_JSVAL(vs->plain); return JS_TRUE; case VIEW_STATE_URI: *vp = STRING_TO_JSVAL(JS_NewStringCopyZ(smjs_ctx, struri(vs->uri))); return JS_TRUE; default: /* Unrecognized integer property ID; someone is using * the object as an array. SMJS builtin classes (e.g. * js_RegExpClass) just return JS_TRUE in this case * and leave *@vp unchanged. Do the same here. * (Actually not quite the same, as we already used * @undef_to_jsval.) */ return JS_TRUE; } } /* @view_state_class.setProperty */ static JSBool view_state_set_property(JSContext *ctx, JSObject *obj, jsval id, jsval *vp) { struct view_state *vs; /* This can be called if @obj if not itself an instance of the * appropriate class but has one in its prototype chain. Fail * such calls. */ if (!JS_InstanceOf(ctx, obj, (JSClass *) &view_state_class, NULL)) return JS_FALSE; vs = JS_GetInstancePrivate(ctx, obj, (JSClass *) &view_state_class, NULL); if (!JSVAL_IS_INT(id)) return JS_FALSE; switch (JSVAL_TO_INT(id)) { case VIEW_STATE_PLAIN: { vs->plain = atol(jsval_to_string(ctx, vp)); return JS_TRUE; } default: /* Unrecognized integer property ID; someone is using * the object as an array. SMJS builtin classes (e.g. * js_RegExpClass) just return JS_TRUE in this case. * Do the same here. */ return JS_TRUE; } } static const JSClass view_state_class = { "view_state", JSCLASS_HAS_PRIVATE, /* struct view_state * */ JS_PropertyStub, JS_PropertyStub, view_state_get_property, view_state_set_property, JS_EnumerateStub, JS_ResolveStub, JS_ConvertStub, JS_FinalizeStub }; JSObject * smjs_get_view_state_object(struct view_state *vs) { JSObject *view_state_object; assert(smjs_ctx); view_state_object = JS_NewObject(smjs_ctx, (JSClass *) &view_state_class, NULL, NULL); if (!view_state_object) return NULL; if (JS_FALSE == JS_SetPrivate(smjs_ctx, view_state_object, vs)) /* to @view_state_class */ return NULL; if (JS_FALSE == JS_DefineProperties(smjs_ctx, view_state_object, (JSPropertySpec *) view_state_props)) return NULL; return view_state_object; }