/* "elinks.action" */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include "elinks.h" #include "config/kbdbind.h" #include "ecmascript/spidermonkey-shared.h" #include "intl/gettext/libintl.h" #include "scripting/smjs/core.h" #include "scripting/smjs/elinks_object.h" #include "session/session.h" #include "terminal/window.h" #include "util/memory.h" #include "viewer/action.h" /*** action method object ***/ struct smjs_action_fn_callback_hop { struct session *ses; action_id_T action_id; }; static void smjs_action_fn_finalize(JSFreeOp *op, JSObject *obj); static JSBool smjs_action_fn_callback(JSContext *ctx, unsigned int argc, jsval *rval); static const JSClass action_fn_class = { "action_fn", JSCLASS_HAS_PRIVATE, /* struct smjs_action_fn_callback_hop * */ JS_PropertyStub, JS_PropertyStub, JS_PropertyStub, JS_StrictPropertyStub, JS_EnumerateStub, JS_ResolveStub, JS_ConvertStub, smjs_action_fn_finalize, NULL, smjs_action_fn_callback, }; /* @action_fn_class.finalize */ static void smjs_action_fn_finalize(JSFreeOp *op, JSObject *obj) { struct smjs_action_fn_callback_hop *hop; #if 0 assert(JS_InstanceOf(ctx, obj, (JSClass *) &action_fn_class, NULL)); if_assert_failed return; hop = JS_GetInstancePrivate(ctx, obj, (JSClass *) &action_fn_class, NULL); #endif hop = JS_GetPrivate(obj); mem_free_if(hop); } /* @action_fn_class.call */ static JSBool smjs_action_fn_callback(JSContext *ctx, unsigned int argc, jsval *rval) { jsval value; jsval *argv = JS_ARGV(ctx, rval); struct smjs_action_fn_callback_hop *hop; JSObject *fn_obj; assert(smjs_ctx); if_assert_failed return JS_FALSE; value = JSVAL_FALSE; if (JS_TRUE != JS_ValueToObject(ctx, JS_CALLEE(ctx, rval), &fn_obj)) { JS_SET_RVAL(ctx, rval, value); return JS_TRUE; } assert(JS_InstanceOf(ctx, fn_obj, (JSClass *) &action_fn_class, NULL)); if_assert_failed return JS_FALSE; hop = JS_GetInstancePrivate(ctx, fn_obj, (JSClass *) &action_fn_class, NULL); if (!hop) { JS_SET_RVAL(ctx, rval, value); return JS_TRUE; } if (!would_window_receive_keypresses(hop->ses->tab)) { /* The user cannot run actions in this tab by pressing * keys, and some actions could crash if run in this * situation; so we don't let user scripts run actions * either. * * In particular, this check should fix bug 943, where * ::ACT_MAIN_TAB_CLOSE called destroy_session(), * which freed struct type_query while BFU dialogs had * pointers to it. That crash could be prevented in * various ways but it seems other similar crashes are * possible, e.g. if the link menu is open and has a * pointer to a session that is then destroyed. * Instead of thoroughly auditing the use of pointers * to sessions and related structures, I'll just * disable the feature, to bring the ELinks 0.12 * release closer. * * The "%s" prevents interpretation of any percent * signs in translations. */ JS_ReportError(ctx, "%s", _("Cannot run actions in a tab that doesn't " "have the focus", hop->ses->tab->term)); return JS_FALSE; /* make JS propagate the exception */ } if (argc >= 1) { int32_t val; if (JS_TRUE == JS_ValueToInt32(smjs_ctx, argv[0], &val)) { set_kbd_repeat_count(hop->ses, val); } } do_action(hop->ses, hop->action_id, 1); value = JSVAL_TRUE; JS_SET_RVAL(ctx, rval, value); return JS_TRUE; } static JSObject * smjs_get_action_fn_object(unsigned char *action_str) { unsigned char *c; struct smjs_action_fn_callback_hop *hop; JSObject *obj; if (!smjs_ses) return NULL; obj = JS_NewObject(smjs_ctx, (JSClass *) &action_fn_class, NULL, NULL); if (!obj) return NULL; hop = mem_alloc(sizeof(*hop)); if (!hop) return NULL; hop->ses = smjs_ses; /* ECMAScript methods cannot have hyphens in the name, so one should * use underscores instead; here, we must convert them back to hyphens * for the action code. */ for (c = action_str; *c; ++c) if (*c == '_') *c = '-'; hop->action_id = get_action_from_string(KEYMAP_MAIN, action_str); if (-1 != hop->action_id) { JS_SetPrivate(obj, hop); /* to @action_fn_class */ return obj; } mem_free(hop); return NULL; } /*** elinks.action object ***/ /* @action_class.getProperty */ static JSBool action_get_property(JSContext *ctx, JSHandleObject hobj, JSHandleId hid, JSMutableHandleValue hvp) { ELINKS_CAST_PROP_PARAMS jsid id = (hid); (void)obj; jsval val; JSObject *action_fn; unsigned char *action_str; *vp = JSVAL_NULL; JS_IdToValue(ctx, id, &val); action_str = JS_EncodeString(ctx, JS_ValueToString(ctx, val)); if (!action_str) return JS_TRUE; action_fn = smjs_get_action_fn_object(action_str); if (!action_fn) return JS_TRUE; *vp = OBJECT_TO_JSVAL(action_fn); return JS_TRUE; } static const JSClass action_class = { "action", 0, JS_PropertyStub, JS_PropertyStub, action_get_property, JS_StrictPropertyStub, JS_EnumerateStub, JS_ResolveStub, JS_ConvertStub, NULL, }; static JSObject * smjs_get_action_object(void) { JSObject *obj; assert(smjs_ctx); obj = JS_NewObject(smjs_ctx, (JSClass *) &action_class, NULL, NULL); return obj; } void smjs_init_action_interface(void) { jsval val; struct JSObject *action_object; if (!smjs_ctx || !smjs_elinks_object) return; action_object = smjs_get_action_object(); if (!action_object) return; val = OBJECT_TO_JSVAL(action_object); JS_SetProperty(smjs_ctx, smjs_elinks_object, "action", &val); }