1use html5ever::{LocalName, Prefix, QualName, local_name, ns};
6use js::context::JSContext;
7use js::rust::HandleObject;
8
9use crate::dom::bindings::error::{report_pending_exception, throw_dom_exception};
10use crate::dom::bindings::inheritance::Castable;
11use crate::dom::bindings::reflector::DomGlobal;
12use crate::dom::bindings::root::DomRoot;
13use crate::dom::customelementregistry::{
14 CustomElementRegistry, CustomElementState, is_valid_custom_element_name, upgrade_element,
15};
16use crate::dom::document::Document;
17use crate::dom::element::{CustomElementCreationMode, Element, ElementCreator};
18use crate::dom::globalscope::GlobalScope;
19use crate::dom::html::htmlanchorelement::HTMLAnchorElement;
20use crate::dom::html::htmlareaelement::HTMLAreaElement;
21use crate::dom::html::htmlaudioelement::HTMLAudioElement;
22use crate::dom::html::htmlbaseelement::HTMLBaseElement;
23use crate::dom::html::htmlbodyelement::HTMLBodyElement;
24use crate::dom::html::htmlbrelement::HTMLBRElement;
25use crate::dom::html::htmlbuttonelement::HTMLButtonElement;
26use crate::dom::html::htmlcanvaselement::HTMLCanvasElement;
27use crate::dom::html::htmldataelement::HTMLDataElement;
28use crate::dom::html::htmldatalistelement::HTMLDataListElement;
29use crate::dom::html::htmldetailselement::HTMLDetailsElement;
30use crate::dom::html::htmldialogelement::HTMLDialogElement;
31use crate::dom::html::htmldirectoryelement::HTMLDirectoryElement;
32use crate::dom::html::htmldivelement::HTMLDivElement;
33use crate::dom::html::htmldlistelement::HTMLDListElement;
34use crate::dom::html::htmlelement::HTMLElement;
35use crate::dom::html::htmlembedelement::HTMLEmbedElement;
36use crate::dom::html::htmlfieldsetelement::HTMLFieldSetElement;
37use crate::dom::html::htmlfontelement::HTMLFontElement;
38use crate::dom::html::htmlformelement::HTMLFormElement;
39use crate::dom::html::htmlframeelement::HTMLFrameElement;
40use crate::dom::html::htmlframesetelement::HTMLFrameSetElement;
41use crate::dom::html::htmlheadelement::HTMLHeadElement;
42use crate::dom::html::htmlheadingelement::{HTMLHeadingElement, HeadingLevel};
43use crate::dom::html::htmlhrelement::HTMLHRElement;
44use crate::dom::html::htmlhtmlelement::HTMLHtmlElement;
45use crate::dom::html::htmliframeelement::HTMLIFrameElement;
46use crate::dom::html::htmlimageelement::HTMLImageElement;
47use crate::dom::html::htmllabelelement::HTMLLabelElement;
48use crate::dom::html::htmllegendelement::HTMLLegendElement;
49use crate::dom::html::htmllielement::HTMLLIElement;
50use crate::dom::html::htmllinkelement::HTMLLinkElement;
51use crate::dom::html::htmlmapelement::HTMLMapElement;
52use crate::dom::html::htmlmenuelement::HTMLMenuElement;
53use crate::dom::html::htmlmetaelement::HTMLMetaElement;
54use crate::dom::html::htmlmeterelement::HTMLMeterElement;
55use crate::dom::html::htmlmodelement::HTMLModElement;
56use crate::dom::html::htmlobjectelement::HTMLObjectElement;
57use crate::dom::html::htmlolistelement::HTMLOListElement;
58use crate::dom::html::htmloptgroupelement::HTMLOptGroupElement;
59use crate::dom::html::htmloptionelement::HTMLOptionElement;
60use crate::dom::html::htmloutputelement::HTMLOutputElement;
61use crate::dom::html::htmlparagraphelement::HTMLParagraphElement;
62use crate::dom::html::htmlparamelement::HTMLParamElement;
63use crate::dom::html::htmlpictureelement::HTMLPictureElement;
64use crate::dom::html::htmlpreelement::HTMLPreElement;
65use crate::dom::html::htmlprogresselement::HTMLProgressElement;
66use crate::dom::html::htmlquoteelement::HTMLQuoteElement;
67use crate::dom::html::htmlscriptelement::HTMLScriptElement;
68use crate::dom::html::htmlselectelement::HTMLSelectElement;
69use crate::dom::html::htmlslotelement::HTMLSlotElement;
70use crate::dom::html::htmlsourceelement::HTMLSourceElement;
71use crate::dom::html::htmlspanelement::HTMLSpanElement;
72use crate::dom::html::htmlstyleelement::HTMLStyleElement;
73use crate::dom::html::htmltablecaptionelement::HTMLTableCaptionElement;
74use crate::dom::html::htmltablecellelement::HTMLTableCellElement;
75use crate::dom::html::htmltablecolelement::HTMLTableColElement;
76use crate::dom::html::htmltableelement::HTMLTableElement;
77use crate::dom::html::htmltablerowelement::HTMLTableRowElement;
78use crate::dom::html::htmltablesectionelement::HTMLTableSectionElement;
79use crate::dom::html::htmltemplateelement::HTMLTemplateElement;
80use crate::dom::html::htmltextareaelement::HTMLTextAreaElement;
81use crate::dom::html::htmltimeelement::HTMLTimeElement;
82use crate::dom::html::htmltitleelement::HTMLTitleElement;
83use crate::dom::html::htmltrackelement::HTMLTrackElement;
84use crate::dom::html::htmlulistelement::HTMLUListElement;
85use crate::dom::html::htmlunknownelement::HTMLUnknownElement;
86use crate::dom::html::htmlvideoelement::HTMLVideoElement;
87use crate::dom::html::input_element::HTMLInputElement;
88use crate::dom::htmlmarqueeelement::HTMLMarqueeElement;
89use crate::dom::svg::svgelement::SVGElement;
90use crate::dom::svg::svgimageelement::SVGImageElement;
91use crate::dom::svg::svgsvgelement::SVGSVGElement;
92use crate::realms::{InRealm, enter_auto_realm};
93use crate::script_runtime::CanGc;
94use crate::script_thread::ScriptThread;
95
96fn create_svg_element(
97 cx: &mut JSContext,
98 name: QualName,
99 prefix: Option<Prefix>,
100 document: &Document,
101 proto: Option<HandleObject>,
102) -> DomRoot<Element> {
103 assert_eq!(name.ns, ns!(svg));
104
105 macro_rules! make(
106 ($ctor:ident) => ({
107 let obj = $ctor::new(cx, name.local, prefix, document, proto);
108 DomRoot::upcast(obj)
109 });
110 );
111
112 match name.local {
113 local_name!("image") => make!(SVGImageElement),
114 local_name!("svg") => make!(SVGSVGElement),
115 _ => make!(SVGElement),
116 }
117}
118
119#[expect(clippy::too_many_arguments)]
121fn create_html_element(
122 cx: &mut JSContext,
123 name: QualName,
124 prefix: Option<Prefix>,
125 is: Option<LocalName>,
126 document: &Document,
127 creator: ElementCreator,
128 mode: CustomElementCreationMode,
129 proto: Option<HandleObject>,
130) -> DomRoot<Element> {
131 assert_eq!(name.ns, ns!(html));
132
133 let registry = CustomElementRegistry::lookup_a_custom_element_registry(document.upcast());
137
138 let definition = document.lookup_custom_element_definition(&name.ns, &name.local, is.as_ref());
141
142 if let Some(definition) = definition {
144 if !definition.is_autonomous() {
147 let element = create_native_html_element(cx, name, prefix, document, creator, proto);
151 element.set_is(definition.name.clone());
152 element.set_custom_element_state(CustomElementState::Undefined);
153 element.set_custom_element_registry(registry);
154
155 match mode {
156 CustomElementCreationMode::Synchronous => {
158 upgrade_element(cx, definition, &element);
160 },
162 CustomElementCreationMode::Asynchronous => {
164 ScriptThread::enqueue_upgrade_reaction(&element, definition)
165 },
166 }
167 return element;
168 } else {
169 match mode {
171 CustomElementCreationMode::Synchronous => {
174 let local_name = name.local;
175 return match definition.create_element(
178 document,
179 prefix.clone(),
180 registry.clone(),
181 CanGc::from_cx(cx),
182 ) {
183 Ok(element) => {
184 element.set_custom_element_definition(definition.clone());
185 element
186 },
187 Err(error) => {
188 let global =
190 GlobalScope::current().unwrap_or_else(|| document.global());
191
192 let mut realm = enter_auto_realm(cx, &*global);
193 let cx = &mut realm.current_realm();
194
195 let in_realm_proof = cx.into();
196 let in_realm = InRealm::Already(&in_realm_proof);
197
198 throw_dom_exception(cx.into(), &global, error, CanGc::from_cx(cx));
201 report_pending_exception(cx.into(), in_realm, CanGc::from_cx(cx));
202
203 let element = DomRoot::upcast::<Element>(HTMLUnknownElement::new(
206 cx, local_name, prefix, document, proto,
207 ));
208 element.set_custom_element_state(CustomElementState::Failed);
209 element.set_custom_element_registry(registry);
210 element
211 },
212 };
213 },
214 CustomElementCreationMode::Asynchronous => {
216 let result = DomRoot::upcast::<Element>(HTMLElement::new(
222 cx, name.local, prefix, document, proto,
223 ));
224 result.set_custom_element_state(CustomElementState::Undefined);
225 result.set_custom_element_registry(registry);
226 ScriptThread::enqueue_upgrade_reaction(&result, definition);
228 return result;
229 },
230 }
231 }
232 }
233
234 let result = create_native_html_element(cx, name.clone(), prefix, document, creator, proto);
241 match is {
244 Some(is) => {
245 result.set_is(is);
246 result.set_custom_element_state(CustomElementState::Undefined);
247 result.set_custom_element_registry(registry);
248 },
249 None => {
250 if is_valid_custom_element_name(&name.local) {
251 result.set_custom_element_state(CustomElementState::Undefined);
252 result.set_custom_element_registry(registry);
253 } else {
254 result.set_initial_custom_element_state_to_uncustomized();
257 }
258 },
259 };
260
261 result
263}
264
265pub(crate) fn create_native_html_element(
266 cx: &mut JSContext,
267 name: QualName,
268 prefix: Option<Prefix>,
269 document: &Document,
270 creator: ElementCreator,
271 proto: Option<HandleObject>,
272) -> DomRoot<Element> {
273 assert_eq!(name.ns, ns!(html));
274
275 macro_rules! make(
276 ($ctor:ident) => ({
277 let obj = $ctor::new(cx, name.local, prefix, document, proto);
278 DomRoot::upcast(obj)
279 });
280 ($ctor:ident, $($arg:expr),+) => ({
281 let obj = $ctor::new(cx, name.local, prefix, document, proto, $($arg),+);
282 DomRoot::upcast(obj)
283 })
284 );
285
286 match name.local {
290 local_name!("a") => make!(HTMLAnchorElement),
291 local_name!("abbr") => make!(HTMLElement),
292 local_name!("acronym") => make!(HTMLElement),
293 local_name!("address") => make!(HTMLElement),
294 local_name!("area") => make!(HTMLAreaElement),
295 local_name!("article") => make!(HTMLElement),
296 local_name!("aside") => make!(HTMLElement),
297 local_name!("audio") => make!(HTMLAudioElement),
298 local_name!("b") => make!(HTMLElement),
299 local_name!("base") => make!(HTMLBaseElement),
300 local_name!("bdi") => make!(HTMLElement),
301 local_name!("bdo") => make!(HTMLElement),
302 local_name!("bgsound") => make!(HTMLUnknownElement),
304 local_name!("big") => make!(HTMLElement),
305 local_name!("blink") => make!(HTMLUnknownElement),
307 local_name!("blockquote") => make!(HTMLQuoteElement),
309 local_name!("body") => make!(HTMLBodyElement),
310 local_name!("br") => make!(HTMLBRElement),
311 local_name!("button") => make!(HTMLButtonElement),
312 local_name!("canvas") => make!(HTMLCanvasElement),
313 local_name!("caption") => make!(HTMLTableCaptionElement),
314 local_name!("center") => make!(HTMLElement),
315 local_name!("cite") => make!(HTMLElement),
316 local_name!("code") => make!(HTMLElement),
317 local_name!("col") => make!(HTMLTableColElement),
318 local_name!("colgroup") => make!(HTMLTableColElement),
319 local_name!("data") => make!(HTMLDataElement),
320 local_name!("datalist") => make!(HTMLDataListElement),
321 local_name!("dd") => make!(HTMLElement),
322 local_name!("del") => make!(HTMLModElement),
323 local_name!("details") => make!(HTMLDetailsElement),
324 local_name!("dfn") => make!(HTMLElement),
325 local_name!("dialog") => make!(HTMLDialogElement),
326 local_name!("dir") => make!(HTMLDirectoryElement),
327 local_name!("div") => make!(HTMLDivElement),
328 local_name!("dl") => make!(HTMLDListElement),
329 local_name!("dt") => make!(HTMLElement),
330 local_name!("em") => make!(HTMLElement),
331 local_name!("embed") => make!(HTMLEmbedElement),
332 local_name!("fieldset") => make!(HTMLFieldSetElement),
333 local_name!("figcaption") => make!(HTMLElement),
334 local_name!("figure") => make!(HTMLElement),
335 local_name!("font") => make!(HTMLFontElement),
336 local_name!("footer") => make!(HTMLElement),
337 local_name!("form") => make!(HTMLFormElement),
338 local_name!("frame") => make!(HTMLFrameElement),
339 local_name!("frameset") => make!(HTMLFrameSetElement),
340 local_name!("h1") => make!(HTMLHeadingElement, HeadingLevel::Heading1),
341 local_name!("h2") => make!(HTMLHeadingElement, HeadingLevel::Heading2),
342 local_name!("h3") => make!(HTMLHeadingElement, HeadingLevel::Heading3),
343 local_name!("h4") => make!(HTMLHeadingElement, HeadingLevel::Heading4),
344 local_name!("h5") => make!(HTMLHeadingElement, HeadingLevel::Heading5),
345 local_name!("h6") => make!(HTMLHeadingElement, HeadingLevel::Heading6),
346 local_name!("head") => make!(HTMLHeadElement),
347 local_name!("header") => make!(HTMLElement),
348 local_name!("hgroup") => make!(HTMLElement),
349 local_name!("hr") => make!(HTMLHRElement),
350 local_name!("html") => make!(HTMLHtmlElement),
351 local_name!("i") => make!(HTMLElement),
352 local_name!("iframe") => make!(HTMLIFrameElement),
353 local_name!("img") => make!(HTMLImageElement, creator),
354 local_name!("input") => make!(HTMLInputElement),
355 local_name!("ins") => make!(HTMLModElement),
356 local_name!("isindex") => make!(HTMLUnknownElement),
358 local_name!("kbd") => make!(HTMLElement),
359 local_name!("keygen") => make!(HTMLUnknownElement),
361 local_name!("label") => make!(HTMLLabelElement),
362 local_name!("legend") => make!(HTMLLegendElement),
363 local_name!("li") => make!(HTMLLIElement),
364 local_name!("link") => make!(HTMLLinkElement, creator),
365 local_name!("listing") => make!(HTMLPreElement),
367 local_name!("main") => make!(HTMLElement),
368 local_name!("map") => make!(HTMLMapElement),
369 local_name!("mark") => make!(HTMLElement),
370 local_name!("marquee") => make!(HTMLMarqueeElement),
371 local_name!("menu") => make!(HTMLMenuElement),
372 local_name!("meta") => make!(HTMLMetaElement),
373 local_name!("meter") => make!(HTMLMeterElement),
374 local_name!("multicol") => make!(HTMLUnknownElement),
376 local_name!("nav") => make!(HTMLElement),
377 local_name!("nextid") => make!(HTMLUnknownElement),
379 local_name!("nobr") => make!(HTMLElement),
380 local_name!("noframes") => make!(HTMLElement),
381 local_name!("noscript") => make!(HTMLElement),
382 local_name!("object") => make!(HTMLObjectElement),
383 local_name!("ol") => make!(HTMLOListElement),
384 local_name!("optgroup") => make!(HTMLOptGroupElement),
385 local_name!("option") => make!(HTMLOptionElement),
386 local_name!("output") => make!(HTMLOutputElement),
387 local_name!("p") => make!(HTMLParagraphElement),
388 local_name!("param") => make!(HTMLParamElement),
389 local_name!("picture") => make!(HTMLPictureElement),
390 local_name!("plaintext") => make!(HTMLPreElement),
391 local_name!("pre") => make!(HTMLPreElement),
392 local_name!("progress") => make!(HTMLProgressElement),
393 local_name!("q") => make!(HTMLQuoteElement),
394 local_name!("rp") => make!(HTMLElement),
395 local_name!("rt") => make!(HTMLElement),
396 local_name!("ruby") => make!(HTMLElement),
397 local_name!("s") => make!(HTMLElement),
398 local_name!("samp") => make!(HTMLElement),
399 local_name!("script") => make!(HTMLScriptElement, creator),
400 local_name!("section") => make!(HTMLElement),
401 local_name!("select") => make!(HTMLSelectElement),
402 local_name!("slot") => make!(HTMLSlotElement),
403 local_name!("small") => make!(HTMLElement),
404 local_name!("source") => make!(HTMLSourceElement),
405 local_name!("spacer") => make!(HTMLUnknownElement),
407 local_name!("span") => make!(HTMLSpanElement),
408 local_name!("strike") => make!(HTMLElement),
409 local_name!("strong") => make!(HTMLElement),
410 local_name!("style") => make!(HTMLStyleElement, creator),
411 local_name!("sub") => make!(HTMLElement),
412 local_name!("summary") => make!(HTMLElement),
413 local_name!("sup") => make!(HTMLElement),
414 local_name!("table") => make!(HTMLTableElement),
415 local_name!("tbody") => make!(HTMLTableSectionElement),
416 local_name!("td") => make!(HTMLTableCellElement),
417 local_name!("template") => make!(HTMLTemplateElement),
418 local_name!("textarea") => make!(HTMLTextAreaElement),
419 local_name!("tfoot") => make!(HTMLTableSectionElement),
421 local_name!("th") => make!(HTMLTableCellElement),
422 local_name!("thead") => make!(HTMLTableSectionElement),
424 local_name!("time") => make!(HTMLTimeElement),
425 local_name!("title") => make!(HTMLTitleElement),
426 local_name!("tr") => make!(HTMLTableRowElement),
427 local_name!("tt") => make!(HTMLElement),
428 local_name!("track") => make!(HTMLTrackElement),
429 local_name!("u") => make!(HTMLElement),
430 local_name!("ul") => make!(HTMLUListElement),
431 local_name!("var") => make!(HTMLElement),
432 local_name!("video") => make!(HTMLVideoElement),
433 local_name!("wbr") => make!(HTMLElement),
434 local_name!("xmp") => make!(HTMLPreElement),
435 _ if is_valid_custom_element_name(&name.local) => make!(HTMLElement),
436 _ => make!(HTMLUnknownElement),
437 }
438}
439
440pub(crate) fn create_element(
441 cx: &mut JSContext,
442 name: QualName,
443 is: Option<LocalName>,
444 document: &Document,
445 creator: ElementCreator,
446 mode: CustomElementCreationMode,
447 proto: Option<HandleObject>,
448) -> DomRoot<Element> {
449 let prefix = name.prefix.clone();
450 match name.ns {
451 ns!(html) => create_html_element(cx, name, prefix, is, document, creator, mode, proto),
452 ns!(svg) => create_svg_element(cx, name, prefix, document, proto),
453 _ => Element::new(cx, name.local, name.ns, prefix, document, proto),
454 }
455}