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_and_cx};
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::virtualmethods::VirtualMethods;
71use crate::dom::node::{Node, NodeTraits};
72use crate::dom::shadowroot::ShadowRoot;
73use crate::dom::window::Window;
74use crate::dom::workerglobalscope::WorkerGlobalScope;
75use crate::realms::{enter_auto_realm, enter_realm};
76use crate::script_runtime::IntroductionType;
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 cx: &mut JSContext,
425 global: &GlobalScope,
426 proto: Option<HandleObject>,
427 ) -> DomRoot<EventTarget> {
428 reflect_dom_object_with_proto_and_cx(
429 Box::new(EventTarget::new_inherited()),
430 global,
431 proto,
432 cx,
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 pub(crate) fn has_non_passive_listener(&self, type_: &Atom) -> bool {
581 self.get_listeners_for(type_)
582 .iter()
583 .any(|listener| !self.is_passive(listener))
584 }
585
586 fn get_inline_event_listener(
587 &self,
588 cx: &mut JSContext,
589 ty: &Atom,
590 ) -> Option<CommonEventHandler> {
591 let handlers = self.handlers.borrow();
592 handlers
593 .get(ty)
594 .and_then(|entry| entry.get_inline_listener(cx, self, ty))
595 }
596
597 pub(crate) fn set_event_handler_uncompiled(
600 &self,
601 url: ServoUrl,
602 line: usize,
603 ty: &str,
604 source: &str,
605 ) {
606 if let Some(element) = self.downcast::<Element>() {
607 let doc = element.owner_document();
608 let global = &doc.global();
609 if global
610 .get_csp_list()
611 .should_elements_inline_type_behavior_be_blocked(
612 global,
613 element.upcast(),
614 InlineCheckType::ScriptAttribute,
615 source,
616 line as u32,
617 )
618 {
619 return;
620 }
621 };
622
623 let handler = InternalRawUncompiledHandler {
624 source: DOMString::from(source),
625 line,
626 url,
627 };
628 self.set_inline_event_listener(
629 Atom::from(ty),
630 Some(InlineEventListener::Uncompiled(handler)),
631 );
632 }
633
634 #[expect(unsafe_code)]
637 fn get_compiled_event_handler(
638 &self,
639 cx: &mut JSContext,
640 handler: InternalRawUncompiledHandler,
641 ty: &Atom,
642 ) -> Option<CommonEventHandler> {
643 let element = self.downcast::<Element>();
645 let document = match element {
646 Some(element) => element.owner_document(),
647 None => self.downcast::<Window>().unwrap().Document(),
648 };
649
650 if !document.scripting_enabled() {
652 return None;
653 }
654
655 let body: Vec<u16> = handler.source.str().encode_utf16().collect();
657
658 let form_owner = element
662 .and_then(|e| e.as_maybe_form_control())
663 .and_then(|f| f.form_owner());
664
665 let window = document.window();
673 let _ac = enter_realm(window);
674
675 let name = CString::new(format!("on{}", &**ty)).unwrap();
678
679 const ARG_NAMES: &[*const c_char] = &[c"event".as_ptr()];
681 const ERROR_ARG_NAMES: &[*const c_char] = &[
682 c"event".as_ptr(),
683 c"source".as_ptr(),
684 c"lineno".as_ptr(),
685 c"colno".as_ptr(),
686 c"error".as_ptr(),
687 ];
688 let is_error = ty == &atom!("error") && self.is::<Window>();
689 let args = if is_error { ERROR_ARG_NAMES } else { ARG_NAMES };
690
691 let url = cformat!("{}", handler.url);
692 let mut options = CompileOptionsWrapper::new(cx, url, handler.line as u32);
693 options.set_introduction_type(IntroductionType::EVENT_HANDLER);
694
695 let scopechain =
697 js::rust::EnvironmentChain::new(unsafe { cx.raw_cx() }, SupportUnscopables::Yes);
698
699 if let Some(element) = element {
700 scopechain.append(document.reflector().get_jsobject().get());
701 if let Some(form_owner) = form_owner {
702 scopechain.append(form_owner.reflector().get_jsobject().get());
703 }
704 scopechain.append(element.reflector().get_jsobject().get());
705 }
706
707 rooted!(&in(cx) let mut handler = unsafe {
708 CompileFunction(
709 cx,
710 scopechain.get(),
711 options.ptr,
712 name.as_ptr(),
713 args.len() as u32,
714 args.as_ptr(),
715 &mut transform_u16_to_source_text(&body),
716 )
717 });
718 if handler.get().is_null() {
719 let mut realm = enter_auto_realm(cx, self);
721 report_pending_exception(&mut realm.current_realm());
722 return None;
723 }
724
725 let funobj = unsafe { JS_GetFunctionObject(handler.get()) };
731 assert!(!funobj.is_null());
732 if is_error {
734 Some(CommonEventHandler::ErrorEventHandler(unsafe {
735 OnErrorEventHandlerNonNull::new(cx.into(), funobj)
736 }))
737 } else if ty == &atom!("beforeunload") {
738 Some(CommonEventHandler::BeforeUnloadEventHandler(unsafe {
739 OnBeforeUnloadEventHandlerNonNull::new(cx.into(), funobj)
740 }))
741 } else {
742 Some(CommonEventHandler::EventHandler(unsafe {
743 EventHandlerNonNull::new(cx.into(), funobj)
744 }))
745 }
746 }
747
748 #[expect(unsafe_code)]
749 pub(crate) fn set_event_handler_common<T: CallbackContainer<crate::DomTypeHolder>>(
750 &self,
751 cx: &mut JSContext,
752 ty: &str,
753 listener: Option<Rc<T>>,
754 ) {
755 let event_listener = listener.map(|listener| {
756 InlineEventListener::Compiled(CommonEventHandler::EventHandler(unsafe {
757 EventHandlerNonNull::new(cx.into(), listener.callback())
758 }))
759 });
760 self.set_inline_event_listener(Atom::from(ty), event_listener);
761 }
762
763 #[expect(unsafe_code)]
764 pub(crate) fn set_error_event_handler<T: CallbackContainer<crate::DomTypeHolder>>(
765 &self,
766 cx: &mut JSContext,
767 ty: &str,
768 listener: Option<Rc<T>>,
769 ) {
770 let event_listener = listener.map(|listener| {
771 InlineEventListener::Compiled(CommonEventHandler::ErrorEventHandler(unsafe {
772 OnErrorEventHandlerNonNull::new(cx.into(), listener.callback())
773 }))
774 });
775 self.set_inline_event_listener(Atom::from(ty), event_listener);
776 }
777
778 #[expect(unsafe_code)]
779 pub(crate) fn set_beforeunload_event_handler<T: CallbackContainer<crate::DomTypeHolder>>(
780 &self,
781 cx: &mut JSContext,
782 ty: &str,
783 listener: Option<Rc<T>>,
784 ) {
785 let event_listener = listener.map(|listener| {
786 InlineEventListener::Compiled(CommonEventHandler::BeforeUnloadEventHandler(unsafe {
787 OnBeforeUnloadEventHandlerNonNull::new(cx.into(), listener.callback())
788 }))
789 });
790 self.set_inline_event_listener(Atom::from(ty), event_listener);
791 }
792
793 #[expect(unsafe_code)]
794 pub(crate) fn get_event_handler_common<T: CallbackContainer<crate::DomTypeHolder>>(
795 &self,
796 cx: &mut JSContext,
797 ty: &str,
798 ) -> Option<Rc<T>> {
799 let listener = self.get_inline_event_listener(cx, &Atom::from(ty));
800 unsafe {
801 listener.map(|listener| {
802 CallbackContainer::new(cx.into(), listener.parent().callback_holder().get())
803 })
804 }
805 }
806
807 pub(crate) fn has_handlers(&self) -> bool {
808 !self.handlers.borrow().is_empty()
809 }
810
811 pub(crate) fn fire_event(&self, cx: &mut js::context::JSContext, name: Atom) -> bool {
813 self.fire_event_with_params(
814 cx,
815 name,
816 EventBubbles::DoesNotBubble,
817 EventCancelable::NotCancelable,
818 EventComposed::NotComposed,
819 )
820 }
821
822 pub(crate) fn fire_bubbling_event(&self, cx: &mut js::context::JSContext, name: Atom) -> bool {
824 self.fire_event_with_params(
825 cx,
826 name,
827 EventBubbles::Bubbles,
828 EventCancelable::NotCancelable,
829 EventComposed::NotComposed,
830 )
831 }
832
833 pub(crate) fn fire_cancelable_event(
835 &self,
836 cx: &mut js::context::JSContext,
837 name: Atom,
838 ) -> bool {
839 self.fire_event_with_params(
840 cx,
841 name,
842 EventBubbles::DoesNotBubble,
843 EventCancelable::Cancelable,
844 EventComposed::NotComposed,
845 )
846 }
847
848 pub(crate) fn fire_bubbling_cancelable_event(
850 &self,
851 cx: &mut js::context::JSContext,
852 name: Atom,
853 ) -> bool {
854 self.fire_event_with_params(
855 cx,
856 name,
857 EventBubbles::Bubbles,
858 EventCancelable::Cancelable,
859 EventComposed::NotComposed,
860 )
861 }
862
863 pub(crate) fn fire_event_with_params(
865 &self,
866 cx: &mut js::context::JSContext,
867 name: Atom,
868 bubbles: EventBubbles,
869 cancelable: EventCancelable,
870 composed: EventComposed,
871 ) -> bool {
872 let event = Event::new(cx, &self.global(), name, bubbles, cancelable);
873 event.set_composed(composed.into());
874 event.fire(cx, self)
875 }
876
877 pub(crate) fn add_event_listener(
880 &self,
881 ty: DOMString,
882 listener: Option<Rc<EventListener>>,
883 options: AddEventListenerOptions,
884 ) {
885 if let Some(signal) = options.signal.as_ref() {
886 if signal.aborted() {
888 return;
889 }
890 signal.add(&AbortAlgorithm::DomEventListener(
892 RemovableDomEventListener {
893 event_target: Dom::from_ref(self),
894 ty: ty.clone(),
895 listener: listener.clone(),
896 options: options.parent.clone(),
897 },
898 ));
899 }
900 let listener = match listener {
902 Some(l) => l,
903 None => return,
904 };
905 let ty = Atom::from(ty);
906 let mut handlers = self.handlers.borrow_mut();
907 let entries = match handlers.entry(ty.clone()) {
908 Occupied(entry) => entry.into_mut(),
909 Vacant(entry) => entry.insert(EventListeners(vec![])),
910 };
911
912 let phase = if options.parent.capture {
913 ListenerPhase::Capturing
914 } else {
915 ListenerPhase::Bubbling
916 };
917 let new_entry = Rc::new(RefCell::new(EventListenerEntry {
919 phase,
920 listener: EventListenerType::Additive(listener),
921 once: options.once,
922 passive: options.passive.unwrap_or(self.default_passive_value(&ty)),
923 removed: false,
924 }));
925
926 if !entries.contains(&new_entry) {
930 entries.push(new_entry);
931 self.notify_listener_added(&ty);
932 }
933 }
934
935 pub(crate) fn remove_event_listener(
938 &self,
939 ty: DOMString,
940 listener: &Option<Rc<EventListener>>,
941 options: &EventListenerOptions,
942 ) {
943 let Some(listener) = listener else {
944 return;
945 };
946 let ty_atom = Atom::from(ty);
947 let mut handlers = self.handlers.borrow_mut();
948 if let Some(entries) = handlers.get_mut(&ty_atom) {
949 let phase = if options.capture {
950 ListenerPhase::Capturing
951 } else {
952 ListenerPhase::Bubbling
953 };
954 let listener_type = EventListenerType::Additive(listener.clone());
955 if let Some(position) = entries
956 .iter()
957 .position(|e| e.borrow().listener == listener_type && e.borrow().phase == phase)
958 {
959 entries.remove(position).borrow_mut().removed = true;
961 self.notify_listener_removed(&ty_atom);
962 }
963 }
964 }
965
966 pub(crate) fn get_the_parent(&self, event: &Event) -> Option<DomRoot<EventTarget>> {
968 if let Some(document) = self.downcast::<Document>() {
969 if event.type_() == atom!("load") || !document.has_browsing_context() {
970 return None;
971 } else {
972 return Some(DomRoot::from_ref(document.window().upcast::<EventTarget>()));
973 }
974 }
975
976 if let Some(shadow_root) = self.downcast::<ShadowRoot>() {
977 if event.should_pass_shadow_boundary(shadow_root) {
978 let host = shadow_root.Host();
979 return Some(DomRoot::from_ref(host.upcast::<EventTarget>()));
980 } else {
981 return None;
982 }
983 }
984
985 if let Some(node) = self.downcast::<Node>() {
986 return node.assigned_slot().map(DomRoot::upcast).or_else(|| {
989 node.GetParentNode()
990 .map(|parent| DomRoot::from_ref(parent.upcast::<EventTarget>()))
991 });
992 }
993
994 if let Some(request) = self.downcast::<IDBRequest>() {
997 return request
998 .transaction()
999 .map(|tx| DomRoot::from_ref(tx.upcast::<EventTarget>()));
1000 }
1001
1002 if let Some(transaction) = self.downcast::<IDBTransaction>() {
1004 return Some(DomRoot::from_ref(
1005 transaction.get_db().upcast::<EventTarget>(),
1006 ));
1007 }
1008
1009 if self.is::<IDBDatabase>() {
1011 return None;
1012 }
1013
1014 None
1015 }
1016
1017 pub(crate) fn retarget(&self, b: &Self) -> DomRoot<EventTarget> {
1021 let mut a = DomRoot::from_ref(self);
1023 loop {
1024 let Some(a_node) = a.downcast::<Node>() else {
1030 return a;
1031 };
1032 let a_root = a_node.GetRootNode(&GetRootNodeOptions::empty());
1033 if !a_root.is::<ShadowRoot>() {
1034 return a;
1035 }
1036 if let Some(b_node) = b.downcast::<Node>() &&
1037 a_root.is_shadow_including_inclusive_ancestor_of(b_node)
1038 {
1039 return a;
1040 }
1041
1042 a = DomRoot::from_ref(
1044 a_root
1045 .downcast::<ShadowRoot>()
1046 .unwrap()
1047 .Host()
1048 .upcast::<EventTarget>(),
1049 );
1050 }
1051 }
1052
1053 pub(crate) fn is_content_event_handler(name: &str) -> bool {
1055 CONTENT_EVENT_HANDLER_NAMES.contains(&name)
1056 }
1057
1058 pub(crate) fn summarize_event_listeners_for_devtools(&self) -> Vec<EventListenerInfo> {
1059 let handlers = self.handlers.borrow();
1060 let mut listener_infos = Vec::with_capacity(handlers.0.len());
1061 for (event_type, event_listeners) in &handlers.0 {
1062 for event_listener in event_listeners.iter() {
1063 let event_listener_entry = event_listener.borrow();
1064 listener_infos.push(EventListenerInfo {
1065 event_type: event_type.to_string(),
1066 capturing: event_listener_entry.phase() == ListenerPhase::Capturing,
1067 });
1068 }
1069 }
1070
1071 listener_infos
1072 }
1073}
1074
1075impl EventTargetMethods<crate::DomTypeHolder> for EventTarget {
1076 fn Constructor(
1078 cx: &mut JSContext,
1079 global: &GlobalScope,
1080 proto: Option<HandleObject>,
1081 ) -> Fallible<DomRoot<EventTarget>> {
1082 Ok(EventTarget::new(cx, global, proto))
1083 }
1084
1085 fn AddEventListener(
1087 &self,
1088 ty: DOMString,
1089 listener: Option<Rc<EventListener>>,
1090 options: AddEventListenerOptionsOrBoolean,
1091 ) {
1092 self.add_event_listener(ty, listener, options.convert())
1093 }
1094
1095 fn RemoveEventListener(
1097 &self,
1098 ty: DOMString,
1099 listener: Option<Rc<EventListener>>,
1100 options: EventListenerOptionsOrBoolean,
1101 ) {
1102 self.remove_event_listener(ty, &listener, &options.convert())
1103 }
1104
1105 fn DispatchEvent(&self, cx: &mut JSContext, event: &Event) -> Fallible<bool> {
1107 if event.dispatching() || !event.initialized() {
1108 return Err(Error::InvalidState(None));
1109 }
1110 event.set_trusted(false);
1111 Ok(event.dispatch(cx, self, false))
1112 }
1113}
1114
1115impl VirtualMethods for EventTarget {
1116 fn super_type(&self) -> Option<&dyn VirtualMethods> {
1117 None
1118 }
1119}
1120
1121impl Convert<AddEventListenerOptions> for AddEventListenerOptionsOrBoolean {
1122 fn convert(self) -> AddEventListenerOptions {
1124 match self {
1127 AddEventListenerOptionsOrBoolean::AddEventListenerOptions(options) => options,
1129 AddEventListenerOptionsOrBoolean::Boolean(capture) => AddEventListenerOptions {
1130 parent: EventListenerOptions { capture },
1131 once: false,
1133 passive: None,
1135 signal: None,
1136 },
1137 }
1138 }
1139}
1140
1141impl Convert<EventListenerOptions> for EventListenerOptionsOrBoolean {
1142 fn convert(self) -> EventListenerOptions {
1143 match self {
1144 EventListenerOptionsOrBoolean::EventListenerOptions(options) => options,
1145 EventListenerOptionsOrBoolean::Boolean(capture) => EventListenerOptions { capture },
1146 }
1147 }
1148}