2011-11-13 03:01:27 -05:00
|
|
|
/* Exports struct terminal to the world of ECMAScript */
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
#include "elinks.h"
|
|
|
|
|
|
|
|
#include "ecmascript/spidermonkey/util.h"
|
|
|
|
#include "scripting/smjs/core.h"
|
|
|
|
#include "scripting/smjs/elinks_object.h"
|
|
|
|
#include "scripting/smjs/session_object.h"
|
|
|
|
#include "terminal/terminal.h"
|
|
|
|
#include "util/error.h"
|
|
|
|
#include "util/memory.h"
|
|
|
|
#include "viewer/text/vs.h"
|
|
|
|
|
2022-10-01 15:08:03 -04:00
|
|
|
static void terminal_finalize(JS::GCContext *op, JSObject *obj);
|
2020-10-05 14:14:55 -04:00
|
|
|
|
2020-10-27 09:53:24 -04:00
|
|
|
static const JSClassOps terminal_ops = {
|
2021-08-27 13:46:05 -04:00
|
|
|
nullptr, // addProperty
|
|
|
|
nullptr, // deleteProperty
|
|
|
|
nullptr, // enumerate
|
|
|
|
nullptr, // newEnumerate
|
|
|
|
nullptr, // resolve
|
|
|
|
nullptr, // mayResolve
|
2022-06-20 10:06:13 -04:00
|
|
|
terminal_finalize, // finalize
|
2021-08-27 13:46:05 -04:00
|
|
|
nullptr, // call
|
|
|
|
nullptr, // construct
|
|
|
|
nullptr // trace JS_GlobalObjectTraceHook
|
2020-10-05 14:14:55 -04:00
|
|
|
};
|
2011-11-13 03:01:27 -05:00
|
|
|
|
2020-10-27 09:53:24 -04:00
|
|
|
static const JSClass terminal_class = {
|
|
|
|
"terminal",
|
2022-10-01 15:08:03 -04:00
|
|
|
JSCLASS_HAS_RESERVED_SLOTS(1), /* struct terminal *; a weak refernce */
|
2020-10-27 09:53:24 -04:00
|
|
|
&terminal_ops
|
|
|
|
};
|
|
|
|
|
2011-11-13 03:01:27 -05:00
|
|
|
enum terminal_prop {
|
|
|
|
TERMINAL_TAB,
|
|
|
|
};
|
|
|
|
|
2021-09-05 11:50:41 -04:00
|
|
|
|
|
|
|
static bool
|
|
|
|
terminal_get_property_tab(JSContext *ctx, unsigned int argc, JS::Value *vp)
|
|
|
|
{
|
|
|
|
#ifdef ECMASCRIPT_DEBUG
|
|
|
|
fprintf(stderr, "%s:%s\n", __FILE__, __FUNCTION__);
|
|
|
|
#endif
|
|
|
|
JS::CallArgs args = CallArgsFromVp(argc, vp);
|
|
|
|
JS::RootedObject hobj(ctx, &args.thisv().toObject());
|
|
|
|
|
|
|
|
struct terminal *term;
|
|
|
|
|
2022-10-01 15:08:03 -04:00
|
|
|
term = JS::GetMaybePtrFromReservedSlot<struct terminal>(hobj, 0);
|
2021-09-05 11:50:41 -04:00
|
|
|
if (!term) return false; /* already detached */
|
|
|
|
|
|
|
|
JSObject *obj = smjs_get_session_array_object(term);
|
|
|
|
|
|
|
|
if (obj) {
|
|
|
|
args.rval().setObject(*obj);
|
|
|
|
} else {
|
|
|
|
args.rval().setUndefined();
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2011-11-13 03:01:27 -05:00
|
|
|
static const JSPropertySpec terminal_props[] = {
|
2021-09-05 11:50:41 -04:00
|
|
|
JS_PSG("tab", terminal_get_property_tab, JSPROP_ENUMERATE),
|
|
|
|
JS_PS_END
|
2011-11-13 03:01:27 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
/** Pointed to by terminal_class.finalize. SpiderMonkey automatically
|
|
|
|
* finalizes all objects before it frees the JSRuntime, so terminal.jsobject
|
|
|
|
* won't be left dangling. */
|
|
|
|
static void
|
2022-10-01 15:08:03 -04:00
|
|
|
terminal_finalize(JS::GCContext *op, JSObject *obj)
|
2011-11-13 03:01:27 -05:00
|
|
|
{
|
|
|
|
struct terminal *term;
|
2022-10-01 15:08:03 -04:00
|
|
|
term = JS::GetMaybePtrFromReservedSlot<struct terminal>(obj, 0);
|
2011-11-13 03:01:27 -05:00
|
|
|
|
|
|
|
if (!term) return; /* already detached */
|
|
|
|
|
2022-10-01 15:08:03 -04:00
|
|
|
JS::SetReservedSlot(obj, 0, JS::UndefinedValue()); /* perhaps not necessary */
|
2011-11-13 03:01:27 -05:00
|
|
|
assert(term->jsobject == obj);
|
|
|
|
if_assert_failed return;
|
|
|
|
term->jsobject = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/** Return an SMJS object through which scripts can access @a term.
|
|
|
|
* If there already is such an object, return that; otherwise create a
|
|
|
|
* new one. The SMJS object holds only a weak reference to @a term. */
|
|
|
|
JSObject *
|
|
|
|
smjs_get_terminal_object(struct terminal *term)
|
|
|
|
{
|
|
|
|
JSObject *obj;
|
|
|
|
|
|
|
|
if (term->jsobject) return term->jsobject;
|
|
|
|
|
|
|
|
assert(smjs_ctx);
|
|
|
|
if_assert_failed return NULL;
|
|
|
|
|
2020-10-16 13:54:02 -04:00
|
|
|
obj = JS_NewObject(smjs_ctx, (JSClass *) &terminal_class);
|
2011-11-13 03:01:27 -05:00
|
|
|
|
|
|
|
if (!obj) return NULL;
|
|
|
|
|
2020-10-12 12:43:56 -04:00
|
|
|
JS::RootedObject robj(smjs_ctx, obj);
|
|
|
|
if (false == JS_DefineProperties(smjs_ctx, robj,
|
2011-11-13 03:01:27 -05:00
|
|
|
(JSPropertySpec *) terminal_props))
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
/* Do this last, so that if any previous step fails, we can
|
|
|
|
* just forget the object and its finalizer won't attempt to
|
|
|
|
* access @cached. */
|
2022-10-01 15:08:03 -04:00
|
|
|
JS::SetReservedSlot(obj, 0, JS::PrivateValue(term)); /* to @terminal_class */
|
2011-11-13 03:01:27 -05:00
|
|
|
|
|
|
|
term->jsobject = obj;
|
|
|
|
return obj;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Ensure that no JSObject contains the pointer @a term. This is called from
|
|
|
|
* destroy_terminal before @a term is freed. If a JSObject was previously
|
|
|
|
* attached to the terminal object, the object will remain in memory but it
|
|
|
|
* will no longer be able to access the terminal object. */
|
|
|
|
void
|
|
|
|
smjs_detach_terminal_object(struct terminal *term)
|
|
|
|
{
|
|
|
|
assert(smjs_ctx);
|
|
|
|
assert(term);
|
|
|
|
if_assert_failed return;
|
|
|
|
|
|
|
|
smjs_detach_session_array_object(term);
|
|
|
|
|
|
|
|
if (!term->jsobject) return;
|
|
|
|
|
2020-10-12 12:43:56 -04:00
|
|
|
JS::RootedObject r_jsobject(smjs_ctx, term->jsobject);
|
2022-10-01 15:08:03 -04:00
|
|
|
JS::SetReservedSlot(term->jsobject, 0, JS::UndefinedValue());
|
2011-11-13 03:01:27 -05:00
|
|
|
term->jsobject = NULL;
|
|
|
|
}
|
|
|
|
|
2020-10-27 09:53:24 -04:00
|
|
|
static const JSClassOps terminal_array_ops = {
|
2021-08-27 13:46:05 -04:00
|
|
|
nullptr, // addProperty
|
|
|
|
nullptr, // deleteProperty
|
|
|
|
nullptr, // enumerate
|
|
|
|
nullptr, // newEnumerate
|
|
|
|
nullptr, // resolve
|
|
|
|
nullptr, // mayResolve
|
|
|
|
nullptr, // finalize
|
|
|
|
nullptr, // call
|
|
|
|
nullptr, // construct
|
2021-09-05 11:50:41 -04:00
|
|
|
JS_GlobalObjectTraceHook
|
2011-11-13 03:01:27 -05:00
|
|
|
};
|
|
|
|
|
2020-10-27 09:53:24 -04:00
|
|
|
static const JSClass terminal_array_class = {
|
|
|
|
"terminal_array",
|
|
|
|
0,
|
|
|
|
&terminal_array_ops
|
|
|
|
};
|
|
|
|
|
2011-11-13 03:01:27 -05:00
|
|
|
/** Return an SMJS object that scripts can use an array to get terminal
|
|
|
|
* objects. */
|
|
|
|
static JSObject *
|
|
|
|
smjs_get_terminal_array_object(void)
|
|
|
|
{
|
|
|
|
assert(smjs_ctx);
|
|
|
|
if_assert_failed return NULL;
|
|
|
|
|
2020-10-16 13:54:02 -04:00
|
|
|
return JS_NewObject(smjs_ctx, (JSClass *) &terminal_array_class);
|
2011-11-13 03:01:27 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
smjs_init_terminal_interface(void)
|
|
|
|
{
|
2020-10-23 16:34:58 -04:00
|
|
|
JS::Value val;
|
2011-11-13 03:01:27 -05:00
|
|
|
struct JSObject *obj;
|
|
|
|
|
|
|
|
if (!smjs_ctx || !smjs_elinks_object)
|
|
|
|
return;
|
|
|
|
|
|
|
|
obj = smjs_get_terminal_array_object();
|
|
|
|
if (!obj) return;
|
|
|
|
|
2020-10-12 12:43:56 -04:00
|
|
|
JS::RootedValue rval(smjs_ctx, val);
|
|
|
|
rval.setObject(*obj);
|
|
|
|
JS::RootedObject r_smjs_elinks_object(smjs_ctx, smjs_elinks_object);
|
2011-11-13 03:01:27 -05:00
|
|
|
|
2020-10-12 12:43:56 -04:00
|
|
|
JS_SetProperty(smjs_ctx, r_smjs_elinks_object, "terminal", rval);
|
2011-11-13 03:01:27 -05:00
|
|
|
}
|