1use 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
78pub(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#[derive(Clone, JSTraceable, MallocSizeOf, PartialEq)]
116struct InternalRawUncompiledHandler {
117 source: DOMString,
118 #[no_trace]
119 url: ServoUrl,
120 line: usize,
121}
122
123#[derive(Clone, JSTraceable, MallocSizeOf, PartialEq)]
125enum InlineEventListener {
126 Uncompiled(InternalRawUncompiledHandler),
127 Compiled(CommonEventHandler),
128 Null,
129}
130
131fn 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
181pub(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 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 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 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 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 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 let should_cancel = value.is_boolean() && !value.to_boolean();
306
307 if should_cancel {
308 event.PreventDefault();
312 }
313 Ok(())
314 },
315 Err(err) => Err(err),
316 }
317 },
318 }
319 },
320 }
321 }
322}
323
324#[derive(Clone, DenyPublicFields, JSTraceable, MallocSizeOf)]
327pub(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 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)]
367pub(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 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 return get_compiled_handler(cx, inline, owner, ty);
397 }
398 }
399
400 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 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 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 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 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 fn default_passive_value(&self, ty: &Atom) -> bool {
490 let event_type = ty.to_ascii_lowercase();
492
493 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 if self.is::<Window>() {
505 return true;
506 }
507
508 if let Some(node) = self.downcast::<Node>() {
510 let node_document = node.owner_document();
511 let event_target = self.upcast::<EventTarget>();
512
513 return event_target == node_document.upcast::<EventTarget>()
515 || node_document.GetDocumentElement().is_some_and(|n| n.upcast::<EventTarget>() == event_target)
517 || node_document.GetBody().is_some_and(|n| n.upcast::<EventTarget>() == event_target);
519 }
520
521 false
522 }
523
524 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 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 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 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 #[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 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 if !document.scripting_enabled() {
645 return None;
646 }
647
648 let body: Vec<u16> = handler.source.str().encode_utf16().collect();
650
651 let form_owner = element
655 .and_then(|e| e.as_maybe_form_control())
656 .and_then(|f| f.form_owner());
657
658 let window = document.window();
666 let _ac = enter_realm(window);
667
668 let name = CString::new(format!("on{}", &**ty)).unwrap();
671
672 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 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 let mut realm = enter_auto_realm(cx, self);
714 report_pending_exception(&mut realm.current_realm());
715 return None;
716 }
717
718 let funobj = unsafe { JS_GetFunctionObject(handler.get()) };
724 assert!(!funobj.is_null());
725 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 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 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 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 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 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 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 if signal.aborted() {
887 return;
888 }
889 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 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 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 if !entries.contains(&new_entry) {
929 entries.push(new_entry);
930 self.notify_listener_added(&ty);
931 }
932 }
933
934 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 entries.remove(position).borrow_mut().removed = true;
960 self.notify_listener_removed(&ty_atom);
961 }
962 }
963 }
964
965 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 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 if let Some(request) = self.downcast::<IDBRequest>() {
996 return request
997 .transaction()
998 .map(|tx| DomRoot::from_ref(tx.upcast::<EventTarget>()));
999 }
1000
1001 if let Some(transaction) = self.downcast::<IDBTransaction>() {
1003 return Some(DomRoot::from_ref(
1004 transaction.get_db().upcast::<EventTarget>(),
1005 ));
1006 }
1007
1008 if self.downcast::<IDBDatabase>().is_some() {
1010 return None;
1011 }
1012
1013 None
1014 }
1015
1016 pub(crate) fn retarget(&self, b: &Self) -> DomRoot<EventTarget> {
1020 let mut a = DomRoot::from_ref(self);
1022 loop {
1023 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 a = DomRoot::from_ref(
1043 a_root
1044 .downcast::<ShadowRoot>()
1045 .unwrap()
1046 .Host()
1047 .upcast::<EventTarget>(),
1048 );
1049 }
1050 }
1051
1052 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 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 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 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 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 fn convert(self) -> AddEventListenerOptions {
1123 match self {
1126 AddEventListenerOptionsOrBoolean::AddEventListenerOptions(options) => options,
1128 AddEventListenerOptionsOrBoolean::Boolean(capture) => AddEventListenerOptions {
1129 parent: EventListenerOptions { capture },
1130 once: false,
1132 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}