1use std::cell::Cell;
6
7use dom_struct::dom_struct;
8use js::jsapi::Heap;
9use js::jsval::JSVal;
10use js::rust::{HandleObject, HandleValue, MutableHandleValue};
11use stylo_atoms::Atom;
12
13use crate::dom::bindings::cell::DomRefCell;
14use crate::dom::bindings::codegen::Bindings::ErrorEventBinding;
15use crate::dom::bindings::codegen::Bindings::ErrorEventBinding::ErrorEventMethods;
16use crate::dom::bindings::codegen::Bindings::EventBinding::EventMethods;
17use crate::dom::bindings::error::Fallible;
18use crate::dom::bindings::inheritance::Castable;
19use crate::dom::bindings::reflector::reflect_dom_object_with_proto;
20use crate::dom::bindings::root::DomRoot;
21use crate::dom::bindings::str::DOMString;
22use crate::dom::bindings::trace::RootedTraceableBox;
23use crate::dom::event::{Event, EventBubbles, EventCancelable};
24use crate::dom::globalscope::GlobalScope;
25use crate::script_runtime::{CanGc, JSContext};
26
27#[dom_struct]
28pub(crate) struct ErrorEvent {
29 event: Event,
30 message: DomRefCell<DOMString>,
31 filename: DomRefCell<DOMString>,
32 lineno: Cell<u32>,
33 colno: Cell<u32>,
34 #[ignore_malloc_size_of = "Defined in rust-mozjs"]
35 error: Heap<JSVal>,
36}
37
38impl ErrorEvent {
39 fn new_inherited() -> ErrorEvent {
40 ErrorEvent {
41 event: Event::new_inherited(),
42 message: DomRefCell::new(DOMString::new()),
43 filename: DomRefCell::new(DOMString::new()),
44 lineno: Cell::new(0),
45 colno: Cell::new(0),
46 error: Heap::default(),
47 }
48 }
49
50 fn new_uninitialized(
51 global: &GlobalScope,
52 proto: Option<HandleObject>,
53 can_gc: CanGc,
54 ) -> DomRoot<ErrorEvent> {
55 reflect_dom_object_with_proto(Box::new(ErrorEvent::new_inherited()), global, proto, can_gc)
56 }
57
58 #[allow(clippy::too_many_arguments)]
59 pub(crate) fn new(
60 global: &GlobalScope,
61 type_: Atom,
62 bubbles: EventBubbles,
63 cancelable: EventCancelable,
64 message: DOMString,
65 filename: DOMString,
66 lineno: u32,
67 colno: u32,
68 error: HandleValue,
69 can_gc: CanGc,
70 ) -> DomRoot<ErrorEvent> {
71 Self::new_with_proto(
72 global, None, type_, bubbles, cancelable, message, filename, lineno, colno, error,
73 can_gc,
74 )
75 }
76
77 #[allow(clippy::too_many_arguments)]
78 fn new_with_proto(
79 global: &GlobalScope,
80 proto: Option<HandleObject>,
81 type_: Atom,
82 bubbles: EventBubbles,
83 cancelable: EventCancelable,
84 message: DOMString,
85 filename: DOMString,
86 lineno: u32,
87 colno: u32,
88 error: HandleValue,
89 can_gc: CanGc,
90 ) -> DomRoot<ErrorEvent> {
91 let ev = ErrorEvent::new_uninitialized(global, proto, can_gc);
92 {
93 let event = ev.upcast::<Event>();
94 event.init_event(type_, bool::from(bubbles), bool::from(cancelable));
95 *ev.message.borrow_mut() = message;
96 *ev.filename.borrow_mut() = filename;
97 ev.lineno.set(lineno);
98 ev.colno.set(colno);
99 }
100 ev.error.set(error.get());
101 ev
102 }
103}
104
105impl ErrorEventMethods<crate::DomTypeHolder> for ErrorEvent {
106 fn Constructor(
108 global: &GlobalScope,
109 proto: Option<HandleObject>,
110 can_gc: CanGc,
111 type_: DOMString,
112 init: RootedTraceableBox<ErrorEventBinding::ErrorEventInit>,
113 ) -> Fallible<DomRoot<ErrorEvent>> {
114 let msg = match init.message.as_ref() {
115 Some(message) => message.clone(),
116 None => DOMString::new(),
117 };
118
119 let file_name = match init.filename.as_ref() {
120 Some(filename) => filename.clone(),
121 None => DOMString::new(),
122 };
123
124 let line_num = init.lineno.unwrap_or(0);
125
126 let col_num = init.colno.unwrap_or(0);
127
128 let bubbles = EventBubbles::from(init.parent.bubbles);
129
130 let cancelable = EventCancelable::from(init.parent.cancelable);
131
132 let event = ErrorEvent::new_with_proto(
133 global,
134 proto,
135 Atom::from(type_),
136 bubbles,
137 cancelable,
138 msg,
139 file_name,
140 line_num,
141 col_num,
142 init.error.handle(),
143 can_gc,
144 );
145 event.upcast::<Event>().set_composed(init.parent.composed);
146 Ok(event)
147 }
148
149 fn Lineno(&self) -> u32 {
151 self.lineno.get()
152 }
153
154 fn Colno(&self) -> u32 {
156 self.colno.get()
157 }
158
159 fn Message(&self) -> DOMString {
161 self.message.borrow().clone()
162 }
163
164 fn Filename(&self) -> DOMString {
166 self.filename.borrow().clone()
167 }
168
169 fn Error(&self, _cx: JSContext, mut retval: MutableHandleValue) {
171 retval.set(self.error.get());
172 }
173
174 fn IsTrusted(&self) -> bool {
176 self.event.IsTrusted()
177 }
178}