1
0
mirror of https://github.com/rkd77/elinks.git synced 2025-01-03 14:57:44 -05:00

[mujs] document.dispatchEvent

This commit is contained in:
Witold Filipczyk 2024-05-26 13:32:41 +02:00
parent 488dc0a4aa
commit 7b2e8e7aca
3 changed files with 141 additions and 29 deletions

View File

@ -33,6 +33,7 @@
#include "ecmascript/libdom/parse.h" #include "ecmascript/libdom/parse.h"
#include "ecmascript/mujs.h" #include "ecmascript/mujs.h"
#include "ecmascript/mujs/collection.h" #include "ecmascript/mujs/collection.h"
#include "ecmascript/mujs/event.h"
#include "ecmascript/mujs/form.h" #include "ecmascript/mujs/form.h"
#include "ecmascript/mujs/forms.h" #include "ecmascript/mujs/forms.h"
#include "ecmascript/mujs/implementation.h" #include "ecmascript/mujs/implementation.h"
@ -63,8 +64,8 @@
//static xmlpp::Document emptyDoc; //static xmlpp::Document emptyDoc;
struct listener { struct el_listener {
LIST_HEAD_EL(struct listener); LIST_HEAD_EL(struct el_listener);
char *typ; char *typ;
const char *fun; const char *fun;
}; };
@ -72,11 +73,14 @@ struct listener {
struct mjs_document_private { struct mjs_document_private {
struct ecmascript_interpreter *interpreter; struct ecmascript_interpreter *interpreter;
const char *thisval; const char *thisval;
LIST_OF(struct listener) listeners; LIST_OF(struct el_listener) listeners;
dom_event_listener *listener;
void *node; void *node;
int ref_count; int ref_count;
}; };
static void document_event_handler(dom_event *event, void *pw);
static void * static void *
mjs_doc_getprivate(js_State *J, int idx) mjs_doc_getprivate(js_State *J, int idx)
{ {
@ -91,6 +95,8 @@ mjs_doc_getprivate(js_State *J, int idx)
static void mjs_push_doctype(js_State *J, void *node); static void mjs_push_doctype(js_State *J, void *node);
static void mjs_document_dispatchEvent(js_State *J);
static void static void
mjs_document_get_property_anchors(js_State *J) mjs_document_get_property_anchors(js_State *J)
{ {
@ -889,6 +895,12 @@ mjs_document_addEventListener(js_State *J)
js_pushnull(J); js_pushnull(J);
return; return;
} }
dom_document *doc = (dom_document *)doc_private->node;
if (!doc) {
js_pushnull(J);
return;
}
const char *str = js_tostring(J, 1); const char *str = js_tostring(J, 1);
if (!str) { if (!str) {
@ -904,25 +916,45 @@ mjs_document_addEventListener(js_State *J)
js_copy(J, 2); js_copy(J, 2);
const char *fun = js_ref(J); const char *fun = js_ref(J);
struct listener *l; struct el_listener *n = (struct el_listener *)mem_calloc(1, sizeof(*n));
foreach(l, doc_private->listeners) { if (!n) {
if (strcmp(l->typ, method)) { js_pushundefined(J);
continue; return;
} }
if (!strcmp(l->fun, fun)) { n->fun = fun;
mem_free(method); n->typ = method;
add_to_list_end(doc_private->listeners, n);
dom_exception exc;
if (doc_private->listener) {
dom_event_listener_ref(doc_private->listener);
} else {
exc = dom_event_listener_create(document_event_handler, doc_private, &doc_private->listener);
if (exc != DOM_NO_ERR || !doc_private->listener) {
js_pushundefined(J); js_pushundefined(J);
return; return;
} }
} }
struct listener *n = (struct listener *)mem_calloc(1, sizeof(*n)); dom_string *typ = NULL;
exc = dom_string_create(method, strlen(method), &typ);
if (n) { if (exc != DOM_NO_ERR || !typ) {
n->typ = method; goto ex;
n->fun = fun;
add_to_list_end(doc_private->listeners, n);
} }
exc = dom_event_target_add_event_listener(doc, typ, doc_private->listener, false);
if (exc == DOM_NO_ERR) {
dom_event_listener_ref(doc_private->listener);
}
ex:
if (typ) {
dom_string_unref(typ);
}
dom_event_listener_unref(doc_private->listener);
js_pushundefined(J); js_pushundefined(J);
} }
@ -938,6 +970,12 @@ mjs_document_removeEventListener(js_State *J)
js_pushnull(J); js_pushnull(J);
return; return;
} }
dom_document *doc = (dom_document *)doc_private->node;
if (!doc) {
js_pushnull(J);
return;
}
const char *str = js_tostring(J, 1); const char *str = js_tostring(J, 1);
if (!str) { if (!str) {
@ -952,18 +990,28 @@ mjs_document_removeEventListener(js_State *J)
} }
js_copy(J, 2); js_copy(J, 2);
const char *fun = js_ref(J); const char *fun = js_ref(J);
struct listener *l; struct el_listener *l;
foreach(l, doc_private->listeners) { foreach(l, doc_private->listeners) {
if (strcmp(l->typ, method)) { if (strcmp(l->typ, method)) {
continue; continue;
} }
if (!strcmp(l->fun, fun)) { if (l->fun == fun) {
dom_string *typ = NULL;
dom_exception exc = dom_string_create(method, strlen(method), &typ);
if (exc != DOM_NO_ERR || !typ) {
continue;
}
dom_event_target_remove_event_listener(doc, typ, doc_private->listener, false);
dom_string_unref(typ);
js_unref(J, l->fun);
del_from_list(l); del_from_list(l);
mem_free_set(&l->typ, NULL); mem_free_set(&l->typ, NULL);
if (l->fun) js_unref(J, l->fun);
mem_free(l); mem_free(l);
mem_free(method); mem_free(method);
js_pushundefined(J); js_pushundefined(J);
return; return;
} }
@ -1445,6 +1493,7 @@ mjs_document_init(js_State *J)
addmethod(J, "createDocumentFragment",mjs_document_createDocumentFragment, 0); addmethod(J, "createDocumentFragment",mjs_document_createDocumentFragment, 0);
addmethod(J, "createElement", mjs_document_createElement, 1); addmethod(J, "createElement", mjs_document_createElement, 1);
addmethod(J, "createTextNode", mjs_document_createTextNode, 1); addmethod(J, "createTextNode", mjs_document_createTextNode, 1);
addmethod(J, "dispatchEvent", mjs_document_dispatchEvent, 1);
addmethod(J, "write", mjs_document_write, 1); addmethod(J, "write", mjs_document_write, 1);
addmethod(J, "writeln", mjs_document_writeln, 1); addmethod(J, "writeln", mjs_document_writeln, 1);
addmethod(J, "replace", mjs_document_replace, 2); addmethod(J, "replace", mjs_document_replace, 2);
@ -1529,7 +1578,11 @@ mjs_doc_private_finalizer(js_State *J, void *priv)
struct mjs_document_private *doc_private = (struct mjs_document_private *)priv; struct mjs_document_private *doc_private = (struct mjs_document_private *)priv;
if (doc_private) { if (doc_private) {
struct listener *l; struct el_listener *l;
if (doc_private->listener) {
dom_event_listener_unref(doc_private->listener);
}
foreach(l, doc_private->listeners) { foreach(l, doc_private->listeners) {
mem_free_set(&l->typ, NULL); mem_free_set(&l->typ, NULL);
@ -1565,6 +1618,7 @@ mjs_push_document(js_State *J, void *doc)
addmethod(J, "createDocumentFragment",mjs_document_createDocumentFragment, 0); addmethod(J, "createDocumentFragment",mjs_document_createDocumentFragment, 0);
addmethod(J, "createElement", mjs_document_createElement, 1); addmethod(J, "createElement", mjs_document_createElement, 1);
addmethod(J, "createTextNode", mjs_document_createTextNode, 1); addmethod(J, "createTextNode", mjs_document_createTextNode, 1);
addmethod(J, "dispatchEvent", mjs_document_dispatchEvent, 1);
addmethod(J, "write", mjs_document_write, 1); addmethod(J, "write", mjs_document_write, 1);
addmethod(J, "writeln", mjs_document_writeln, 1); addmethod(J, "writeln", mjs_document_writeln, 1);
addmethod(J, "replace", mjs_document_replace, 2); addmethod(J, "replace", mjs_document_replace, 2);
@ -1612,3 +1666,66 @@ mjs_push_document(js_State *J, void *doc)
doc_private->ref_count = 1; doc_private->ref_count = 1;
doc_private->thisval = js_ref(J); doc_private->thisval = js_ref(J);
} }
static void
mjs_document_dispatchEvent(js_State *J)
{
#ifdef ECMASCRIPT_DEBUG
fprintf(stderr, "%s:%s\n", __FILE__, __FUNCTION__);
#endif
dom_document *doc = (dom_document *)mjs_doc_getprivate(J, 0);
if (!doc) {
js_pushboolean(J, 0);
return;
}
dom_event *event = (dom_event *)js_touserdata(J, 1, "event");
if (!event) {
js_pushboolean(J, 0);
return;
}
bool result = false;
dom_exception exc = dom_event_target_dispatch_event(doc, event, &result);
js_pushboolean(J, result);
}
static void
document_event_handler(dom_event *event, void *pw)
{
#ifdef ECMASCRIPT_DEBUG
fprintf(stderr, "%s:%s\n", __FILE__, __FUNCTION__);
#endif
struct mjs_document_private *doc_private = (struct mjs_document_private *)pw;
struct ecmascript_interpreter *interpreter = (struct ecmascript_interpreter *)doc_private->interpreter;
js_State *J = (js_State *)interpreter->backend_data;
dom_document *doc = (dom_document *)doc_private->node;
if (!event) {
return;
}
dom_string *typ = NULL;
dom_exception exc = dom_event_get_type(event, &typ);
if (exc != DOM_NO_ERR || !typ) {
return;
}
// interpreter->heartbeat = add_heartbeat(interpreter);
struct el_listener *l;
foreach(l, doc_private->listeners) {
if (strcmp(l->typ, dom_string_data(typ))) {
continue;
}
js_getregistry(J, l->fun); /* retrieve the js function from the registry */
js_getregistry(J, doc_private->thisval);
mjs_push_event(J, event);
js_pcall(J, 1);
js_pop(J, 1);
}
// done_heartbeat(interpreter->heartbeat);
check_for_rerender(interpreter, dom_string_data(typ));
dom_string_unref(typ);
}

View File

@ -38,22 +38,16 @@ mjs_event_finalizer(js_State *J, void *val)
} }
void void
mjs_push_event(js_State *J, char *type_) mjs_push_event(js_State *J, void *eve)
{ {
dom_event *event = NULL; dom_event *event = (dom_event *)eve;
dom_exception exc = dom_event_create(&event);
if (exc != DOM_NO_ERR) { if (!event) {
js_error(J, "out of memory"); js_error(J, "out of memory");
return; return;
} }
dom_event_ref(event);
if (type_) {
dom_string *typ = NULL;
exc = dom_string_create(type_, strlen(type_), &typ);
dom_event_init(event, typ, false, false);
if (typ) dom_string_unref(typ);
}
js_newobject(J); js_newobject(J);
{ {
js_newuserdata(J, "event", event, mjs_event_finalizer); js_newuserdata(J, "event", event, mjs_event_finalizer);

View File

@ -8,6 +8,7 @@ extern "C" {
#endif #endif
int mjs_event_init(js_State *J); int mjs_event_init(js_State *J);
void mjs_push_event(js_State *J, void *eve);
#ifdef __cplusplus #ifdef __cplusplus
} }