Fix add/remove event listeners in core:sys/wasm
#4951
Merged
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There were multiple issues here:
use_capture
parameter inremove_event_listener
/remove_window_event_listener
The key used to store the listener function in
listenerMap
was a javascriptObject
, when used as a key it was thus serialized to the string"[object Object]"
, meaning all listeners where effectively set to the same key when callingadd_event_listener
/add_window_event_listener
. Later on when callingremove_event_listener
/remove_window_event_listener
, it then tried to remove the incorrect one or none at all if there was a mix of the same event name registered on an element or the window.To fix I implemented a function
listener_key
in the javascript code which will generate a different key based on the event's:id
: dom element's id or 'window' (when event listener added to the window)name
: the event name (eg:click
), each event handler should be removed for the event name it was register on.data
: we can register events with different data, each one generate a new listener which has to be removed.callback
: same asdata
, if you register two similar handler but with two different callback, each one should be removed.useCapture
: this one is a bit tricky, but when you register an event handler in javascript, if you don't passuseCapture
, it defaults tofalse
. When you remove an handler, you have to pass the exact sameuseCapture
option you registered it with. In this case, we allowed to register an event with differentuseCapture
, but didn't allow to pass theuseCapture
when removing it. We always calledremoveEventListener
without theuseCapture
parameter which removed the handler properly only when it was registered withuseCapture=false
.I also switched the
WasmMemoryInterface.listenerMap
from{}
(javascript object) to anew Map()
, which is available everywhere nowadays.