/* * QuickJS: Example of C module with a class * * Copyright (c) 2019 Fabrice Bellard * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN * THE SOFTWARE. */ #include "../quickjs.h" #include #define countof(x) (sizeof(x) / sizeof((x)[0])) /* Point Class */ typedef struct { int x; int y; } JSPointData; static JSClassID js_point_class_id; static void js_point_finalizer(JSRuntime *rt, JSValue val) { JSPointData *s = JS_GetOpaque(val, js_point_class_id); /* Note: 's' can be NULL in case JS_SetOpaque() was not called */ js_free_rt(rt, s); } static JSValue js_point_ctor(JSContext *ctx, JSValueConst new_target, int argc, JSValueConst *argv) { JSPointData *s; JSValue obj = JS_UNDEFINED; JSValue proto; s = js_mallocz(ctx, sizeof(*s)); if (!s) return JS_EXCEPTION; if (JS_ToInt32(ctx, &s->x, argv[0])) goto fail; if (JS_ToInt32(ctx, &s->y, argv[1])) goto fail; /* using new_target to get the prototype is necessary when the class is extended. */ proto = JS_GetPropertyStr(ctx, new_target, "prototype"); if (JS_IsException(proto)) goto fail; obj = JS_NewObjectProtoClass(ctx, proto, js_point_class_id); JS_FreeValue(ctx, proto); if (JS_IsException(obj)) goto fail; JS_SetOpaque(obj, s); return obj; fail: js_free(ctx, s); JS_FreeValue(ctx, obj); return JS_EXCEPTION; } static JSValue js_point_get_xy(JSContext *ctx, JSValueConst this_val, int magic) { JSPointData *s = JS_GetOpaque2(ctx, this_val, js_point_class_id); if (!s) return JS_EXCEPTION; if (magic == 0) return JS_NewInt32(ctx, s->x); else return JS_NewInt32(ctx, s->y); } static JSValue js_point_set_xy(JSContext *ctx, JSValueConst this_val, JSValue val, int magic) { JSPointData *s = JS_GetOpaque2(ctx, this_val, js_point_class_id); int v; if (!s) return JS_EXCEPTION; if (JS_ToInt32(ctx, &v, val)) return JS_EXCEPTION; if (magic == 0) s->x = v; else s->y = v; return JS_UNDEFINED; } static JSValue js_point_norm(JSContext *ctx, JSValueConst this_val, int argc, JSValueConst *argv) { JSPointData *s = JS_GetOpaque2(ctx, this_val, js_point_class_id); if (!s) return JS_EXCEPTION; return JS_NewFloat64(ctx, sqrt((double)s->x * s->x + (double)s->y * s->y)); } static JSClassDef js_point_class = { "Point", .finalizer = js_point_finalizer, }; static const JSCFunctionListEntry js_point_proto_funcs[] = { JS_CGETSET_MAGIC_DEF("x", js_point_get_xy, js_point_set_xy, 0), JS_CGETSET_MAGIC_DEF("y", js_point_get_xy, js_point_set_xy, 1), JS_CFUNC_DEF("norm", 0, js_point_norm), }; /*PointEnumerable Class*/ static JSClassID js_pointEnumerable_class_id; static void js_pointEnumerable_finalizer(JSRuntime *rt, JSValue val) { return; } static JSValue js_pointEnumerable_ctor(JSContext *ctx, JSValueConst new_target, int argc, JSValueConst *argv) { JSValue obj = JS_UNDEFINED; JSValue proto; /* using new_target to get the prototype is necessary when the class is extended. */ proto = JS_GetPropertyStr(ctx, new_target, "prototype"); if (JS_IsException(proto)) goto fail; obj = JS_NewObjectProtoClass(ctx, proto, js_pointEnumerable_class_id); JS_FreeValue(ctx, proto); JS_DefinePropertyValueStr(ctx, obj, "x", argv[0], JS_PROP_C_W_E); JS_DefinePropertyValueStr(ctx, obj, "y", argv[1], JS_PROP_C_W_E); if (JS_IsException(obj)) goto fail; return obj; fail: JS_FreeValue(ctx, obj); return JS_EXCEPTION; } static JSValue js_pointEnumerable_norm(JSContext *ctx, JSValueConst this_val, int argc, JSValueConst *argv) { JSValue val; int ret, x, y; val = JS_GetPropertyStr(ctx, this_val, "x"); if (JS_IsException(val)) return JS_EXCEPTION; ret = JS_ToInt32(ctx, &x, val); JS_FreeValue(ctx, val); if (ret) return JS_EXCEPTION; val = JS_GetPropertyStr(ctx, this_val, "y"); if (JS_IsException(val)) return JS_EXCEPTION; ret = JS_ToInt32(ctx, &y, val); JS_FreeValue(ctx, val); if (ret) return JS_EXCEPTION; return JS_NewFloat64(ctx, sqrt((double)x * x + (double)y * y)); } static JSClassDef js_pointEnumerable_class = { "PointEnumerable", .finalizer = js_pointEnumerable_finalizer, }; static const JSCFunctionListEntry js_pointEnumerable_proto_funcs[] = { JS_CFUNC_DEF("norm", 0, js_pointEnumerable_norm), }; static int js_point_init(JSContext *ctx, JSModuleDef *m) { JSValue point_proto, point_class, pointEnumerable_proto, pointEnumerable_class; /* create the Point class */ JS_NewClassID(&js_point_class_id); JS_NewClass(JS_GetRuntime(ctx), js_point_class_id, &js_point_class); point_proto = JS_NewObject(ctx); JS_SetPropertyFunctionList(ctx, point_proto, js_point_proto_funcs, countof(js_point_proto_funcs)); point_class = JS_NewCFunction2(ctx, js_point_ctor, "Point", 2, JS_CFUNC_constructor, 0); /* set proto.constructor and ctor.prototype */ JS_SetConstructor(ctx, point_class, point_proto); JS_SetClassProto(ctx, js_point_class_id, point_proto); JS_SetModuleExport(ctx, m, "Point", point_class); /* create the PointEnumerable class */ JS_NewClassID(&js_pointEnumerable_class_id); JS_NewClass(JS_GetRuntime(ctx), js_pointEnumerable_class_id, &js_pointEnumerable_class); pointEnumerable_proto = JS_NewObject(ctx); JS_SetPropertyFunctionList(ctx, pointEnumerable_proto, js_pointEnumerable_proto_funcs, countof(js_pointEnumerable_proto_funcs)); pointEnumerable_class = JS_NewCFunction2(ctx, js_pointEnumerable_ctor, "PointEnumerable", 2, JS_CFUNC_constructor, 0); /* set proto.constructor and ctor.prototype */ JS_SetConstructor(ctx, pointEnumerable_class, pointEnumerable_proto); JS_SetClassProto(ctx, js_pointEnumerable_class_id, pointEnumerable_proto); JS_SetModuleExport(ctx, m, "PointEnumerable", pointEnumerable_class); return 0; } JSModuleDef *js_init_module(JSContext *ctx, const char *module_name) { JSModuleDef *m; m = JS_NewCModule(ctx, module_name, js_point_init); if (!m) return NULL; JS_AddModuleExport(ctx, m, "Point"); JS_AddModuleExport(ctx, m, "PointEnumerable"); return m; }