Skip to main content

script/dom/event/
eventtarget.rs

1/* This Source Code Form is subject to the terms of the Mozilla Public
2 * License, v. 2.0. If a copy of the MPL was not distributed with this
3 * file, You can obtain one at https://mozilla.org/MPL/2.0/. */
4
5use std::cell::RefCell;
6use std::collections::hash_map::Entry::{Occupied, Vacant};
7use std::default::Default;
8use std::ffi::CString;
9use std::mem;
10use std::ops::{Deref, DerefMut};
11use std::rc::Rc;
12use std::sync::LazyLock;
13
14use deny_public_fields::DenyPublicFields;
15use devtools_traits::EventListenerInfo;
16use dom_struct::dom_struct;
17use js::context::JSContext;
18use js::jsapi::{JS_GetFunctionObject, SupportUnscopables};
19use js::jsval::JSVal;
20use js::rust::wrappers2::CompileFunction;
21use js::rust::{CompileOptionsWrapper, HandleObject, transform_u16_to_source_text};
22use libc::c_char;
23use rustc_hash::{FxBuildHasher, FxHashSet};
24use script_bindings::cell::DomRefCell;
25use script_bindings::cformat;
26use script_bindings::reflector::{DomObject, Reflector, reflect_dom_object_with_proto};
27use servo_constellation_traits::ConstellationInterest;
28use servo_url::ServoUrl;
29use style::str::HTML_SPACE_CHARACTERS;
30use stylo_atoms::Atom;
31
32use crate::conversions::Convert;
33use crate::dom::abortsignal::{AbortAlgorithm, RemovableDomEventListener};
34use crate::dom::beforeunloadevent::BeforeUnloadEvent;
35use crate::dom::bindings::callback::{CallbackContainer, CallbackFunction, ExceptionHandling};
36use crate::dom::bindings::codegen::Bindings::BeforeUnloadEventBinding::BeforeUnloadEventMethods;
37use crate::dom::bindings::codegen::Bindings::ErrorEventBinding::ErrorEventMethods;
38use crate::dom::bindings::codegen::Bindings::EventBinding::EventMethods;
39use crate::dom::bindings::codegen::Bindings::EventHandlerBinding::{
40    EventHandlerNonNull, OnBeforeUnloadEventHandlerNonNull, OnErrorEventHandlerNonNull,
41};
42use crate::dom::bindings::codegen::Bindings::EventListenerBinding::EventListener;
43use crate::dom::bindings::codegen::Bindings::EventTargetBinding::{
44    AddEventListenerOptions, EventListenerOptions, EventTargetMethods,
45};
46use crate::dom::bindings::codegen::Bindings::NodeBinding::GetRootNodeOptions;
47use crate::dom::bindings::codegen::Bindings::NodeBinding::Node_Binding::NodeMethods;
48use crate::dom::bindings::codegen::Bindings::ShadowRootBinding::ShadowRoot_Binding::ShadowRootMethods;
49use crate::dom::bindings::codegen::Bindings::WindowBinding::WindowMethods;
50use crate::dom::bindings::codegen::GenericBindings::DocumentBinding::Document_Binding::DocumentMethods;
51use crate::dom::bindings::codegen::UnionTypes::{
52    AddEventListenerOptionsOrBoolean, EventListenerOptionsOrBoolean, EventOrString,
53};
54use crate::dom::bindings::error::{Error, Fallible, report_pending_exception};
55use crate::dom::bindings::inheritance::Castable;
56use crate::dom::bindings::reflector::DomGlobal;
57use crate::dom::bindings::root::{Dom, DomRoot};
58use crate::dom::bindings::str::DOMString;
59use crate::dom::bindings::trace::HashMapTracedValues;
60use crate::dom::csp::{CspReporting, InlineCheckType};
61use crate::dom::document::Document;
62use crate::dom::element::Element;
63use crate::dom::errorevent::ErrorEvent;
64use crate::dom::event::{Event, EventBubbles, EventCancelable, EventComposed};
65use crate::dom::globalscope::GlobalScope;
66use crate::dom::html::htmlformelement::FormControlElementHelpers;
67use crate::dom::indexeddb::idbdatabase::IDBDatabase;
68use crate::dom::indexeddb::idbrequest::IDBRequest;
69use crate::dom::indexeddb::idbtransaction::IDBTransaction;
70use crate::dom::node::{Node, NodeTraits};
71use crate::dom::shadowroot::ShadowRoot;
72use crate::dom::virtualmethods::VirtualMethods;
73use crate::dom::window::Window;
74use crate::dom::workerglobalscope::WorkerGlobalScope;
75use crate::realms::{enter_auto_realm, enter_realm};
76use crate::script_runtime::CanGc;
77
78/// <https://html.spec.whatwg.org/multipage/#event-handler-content-attributes>
79/// Generated from WebIDL definitions of EventHandler attributes on interfaces
80/// that inherit from Node.
81pub(crate) static CONTENT_EVENT_HANDLER_NAMES: LazyLock<FxHashSet<&str>> = LazyLock::new(|| {
82    FxHashSet::from_iter(include!(concat!(
83        env!("OUT_DIR"),
84        "/ContentEventHandlerNames.rs"
85    )))
86});
87
88#[derive(Clone, JSTraceable, MallocSizeOf, PartialEq)]
89#[expect(clippy::enum_variant_names)]
90pub(crate) enum CommonEventHandler {
91    EventHandler(#[conditional_malloc_size_of] Rc<EventHandlerNonNull>),
92
93    ErrorEventHandler(#[conditional_malloc_size_of] Rc<OnErrorEventHandlerNonNull>),
94
95    BeforeUnloadEventHandler(#[conditional_malloc_size_of] Rc<OnBeforeUnloadEventHandlerNonNull>),
96}
97
98impl CommonEventHandler {
99    fn parent(&self) -> &CallbackFunction<crate::DomTypeHolder> {
100        match *self {
101            CommonEventHandler::EventHandler(ref handler) => &handler.parent,
102            CommonEventHandler::ErrorEventHandler(ref handler) => &handler.parent,
103            CommonEventHandler::BeforeUnloadEventHandler(ref handler) => &handler.parent,
104        }
105    }
106}
107
108#[derive(Clone, Copy, Debug, JSTraceable, MallocSizeOf, PartialEq)]
109pub(crate) enum ListenerPhase {
110    Capturing,
111    Bubbling,
112}
113
114/// <https://html.spec.whatwg.org/multipage/#internal-raw-uncompiled-handler>
115#[derive(Clone, JSTraceable, MallocSizeOf, PartialEq)]
116struct InternalRawUncompiledHandler {
117    source: DOMString,
118    #[no_trace]
119    url: ServoUrl,
120    line: usize,
121}
122
123/// A representation of an event handler, either compiled or uncompiled raw source, or null.
124#[derive(Clone, JSTraceable, MallocSizeOf, PartialEq)]
125enum InlineEventListener {
126    Uncompiled(InternalRawUncompiledHandler),
127    Compiled(CommonEventHandler),
128    Null,
129}
130
131/// Get a compiled representation of this event handler, compiling it from its
132/// raw source if necessary.
133/// <https://html.spec.whatwg.org/multipage/#getting-the-current-value-of-the-event-handler>
134fn get_compiled_handler(
135    cx: &mut JSContext,
136    inline_listener: &RefCell<InlineEventListener>,
137    owner: &EventTarget,
138    ty: &Atom,
139) -> Option<CommonEventHandler> {
140    let listener = mem::replace(
141        &mut *inline_listener.borrow_mut(),
142        InlineEventListener::Null,
143    );
144    let compiled = match listener {
145        InlineEventListener::Null => None,
146        InlineEventListener::Uncompiled(handler) => {
147            owner.get_compiled_event_handler(cx, handler, ty)
148        },
149        InlineEventListener::Compiled(handler) => Some(handler),
150    };
151    if let Some(ref compiled) = compiled {
152        *inline_listener.borrow_mut() = InlineEventListener::Compiled(compiled.clone());
153    }
154    compiled
155}
156
157#[derive(Clone, JSTraceable, MallocSizeOf, PartialEq)]
158enum EventListenerType {
159    Additive(#[conditional_malloc_size_of] Rc<EventListener>),
160    Inline(RefCell<InlineEventListener>),
161}
162
163impl EventListenerType {
164    fn get_compiled_listener(
165        &self,
166        cx: &mut JSContext,
167        owner: &EventTarget,
168        ty: &Atom,
169    ) -> Option<CompiledEventListener> {
170        match *self {
171            EventListenerType::Inline(ref inline) => {
172                get_compiled_handler(cx, inline, owner, ty).map(CompiledEventListener::Handler)
173            },
174            EventListenerType::Additive(ref listener) => {
175                Some(CompiledEventListener::Listener(listener.clone()))
176            },
177        }
178    }
179}
180
181/// A representation of an EventListener/EventHandler object that has previously
182/// been compiled successfully, if applicable.
183pub(crate) enum CompiledEventListener {
184    Listener(Rc<EventListener>),
185    Handler(CommonEventHandler),
186}
187
188impl CompiledEventListener {
189    #[expect(unsafe_code)]
190    pub(crate) fn associated_global(&self) -> DomRoot<GlobalScope> {
191        let obj = match self {
192            CompiledEventListener::Listener(listener) => listener.callback(),
193            CompiledEventListener::Handler(CommonEventHandler::EventHandler(handler)) => {
194                handler.callback()
195            },
196            CompiledEventListener::Handler(CommonEventHandler::ErrorEventHandler(handler)) => {
197                handler.callback()
198            },
199            CompiledEventListener::Handler(CommonEventHandler::BeforeUnloadEventHandler(
200                handler,
201            )) => handler.callback(),
202        };
203        unsafe { GlobalScope::from_object(obj) }
204    }
205
206    // https://html.spec.whatwg.org/multipage/#the-event-handler-processing-algorithm
207    pub(crate) fn call_or_handle_event(
208        &self,
209        cx: &mut JSContext,
210        object: &EventTarget,
211        event: &Event,
212        exception_handle: ExceptionHandling,
213    ) -> Fallible<()> {
214        // Step 3
215        match *self {
216            CompiledEventListener::Listener(ref listener) => {
217                listener.HandleEvent_(cx, object, event, exception_handle)
218            },
219            CompiledEventListener::Handler(ref handler) => {
220                match *handler {
221                    CommonEventHandler::ErrorEventHandler(ref handler) => {
222                        if let Some(event) = event.downcast::<ErrorEvent>() &&
223                            (object.is::<Window>() || object.is::<WorkerGlobalScope>())
224                        {
225                            rooted!(&in(cx) let mut error: JSVal);
226                            event.Error(cx.into(), error.handle_mut());
227                            rooted!(&in(cx) let mut rooted_return_value: JSVal);
228                            let return_value = handler.Call_(
229                                cx,
230                                object,
231                                EventOrString::String(event.Message()),
232                                Some(event.Filename()),
233                                Some(event.Lineno()),
234                                Some(event.Colno()),
235                                Some(error.handle()),
236                                rooted_return_value.handle_mut(),
237                                exception_handle,
238                            );
239                            // Step 4
240                            if let Ok(()) = return_value &&
241                                rooted_return_value.handle().is_boolean() &&
242                                rooted_return_value.handle().to_boolean()
243                            {
244                                event.upcast::<Event>().PreventDefault();
245                            }
246                            return return_value;
247                        }
248
249                        rooted!(&in(cx) let mut rooted_return_value: JSVal);
250                        handler.Call_(
251                            cx,
252                            object,
253                            EventOrString::Event(DomRoot::from_ref(event)),
254                            None,
255                            None,
256                            None,
257                            None,
258                            rooted_return_value.handle_mut(),
259                            exception_handle,
260                        )
261                    },
262
263                    CommonEventHandler::BeforeUnloadEventHandler(ref handler) => {
264                        if let Some(event) = event.downcast::<BeforeUnloadEvent>() {
265                            // Step 5
266                            match handler.Call_(
267                                cx,
268                                object,
269                                event.upcast::<Event>(),
270                                exception_handle,
271                            ) {
272                                Ok(value) => {
273                                    let rv = event.ReturnValue();
274                                    if let Some(v) = value {
275                                        if rv.is_empty() {
276                                            event.SetReturnValue(v);
277                                        }
278                                        event.upcast::<Event>().PreventDefault();
279                                    }
280                                    Ok(())
281                                },
282                                Err(err) => Err(err),
283                            }
284                        } else {
285                            // Step 5, "Otherwise" clause
286                            handler
287                                .Call_(cx, object, event.upcast::<Event>(), exception_handle)
288                                .map(|_| ())
289                        }
290                    },
291
292                    CommonEventHandler::EventHandler(ref handler) => {
293                        rooted!(&in(cx) let mut rooted_return_value: JSVal);
294                        match handler.Call_(
295                            cx,
296                            object,
297                            event,
298                            rooted_return_value.handle_mut(),
299                            exception_handle,
300                        ) {
301                            Ok(()) => {
302                                let value = rooted_return_value.handle();
303
304                                // Step 5
305                                let should_cancel = value.is_boolean() && !value.to_boolean();
306
307                                if should_cancel {
308                                    // FIXME: spec says to set the cancelled flag directly
309                                    // here, not just to prevent default;
310                                    // can that ever make a difference?
311                                    event.PreventDefault();
312                                }
313                                Ok(())
314                            },
315                            Err(err) => Err(err),
316                        }
317                    },
318                }
319            },
320        }
321    }
322}
323
324// https://dom.spec.whatwg.org/#concept-event-listener
325// (as distinct from https://dom.spec.whatwg.org/#callbackdef-eventlistener)
326#[derive(Clone, DenyPublicFields, JSTraceable, MallocSizeOf)]
327/// A listener in a collection of event listeners.
328pub(crate) struct EventListenerEntry {
329    phase: ListenerPhase,
330    listener: EventListenerType,
331    once: bool,
332    passive: bool,
333    removed: bool,
334}
335
336impl EventListenerEntry {
337    pub(crate) fn phase(&self) -> ListenerPhase {
338        self.phase
339    }
340
341    pub(crate) fn once(&self) -> bool {
342        self.once
343    }
344
345    pub(crate) fn removed(&self) -> bool {
346        self.removed
347    }
348
349    /// <https://html.spec.whatwg.org/multipage/#getting-the-current-value-of-the-event-handler>
350    pub(crate) fn get_compiled_listener(
351        &self,
352        cx: &mut JSContext,
353        owner: &EventTarget,
354        ty: &Atom,
355    ) -> Option<CompiledEventListener> {
356        self.listener.get_compiled_listener(cx, owner, ty)
357    }
358}
359
360impl std::cmp::PartialEq for EventListenerEntry {
361    fn eq(&self, other: &Self) -> bool {
362        self.phase == other.phase && self.listener == other.listener
363    }
364}
365
366#[derive(Clone, JSTraceable, MallocSizeOf)]
367/// A mix of potentially uncompiled and compiled event listeners of the same type.
368pub(crate) struct EventListeners(
369    #[conditional_malloc_size_of] Vec<Rc<RefCell<EventListenerEntry>>>,
370);
371
372impl Deref for EventListeners {
373    type Target = Vec<Rc<RefCell<EventListenerEntry>>>;
374    fn deref(&self) -> &Vec<Rc<RefCell<EventListenerEntry>>> {
375        &self.0
376    }
377}
378
379impl DerefMut for EventListeners {
380    fn deref_mut(&mut self) -> &mut Vec<Rc<RefCell<EventListenerEntry>>> {
381        &mut self.0
382    }
383}
384
385impl EventListeners {
386    /// <https://html.spec.whatwg.org/multipage/#getting-the-current-value-of-the-event-handler>
387    fn get_inline_listener(
388        &self,
389        cx: &mut JSContext,
390        owner: &EventTarget,
391        ty: &Atom,
392    ) -> Option<CommonEventHandler> {
393        for entry in &self.0 {
394            if let EventListenerType::Inline(ref inline) = entry.borrow().listener {
395                // Step 1.1-1.8 and Step 2
396                return get_compiled_handler(cx, inline, owner, ty);
397            }
398        }
399
400        // Step 2
401        None
402    }
403
404    fn has_listeners(&self) -> bool {
405        !self.0.is_empty()
406    }
407}
408
409#[dom_struct]
410pub struct EventTarget {
411    reflector_: Reflector,
412    handlers: DomRefCell<HashMapTracedValues<Atom, EventListeners, FxBuildHasher>>,
413}
414
415impl EventTarget {
416    pub(crate) fn new_inherited() -> EventTarget {
417        EventTarget {
418            reflector_: Reflector::new(),
419            handlers: DomRefCell::new(Default::default()),
420        }
421    }
422
423    fn new(
424        global: &GlobalScope,
425        proto: Option<HandleObject>,
426        can_gc: CanGc,
427    ) -> DomRoot<EventTarget> {
428        reflect_dom_object_with_proto(
429            Box::new(EventTarget::new_inherited()),
430            global,
431            proto,
432            can_gc,
433        )
434    }
435
436    /// Returns the [`ConstellationInterest`] associated with a given event type
437    /// on this specific target, if any. The mapping depends on the concrete type
438    /// of the EventTarget since the same event name can be used in multiple contexts.
439    fn interest_for_event_type(&self, ty: &Atom) -> Option<ConstellationInterest> {
440        if self.is::<Window>() && *ty == atom!("storage") {
441            return Some(ConstellationInterest::StorageEvent);
442        }
443        None
444    }
445
446    /// Notify the global about a listener being added for a given event type.
447    fn notify_listener_added(&self, ty: &Atom) {
448        if let Some(interest) = self.interest_for_event_type(ty) {
449            self.global().register_interest(interest);
450        }
451    }
452
453    /// Notify the global about a listener being removed for a given event type.
454    fn notify_listener_removed(&self, ty: &Atom) {
455        if let Some(interest) = self.interest_for_event_type(ty) {
456            self.global().unregister_interest(interest);
457        }
458    }
459
460    /// Determine if there are any listeners for a given event type.
461    /// See <https://github.com/whatwg/dom/issues/453>.
462    pub(crate) fn has_listeners_for(&self, type_: &Atom) -> bool {
463        match self.handlers.borrow().get(type_) {
464            Some(listeners) => listeners.has_listeners(),
465            None => false,
466        }
467    }
468
469    pub(crate) fn get_listeners_for(&self, type_: &Atom) -> EventListeners {
470        self.handlers
471            .borrow()
472            .get(type_)
473            .map_or(EventListeners(vec![]), |listeners| listeners.clone())
474    }
475
476    pub(crate) fn remove_all_listeners(&self) {
477        let mut handlers = self.handlers.borrow_mut();
478        for (ty, entries) in handlers.iter() {
479            entries
480                .iter()
481                .for_each(|entry| entry.borrow_mut().removed = true);
482            self.notify_listener_removed(ty);
483        }
484
485        *handlers = Default::default();
486    }
487
488    /// <https://dom.spec.whatwg.org/#default-passive-value>
489    fn default_passive_value(&self, ty: &Atom) -> bool {
490        // Return true if all of the following are true:
491        let event_type = ty.to_ascii_lowercase();
492
493        // type is one of "touchstart", "touchmove", "wheel", or "mousewheel"
494        let matches_event_type = matches!(
495            event_type.trim_matches(HTML_SPACE_CHARACTERS),
496            "touchstart" | "touchmove" | "wheel" | "mousewheel"
497        );
498
499        if !matches_event_type {
500            return false;
501        }
502
503        // eventTarget is a Window object
504        if self.is::<Window>() {
505            return true;
506        }
507
508        // or ...
509        if let Some(node) = self.downcast::<Node>() {
510            let node_document = node.owner_document();
511            let event_target = self.upcast::<EventTarget>();
512
513            // is a node whose node document is eventTarget
514            return event_target == node_document.upcast::<EventTarget>()
515                // or is a node whose node document’s document element is eventTarget
516                || node_document.GetDocumentElement().is_some_and(|n| n.upcast::<EventTarget>() == event_target)
517                // or is a node whose node document’s body element is eventTarget
518                || node_document.GetBody().is_some_and(|n| n.upcast::<EventTarget>() == event_target);
519        }
520
521        false
522    }
523
524    /// <https://html.spec.whatwg.org/multipage/#event-handler-attributes:event-handlers-11>
525    fn set_inline_event_listener(&self, ty: Atom, listener: Option<InlineEventListener>) {
526        let mut handlers = self.handlers.borrow_mut();
527        let entries = match handlers.entry(ty.clone()) {
528            Occupied(entry) => entry.into_mut(),
529            Vacant(entry) => entry.insert(EventListeners(vec![])),
530        };
531
532        let idx = entries
533            .iter()
534            .position(|entry| matches!(entry.borrow().listener, EventListenerType::Inline(_)));
535
536        match idx {
537            Some(idx) => match listener {
538                // Replace if there's something to replace with,
539                // but remove entirely if there isn't.
540                Some(listener) => {
541                    entries[idx].borrow_mut().listener = EventListenerType::Inline(listener.into());
542                },
543                None => {
544                    entries.remove(idx).borrow_mut().removed = true;
545                    self.notify_listener_removed(&ty);
546                },
547            },
548            None => {
549                if let Some(listener) = listener {
550                    entries.push(Rc::new(RefCell::new(EventListenerEntry {
551                        phase: ListenerPhase::Bubbling,
552                        listener: EventListenerType::Inline(listener.into()),
553                        once: false,
554                        passive: self.default_passive_value(&ty),
555                        removed: false,
556                    })));
557                    self.notify_listener_added(&ty)
558                }
559            },
560        }
561    }
562
563    pub(crate) fn remove_listener(&self, ty: &Atom, entry: &Rc<RefCell<EventListenerEntry>>) {
564        let mut handlers = self.handlers.borrow_mut();
565
566        if let Some(entries) = handlers.get_mut(ty) &&
567            let Some(position) = entries.iter().position(|e| *e == *entry)
568        {
569            entries.remove(position).borrow_mut().removed = true;
570            self.notify_listener_removed(ty);
571        }
572    }
573
574    /// Determines the `passive` attribute of an associated event listener
575    pub(crate) fn is_passive(&self, listener: &Rc<RefCell<EventListenerEntry>>) -> bool {
576        listener.borrow().passive
577    }
578
579    fn get_inline_event_listener(
580        &self,
581        cx: &mut JSContext,
582        ty: &Atom,
583    ) -> Option<CommonEventHandler> {
584        let handlers = self.handlers.borrow();
585        handlers
586            .get(ty)
587            .and_then(|entry| entry.get_inline_listener(cx, self, ty))
588    }
589
590    /// Store the raw uncompiled event handler for on-demand compilation later.
591    /// <https://html.spec.whatwg.org/multipage/#event-handler-attributes:event-handler-content-attributes-3>
592    pub(crate) fn set_event_handler_uncompiled(
593        &self,
594        url: ServoUrl,
595        line: usize,
596        ty: &str,
597        source: &str,
598    ) {
599        if let Some(element) = self.downcast::<Element>() {
600            let doc = element.owner_document();
601            let global = &doc.global();
602            if global
603                .get_csp_list()
604                .should_elements_inline_type_behavior_be_blocked(
605                    global,
606                    element.upcast(),
607                    InlineCheckType::ScriptAttribute,
608                    source,
609                    line as u32,
610                )
611            {
612                return;
613            }
614        };
615
616        let handler = InternalRawUncompiledHandler {
617            source: DOMString::from(source),
618            line,
619            url,
620        };
621        self.set_inline_event_listener(
622            Atom::from(ty),
623            Some(InlineEventListener::Uncompiled(handler)),
624        );
625    }
626
627    // https://html.spec.whatwg.org/multipage/#getting-the-current-value-of-the-event-handler
628    // step 3
629    #[expect(unsafe_code)]
630    fn get_compiled_event_handler(
631        &self,
632        cx: &mut JSContext,
633        handler: InternalRawUncompiledHandler,
634        ty: &Atom,
635    ) -> Option<CommonEventHandler> {
636        // Step 3.1
637        let element = self.downcast::<Element>();
638        let document = match element {
639            Some(element) => element.owner_document(),
640            None => self.downcast::<Window>().unwrap().Document(),
641        };
642
643        // Step 3.2
644        if !document.scripting_enabled() {
645            return None;
646        }
647
648        // Step 3.3
649        let body: Vec<u16> = handler.source.str().encode_utf16().collect();
650
651        // Step 3.4 is handler.line
652
653        // Step 3.5
654        let form_owner = element
655            .and_then(|e| e.as_maybe_form_control())
656            .and_then(|f| f.form_owner());
657
658        // Step 3.6 TODO: settings objects not implemented
659
660        // Step 3.7 is written as though we call the parser separately
661        // from the compiler; since we just call CompileFunction with
662        // source text, we handle parse errors later
663
664        // Step 3.8 TODO: settings objects not implemented
665        let window = document.window();
666        let _ac = enter_realm(window);
667
668        // Step 3.9
669
670        let name = CString::new(format!("on{}", &**ty)).unwrap();
671
672        // Step 3.9, subsection ParameterList
673        const ARG_NAMES: &[*const c_char] = &[c"event".as_ptr()];
674        const ERROR_ARG_NAMES: &[*const c_char] = &[
675            c"event".as_ptr(),
676            c"source".as_ptr(),
677            c"lineno".as_ptr(),
678            c"colno".as_ptr(),
679            c"error".as_ptr(),
680        ];
681        let is_error = ty == &atom!("error") && self.is::<Window>();
682        let args = if is_error { ERROR_ARG_NAMES } else { ARG_NAMES };
683
684        let url = cformat!("{}", handler.url);
685        let options =
686            unsafe { CompileOptionsWrapper::new_raw(cx.raw_cx(), url, handler.line as u32) };
687
688        // Step 3.9, subsection Scope steps 1-6
689        let scopechain =
690            js::rust::EnvironmentChain::new(unsafe { cx.raw_cx() }, SupportUnscopables::Yes);
691
692        if let Some(element) = element {
693            scopechain.append(document.reflector().get_jsobject().get());
694            if let Some(form_owner) = form_owner {
695                scopechain.append(form_owner.reflector().get_jsobject().get());
696            }
697            scopechain.append(element.reflector().get_jsobject().get());
698        }
699
700        rooted!(&in(cx) let mut handler = unsafe {
701            CompileFunction(
702                cx,
703                scopechain.get(),
704                options.ptr,
705                name.as_ptr(),
706                args.len() as u32,
707                args.as_ptr(),
708                &mut transform_u16_to_source_text(&body),
709            )
710        });
711        if handler.get().is_null() {
712            // Step 3.7
713            let mut realm = enter_auto_realm(cx, self);
714            report_pending_exception(&mut realm.current_realm());
715            return None;
716        }
717
718        // Step 3.10 happens when we drop _ac
719
720        // TODO Step 3.11
721
722        // Step 3.12
723        let funobj = unsafe { JS_GetFunctionObject(handler.get()) };
724        assert!(!funobj.is_null());
725        // Step 1.14
726        if is_error {
727            Some(CommonEventHandler::ErrorEventHandler(unsafe {
728                OnErrorEventHandlerNonNull::new(cx.into(), funobj)
729            }))
730        } else if ty == &atom!("beforeunload") {
731            Some(CommonEventHandler::BeforeUnloadEventHandler(unsafe {
732                OnBeforeUnloadEventHandlerNonNull::new(cx.into(), funobj)
733            }))
734        } else {
735            Some(CommonEventHandler::EventHandler(unsafe {
736                EventHandlerNonNull::new(cx.into(), funobj)
737            }))
738        }
739    }
740
741    #[expect(unsafe_code)]
742    pub(crate) fn set_event_handler_common<T: CallbackContainer<crate::DomTypeHolder>>(
743        &self,
744        cx: &mut JSContext,
745        ty: &str,
746        listener: Option<Rc<T>>,
747    ) {
748        let event_listener = listener.map(|listener| {
749            InlineEventListener::Compiled(CommonEventHandler::EventHandler(unsafe {
750                EventHandlerNonNull::new(cx.into(), listener.callback())
751            }))
752        });
753        self.set_inline_event_listener(Atom::from(ty), event_listener);
754    }
755
756    #[expect(unsafe_code)]
757    pub(crate) fn set_error_event_handler<T: CallbackContainer<crate::DomTypeHolder>>(
758        &self,
759        cx: &mut JSContext,
760        ty: &str,
761        listener: Option<Rc<T>>,
762    ) {
763        let event_listener = listener.map(|listener| {
764            InlineEventListener::Compiled(CommonEventHandler::ErrorEventHandler(unsafe {
765                OnErrorEventHandlerNonNull::new(cx.into(), listener.callback())
766            }))
767        });
768        self.set_inline_event_listener(Atom::from(ty), event_listener);
769    }
770
771    #[expect(unsafe_code)]
772    pub(crate) fn set_beforeunload_event_handler<T: CallbackContainer<crate::DomTypeHolder>>(
773        &self,
774        cx: &mut JSContext,
775        ty: &str,
776        listener: Option<Rc<T>>,
777    ) {
778        let event_listener = listener.map(|listener| {
779            InlineEventListener::Compiled(CommonEventHandler::BeforeUnloadEventHandler(unsafe {
780                OnBeforeUnloadEventHandlerNonNull::new(cx.into(), listener.callback())
781            }))
782        });
783        self.set_inline_event_listener(Atom::from(ty), event_listener);
784    }
785
786    #[expect(unsafe_code)]
787    pub(crate) fn get_event_handler_common<T: CallbackContainer<crate::DomTypeHolder>>(
788        &self,
789        cx: &mut JSContext,
790        ty: &str,
791    ) -> Option<Rc<T>> {
792        let listener = self.get_inline_event_listener(cx, &Atom::from(ty));
793        unsafe {
794            listener.map(|listener| {
795                CallbackContainer::new(cx.into(), listener.parent().callback_holder().get())
796            })
797        }
798    }
799
800    pub(crate) fn has_handlers(&self) -> bool {
801        !self.handlers.borrow().is_empty()
802    }
803
804    // https://dom.spec.whatwg.org/#concept-event-fire
805    pub(crate) fn fire_event(&self, cx: &mut js::context::JSContext, name: Atom) -> bool {
806        self.fire_event_with_params(
807            cx,
808            name,
809            EventBubbles::DoesNotBubble,
810            EventCancelable::NotCancelable,
811            EventComposed::NotComposed,
812        )
813    }
814
815    // https://dom.spec.whatwg.org/#concept-event-fire
816    pub(crate) fn fire_bubbling_event(&self, cx: &mut js::context::JSContext, name: Atom) -> bool {
817        self.fire_event_with_params(
818            cx,
819            name,
820            EventBubbles::Bubbles,
821            EventCancelable::NotCancelable,
822            EventComposed::NotComposed,
823        )
824    }
825
826    // https://dom.spec.whatwg.org/#concept-event-fire
827    pub(crate) fn fire_cancelable_event(
828        &self,
829        cx: &mut js::context::JSContext,
830        name: Atom,
831    ) -> bool {
832        self.fire_event_with_params(
833            cx,
834            name,
835            EventBubbles::DoesNotBubble,
836            EventCancelable::Cancelable,
837            EventComposed::NotComposed,
838        )
839    }
840
841    // https://dom.spec.whatwg.org/#concept-event-fire
842    pub(crate) fn fire_bubbling_cancelable_event(
843        &self,
844        cx: &mut js::context::JSContext,
845        name: Atom,
846    ) -> bool {
847        self.fire_event_with_params(
848            cx,
849            name,
850            EventBubbles::Bubbles,
851            EventCancelable::Cancelable,
852            EventComposed::NotComposed,
853        )
854    }
855
856    /// <https://dom.spec.whatwg.org/#concept-event-fire>
857    pub(crate) fn fire_event_with_params(
858        &self,
859        cx: &mut js::context::JSContext,
860        name: Atom,
861        bubbles: EventBubbles,
862        cancelable: EventCancelable,
863        composed: EventComposed,
864    ) -> bool {
865        let event = Event::new(
866            &self.global(),
867            name,
868            bubbles,
869            cancelable,
870            CanGc::from_cx(cx),
871        );
872        event.set_composed(composed.into());
873        event.fire(cx, self)
874    }
875
876    /// <https://dom.spec.whatwg.org/#dom-eventtarget-addeventlistener>
877    /// and <https://dom.spec.whatwg.org/#add-an-event-listener>
878    pub(crate) fn add_event_listener(
879        &self,
880        ty: DOMString,
881        listener: Option<Rc<EventListener>>,
882        options: AddEventListenerOptions,
883    ) {
884        if let Some(signal) = options.signal.as_ref() {
885            // Step 2. If listener’s signal is not null and is aborted, then return.
886            if signal.aborted() {
887                return;
888            }
889            // Step 6. If listener’s signal is not null, then add the following abort steps to it:
890            signal.add(&AbortAlgorithm::DomEventListener(
891                RemovableDomEventListener {
892                    event_target: Dom::from_ref(self),
893                    ty: ty.clone(),
894                    listener: listener.clone(),
895                    options: options.parent.clone(),
896                },
897            ));
898        }
899        // Step 3. If listener’s callback is null, then return.
900        let listener = match listener {
901            Some(l) => l,
902            None => return,
903        };
904        let ty = Atom::from(ty);
905        let mut handlers = self.handlers.borrow_mut();
906        let entries = match handlers.entry(ty.clone()) {
907            Occupied(entry) => entry.into_mut(),
908            Vacant(entry) => entry.insert(EventListeners(vec![])),
909        };
910
911        let phase = if options.parent.capture {
912            ListenerPhase::Capturing
913        } else {
914            ListenerPhase::Bubbling
915        };
916        // Step 4. If listener’s passive is null, then set it to the default passive value given listener’s type and eventTarget.
917        let new_entry = Rc::new(RefCell::new(EventListenerEntry {
918            phase,
919            listener: EventListenerType::Additive(listener),
920            once: options.once,
921            passive: options.passive.unwrap_or(self.default_passive_value(&ty)),
922            removed: false,
923        }));
924
925        // Step 5. If eventTarget’s event listener list does not contain
926        // an event listener whose type is listener’s type, callback is listener’s callback,
927        // and capture is listener’s capture, then append listener to eventTarget’s event listener list.
928        if !entries.contains(&new_entry) {
929            entries.push(new_entry);
930            self.notify_listener_added(&ty);
931        }
932    }
933
934    /// <https://dom.spec.whatwg.org/#dom-eventtarget-removeeventlistener>
935    /// and <https://dom.spec.whatwg.org/#remove-an-event-listener>
936    pub(crate) fn remove_event_listener(
937        &self,
938        ty: DOMString,
939        listener: &Option<Rc<EventListener>>,
940        options: &EventListenerOptions,
941    ) {
942        let Some(listener) = listener else {
943            return;
944        };
945        let ty_atom = Atom::from(ty);
946        let mut handlers = self.handlers.borrow_mut();
947        if let Some(entries) = handlers.get_mut(&ty_atom) {
948            let phase = if options.capture {
949                ListenerPhase::Capturing
950            } else {
951                ListenerPhase::Bubbling
952            };
953            let listener_type = EventListenerType::Additive(listener.clone());
954            if let Some(position) = entries
955                .iter()
956                .position(|e| e.borrow().listener == listener_type && e.borrow().phase == phase)
957            {
958                // Step 2. Set listener’s removed to true and remove listener from eventTarget’s event listener list.
959                entries.remove(position).borrow_mut().removed = true;
960                self.notify_listener_removed(&ty_atom);
961            }
962        }
963    }
964
965    /// <https://dom.spec.whatwg.org/#get-the-parent>
966    pub(crate) fn get_the_parent(&self, event: &Event) -> Option<DomRoot<EventTarget>> {
967        if let Some(document) = self.downcast::<Document>() {
968            if event.type_() == atom!("load") || !document.has_browsing_context() {
969                return None;
970            } else {
971                return Some(DomRoot::from_ref(document.window().upcast::<EventTarget>()));
972            }
973        }
974
975        if let Some(shadow_root) = self.downcast::<ShadowRoot>() {
976            if event.should_pass_shadow_boundary(shadow_root) {
977                let host = shadow_root.Host();
978                return Some(DomRoot::from_ref(host.upcast::<EventTarget>()));
979            } else {
980                return None;
981            }
982        }
983
984        if let Some(node) = self.downcast::<Node>() {
985            // > A node’s get the parent algorithm, given an event, returns the node’s assigned slot,
986            // > if node is assigned; otherwise node’s parent.
987            return node.assigned_slot().map(DomRoot::upcast).or_else(|| {
988                node.GetParentNode()
989                    .map(|parent| DomRoot::from_ref(parent.upcast::<EventTarget>()))
990            });
991        }
992
993        // https://w3c.github.io/IndexedDB/#events
994        // The get the parent algorithm for an IDBRequest returns the request's transaction.
995        if let Some(request) = self.downcast::<IDBRequest>() {
996            return request
997                .transaction()
998                .map(|tx| DomRoot::from_ref(tx.upcast::<EventTarget>()));
999        }
1000
1001        // The get the parent algorithm for an IDBTransaction returns the transaction's connection.
1002        if let Some(transaction) = self.downcast::<IDBTransaction>() {
1003            return Some(DomRoot::from_ref(
1004                transaction.get_db().upcast::<EventTarget>(),
1005            ));
1006        }
1007
1008        // The get the parent algorithm for an IDBDatabase returns null.
1009        if self.downcast::<IDBDatabase>().is_some() {
1010            return None;
1011        }
1012
1013        None
1014    }
1015
1016    // FIXME: This algorithm operates on "objects", which may not be event targets.
1017    // All our current use-cases only work on event targets, but this might change in the future
1018    /// <https://dom.spec.whatwg.org/#retarget>
1019    pub(crate) fn retarget(&self, b: &Self) -> DomRoot<EventTarget> {
1020        // To retarget an object A against an object B, repeat these steps until they return an object:
1021        let mut a = DomRoot::from_ref(self);
1022        loop {
1023            // Step 1. If one of the following is true
1024            // * A is not a node
1025            // * A’s root is not a shadow root
1026            // * B is a node and A’s root is a shadow-including inclusive ancestor of B
1027            // then return A.
1028            let Some(a_node) = a.downcast::<Node>() else {
1029                return a;
1030            };
1031            let a_root = a_node.GetRootNode(&GetRootNodeOptions::empty());
1032            if !a_root.is::<ShadowRoot>() {
1033                return a;
1034            }
1035            if let Some(b_node) = b.downcast::<Node>() &&
1036                a_root.is_shadow_including_inclusive_ancestor_of(b_node)
1037            {
1038                return a;
1039            }
1040
1041            // Step 2. Set A to A’s root’s host.
1042            a = DomRoot::from_ref(
1043                a_root
1044                    .downcast::<ShadowRoot>()
1045                    .unwrap()
1046                    .Host()
1047                    .upcast::<EventTarget>(),
1048            );
1049        }
1050    }
1051
1052    /// <https://html.spec.whatwg.org/multipage/#event-handler-content-attributes>
1053    pub(crate) fn is_content_event_handler(name: &str) -> bool {
1054        CONTENT_EVENT_HANDLER_NAMES.contains(&name)
1055    }
1056
1057    pub(crate) fn summarize_event_listeners_for_devtools(&self) -> Vec<EventListenerInfo> {
1058        let handlers = self.handlers.borrow();
1059        let mut listener_infos = Vec::with_capacity(handlers.0.len());
1060        for (event_type, event_listeners) in &handlers.0 {
1061            for event_listener in event_listeners.iter() {
1062                let event_listener_entry = event_listener.borrow();
1063                listener_infos.push(EventListenerInfo {
1064                    event_type: event_type.to_string(),
1065                    capturing: event_listener_entry.phase() == ListenerPhase::Capturing,
1066                });
1067            }
1068        }
1069
1070        listener_infos
1071    }
1072}
1073
1074impl EventTargetMethods<crate::DomTypeHolder> for EventTarget {
1075    /// <https://dom.spec.whatwg.org/#dom-eventtarget-eventtarget>
1076    fn Constructor(
1077        global: &GlobalScope,
1078        proto: Option<HandleObject>,
1079        can_gc: CanGc,
1080    ) -> Fallible<DomRoot<EventTarget>> {
1081        Ok(EventTarget::new(global, proto, can_gc))
1082    }
1083
1084    /// <https://dom.spec.whatwg.org/#dom-eventtarget-addeventlistener>
1085    fn AddEventListener(
1086        &self,
1087        ty: DOMString,
1088        listener: Option<Rc<EventListener>>,
1089        options: AddEventListenerOptionsOrBoolean,
1090    ) {
1091        self.add_event_listener(ty, listener, options.convert())
1092    }
1093
1094    /// <https://dom.spec.whatwg.org/#dom-eventtarget-removeeventlistener>
1095    fn RemoveEventListener(
1096        &self,
1097        ty: DOMString,
1098        listener: Option<Rc<EventListener>>,
1099        options: EventListenerOptionsOrBoolean,
1100    ) {
1101        self.remove_event_listener(ty, &listener, &options.convert())
1102    }
1103
1104    /// <https://dom.spec.whatwg.org/#dom-eventtarget-dispatchevent>
1105    fn DispatchEvent(&self, cx: &mut JSContext, event: &Event) -> Fallible<bool> {
1106        if event.dispatching() || !event.initialized() {
1107            return Err(Error::InvalidState(None));
1108        }
1109        event.set_trusted(false);
1110        Ok(event.dispatch(cx, self, false))
1111    }
1112}
1113
1114impl VirtualMethods for EventTarget {
1115    fn super_type(&self) -> Option<&dyn VirtualMethods> {
1116        None
1117    }
1118}
1119
1120impl Convert<AddEventListenerOptions> for AddEventListenerOptionsOrBoolean {
1121    /// <https://dom.spec.whatwg.org/#event-flatten-more>
1122    fn convert(self) -> AddEventListenerOptions {
1123        // Step 1. Let capture be the result of flattening options.
1124        // Step 5. Return capture, passive, once, and signal.
1125        match self {
1126            // Step 4. If options is a dictionary:
1127            AddEventListenerOptionsOrBoolean::AddEventListenerOptions(options) => options,
1128            AddEventListenerOptionsOrBoolean::Boolean(capture) => AddEventListenerOptions {
1129                parent: EventListenerOptions { capture },
1130                // Step 2. Let once be false.
1131                once: false,
1132                // Step 3. Let passive and signal be null.
1133                passive: None,
1134                signal: None,
1135            },
1136        }
1137    }
1138}
1139
1140impl Convert<EventListenerOptions> for EventListenerOptionsOrBoolean {
1141    fn convert(self) -> EventListenerOptions {
1142        match self {
1143            EventListenerOptionsOrBoolean::EventListenerOptions(options) => options,
1144            EventListenerOptionsOrBoolean::Boolean(capture) => EventListenerOptions { capture },
1145        }
1146    }
1147}