/* * Copyright (c) 2018-2022, Andreas Kling * Copyright (c) 2020-2023, Luke Wilde * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace Web::DOM { ErrorOr> valid_local_names_for_given_html_element_interface(StringView html_element_interface_name) { if (html_element_interface_name == "HTMLAnchorElement"sv) return FixedArray::create({ HTML::TagNames::a }); if (html_element_interface_name == "HTMLAreaElement"sv) return FixedArray::create({ HTML::TagNames::area }); if (html_element_interface_name == "HTMLAudioElement"sv) return FixedArray::create({ HTML::TagNames::audio }); if (html_element_interface_name == "HTMLBaseElement"sv) return FixedArray::create({ HTML::TagNames::base }); if (html_element_interface_name == "HTMLBodyElement"sv) return FixedArray::create({ HTML::TagNames::body }); if (html_element_interface_name == "HTMLBRElement"sv) return FixedArray::create({ HTML::TagNames::br }); if (html_element_interface_name == "HTMLButtonElement"sv) return FixedArray::create({ HTML::TagNames::button }); if (html_element_interface_name == "HTMLCanvasElement"sv) return FixedArray::create({ HTML::TagNames::canvas }); if (html_element_interface_name == "HTMLDataElement"sv) return FixedArray::create({ HTML::TagNames::data }); if (html_element_interface_name == "HTMLDataListElement"sv) return FixedArray::create({ HTML::TagNames::datalist }); if (html_element_interface_name == "HTMLDetailsElement"sv) return FixedArray::create({ HTML::TagNames::details }); if (html_element_interface_name == "HTMLDialogElement"sv) return FixedArray::create({ HTML::TagNames::dialog }); if (html_element_interface_name == "HTMLDirectoryElement"sv) return FixedArray::create({ HTML::TagNames::dir }); if (html_element_interface_name == "HTMLDivElement"sv) return FixedArray::create({ HTML::TagNames::div }); if (html_element_interface_name == "HTMLDListElement"sv) return FixedArray::create({ HTML::TagNames::dl }); if (html_element_interface_name == "HTMLEmbedElement"sv) return FixedArray::create({ HTML::TagNames::embed }); if (html_element_interface_name == "HTMLFieldsetElement"sv) return FixedArray::create({ HTML::TagNames::fieldset }); if (html_element_interface_name == "HTMLFontElement"sv) return FixedArray::create({ HTML::TagNames::font }); if (html_element_interface_name == "HTMLFormElement"sv) return FixedArray::create({ HTML::TagNames::form }); if (html_element_interface_name == "HTMLFrameElement"sv) return FixedArray::create({ HTML::TagNames::frame }); if (html_element_interface_name == "HTMLFrameSetElement"sv) return FixedArray::create({ HTML::TagNames::frameset }); if (html_element_interface_name == "HTMLHeadElement"sv) return FixedArray::create({ HTML::TagNames::head }); if (html_element_interface_name == "HTMLHeadingElement"sv) return FixedArray::create({ HTML::TagNames::h1, HTML::TagNames::h2, HTML::TagNames::h3, HTML::TagNames::h4, HTML::TagNames::h5, HTML::TagNames::h6 }); if (html_element_interface_name == "HTMLHRElement"sv) return FixedArray::create({ HTML::TagNames::hr }); if (html_element_interface_name == "HTMLHtmlElement"sv) return FixedArray::create({ HTML::TagNames::html }); if (html_element_interface_name == "HTMLIFrameElement"sv) return FixedArray::create({ HTML::TagNames::iframe }); if (html_element_interface_name == "HTMLImageElement"sv) return FixedArray::create({ HTML::TagNames::img }); if (html_element_interface_name == "HTMLInputElement"sv) return FixedArray::create({ HTML::TagNames::input }); if (html_element_interface_name == "HTMLLabelElement"sv) return FixedArray::create({ HTML::TagNames::label }); if (html_element_interface_name == "HTMLLIElement"sv) return FixedArray::create({ HTML::TagNames::li }); if (html_element_interface_name == "HTMLLinkElement"sv) return FixedArray::create({ HTML::TagNames::link }); if (html_element_interface_name == "HTMLMapElement"sv) return FixedArray::create({ HTML::TagNames::map }); if (html_element_interface_name == "HTMLMarqueeElement"sv) return FixedArray::create({ HTML::TagNames::marquee }); if (html_element_interface_name == "HTMLMenuElement"sv) return FixedArray::create({ HTML::TagNames::menu }); if (html_element_interface_name == "HTMLMeterElement"sv) return FixedArray::create({ HTML::TagNames::meter }); if (html_element_interface_name == "HTMLModElement"sv) return FixedArray::create({ HTML::TagNames::ins, HTML::TagNames::del }); if (html_element_interface_name == "HTMLObjectElement"sv) return FixedArray::create({ HTML::TagNames::object }); if (html_element_interface_name == "HTMLOutputElement"sv) return FixedArray::create({ HTML::TagNames::output }); if (html_element_interface_name == "HTMLParagraphElement"sv) return FixedArray::create({ HTML::TagNames::p }); if (html_element_interface_name == "HTMLParamElement"sv) return FixedArray::create({ HTML::TagNames::param }); if (html_element_interface_name == "HTMLPictureElement"sv) return FixedArray::create({ HTML::TagNames::picture }); if (html_element_interface_name == "HTMLPreElement"sv) return FixedArray::create({ HTML::TagNames::pre, HTML::TagNames::listing, HTML::TagNames::xmp }); if (html_element_interface_name == "HTMLProgressElement"sv) return FixedArray::create({ HTML::TagNames::progress }); if (html_element_interface_name == "HTMLQuoteElement"sv) return FixedArray::create({ HTML::TagNames::blockquote, HTML::TagNames::q }); if (html_element_interface_name == "HTMLScriptElement"sv) return FixedArray::create({ HTML::TagNames::script }); if (html_element_interface_name == "HTMLSelectElement"sv) return FixedArray::create({ HTML::TagNames::select }); if (html_element_interface_name == "HTMLSlotElement"sv) return FixedArray::create({ HTML::TagNames::slot }); if (html_element_interface_name == "HTMLSourceElement"sv) return FixedArray::create({ HTML::TagNames::source }); if (html_element_interface_name == "HTMLSpanElement"sv) return FixedArray::create({ HTML::TagNames::span }); if (html_element_interface_name == "HTMLStyleElement"sv) return FixedArray::create({ HTML::TagNames::style }); if (html_element_interface_name == "HTMLTableCaptionElement"sv) return FixedArray::create({ HTML::TagNames::caption }); if (html_element_interface_name == "HTMLTableCellElement"sv) return FixedArray::create({ HTML::TagNames::td, HTML::TagNames::th }); if (html_element_interface_name == "HTMLTableColElement"sv) return FixedArray::create({ HTML::TagNames::colgroup, HTML::TagNames::col }); if (html_element_interface_name == "HTMLTableElement"sv) return FixedArray::create({ HTML::TagNames::table }); if (html_element_interface_name == "HTMLTableSectionElement"sv) return FixedArray::create({ HTML::TagNames::tbody, HTML::TagNames::thead, HTML::TagNames::tfoot }); if (html_element_interface_name == "HTMLTemplateElement"sv) return FixedArray::create({ HTML::TagNames::template_ }); if (html_element_interface_name == "HTMLTextAreaElement"sv) return FixedArray::create({ HTML::TagNames::textarea }); if (html_element_interface_name == "HTMLTimeElement"sv) return FixedArray::create({ HTML::TagNames::time }); if (html_element_interface_name == "HTMLTitleElement"sv) return FixedArray::create({ HTML::TagNames::title }); if (html_element_interface_name == "HTMLTrackElement"sv) return FixedArray::create({ HTML::TagNames::track }); if (html_element_interface_name == "HTMLUListElement"sv) return FixedArray::create({ HTML::TagNames::ul }); if (html_element_interface_name == "HTMLVideoElement"sv) return FixedArray::create({ HTML::TagNames::video }); if (html_element_interface_name == "HTMLElement"sv) return FixedArray::create({ HTML::TagNames::article, HTML::TagNames::section, HTML::TagNames::nav, HTML::TagNames::aside, HTML::TagNames::hgroup, HTML::TagNames::header, HTML::TagNames::footer, HTML::TagNames::address, HTML::TagNames::dt, HTML::TagNames::dd, HTML::TagNames::figure, HTML::TagNames::figcaption, HTML::TagNames::main, HTML::TagNames::em, HTML::TagNames::strong, HTML::TagNames::small, HTML::TagNames::s, HTML::TagNames::cite, HTML::TagNames::dfn, HTML::TagNames::abbr, HTML::TagNames::ruby, HTML::TagNames::rt, HTML::TagNames::rp, HTML::TagNames::code, HTML::TagNames::var, HTML::TagNames::samp, HTML::TagNames::kbd, HTML::TagNames::sub, HTML::TagNames::sup, HTML::TagNames::i, HTML::TagNames::b, HTML::TagNames::u, HTML::TagNames::mark, HTML::TagNames::bdi, HTML::TagNames::bdo, HTML::TagNames::wbr, HTML::TagNames::summary, HTML::TagNames::noscript, HTML::TagNames::acronym, HTML::TagNames::basefont, HTML::TagNames::big, HTML::TagNames::center, HTML::TagNames::nobr, HTML::TagNames::noembed, HTML::TagNames::noframes, HTML::TagNames::plaintext, HTML::TagNames::rb, HTML::TagNames::rtc, HTML::TagNames::strike, HTML::TagNames::tt }); return FixedArray::create({}); } // https://html.spec.whatwg.org/multipage/dom.html#elements-in-the-dom%3Aelement-interface bool is_unknown_html_element(DeprecatedFlyString const& tag_name) { // NOTE: This is intentionally case-sensitive. // 1. If name is applet, bgsound, blink, isindex, keygen, multicol, nextid, or spacer, then return HTMLUnknownElement. if (tag_name.is_one_of(HTML::TagNames::applet, HTML::TagNames::bgsound, HTML::TagNames::blink, HTML::TagNames::isindex, HTML::TagNames::keygen, HTML::TagNames::multicol, HTML::TagNames::nextid, HTML::TagNames::spacer)) return true; // 2. If name is acronym, basefont, big, center, nobr, noembed, noframes, plaintext, rb, rtc, strike, or tt, then return HTMLElement. // 3. If name is listing or xmp, then return HTMLPreElement. // 4. Otherwise, if this specification defines an interface appropriate for the element type corresponding to the local name name, then return that interface. // 5. If other applicable specifications define an appropriate interface for name, then return the interface they define. #define __ENUMERATE_HTML_TAG(name) \ if (tag_name == HTML::TagNames::name) \ return false; ENUMERATE_HTML_TAGS #undef __ENUMERATE_HTML_TAG // 6. If name is a valid custom element name, then return HTMLElement. if (HTML::is_valid_custom_element_name(tag_name)) return false; // 7. Return HTMLUnknownElement. return true; } // https://html.spec.whatwg.org/#elements-in-the-dom:element-interface static WebIDL::ExceptionOr> create_html_element(JS::Realm& realm, Document& document, QualifiedName qualified_name) { auto lowercase_tag_name = qualified_name.local_name().to_lowercase(); if (lowercase_tag_name == HTML::TagNames::a) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::area) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::audio) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::base) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::blink) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::body) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::br) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::button) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::canvas) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::data) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::datalist) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::details) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::dialog) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::dir) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::div) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::dl) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::embed) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::fieldset) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::font) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::form) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::frame) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::frameset) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::head) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name.is_one_of(HTML::TagNames::h1, HTML::TagNames::h2, HTML::TagNames::h3, HTML::TagNames::h4, HTML::TagNames::h5, HTML::TagNames::h6)) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::hr) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::html) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::iframe) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::img) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::input) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::label) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::legend) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::li) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::link) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::map) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::marquee) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::menu) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::meta) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::meter) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name.is_one_of(HTML::TagNames::ins, HTML::TagNames::del)) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::object) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::ol) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::optgroup) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::option) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::output) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::p) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::param) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::picture) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); // NOTE: The obsolete elements "listing" and "xmp" are explicitly mapped to HTMLPreElement in the specification. if (lowercase_tag_name.is_one_of(HTML::TagNames::pre, HTML::TagNames::listing, HTML::TagNames::xmp)) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::progress) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name.is_one_of(HTML::TagNames::blockquote, HTML::TagNames::q)) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::script) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::select) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::slot) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::source) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::span) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::style) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::caption) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name.is_one_of(Web::HTML::TagNames::td, Web::HTML::TagNames::th)) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name.is_one_of(HTML::TagNames::colgroup, HTML::TagNames::col)) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::table) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::tr) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name.is_one_of(HTML::TagNames::tbody, HTML::TagNames::thead, HTML::TagNames::tfoot)) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::template_) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::textarea) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::time) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::title) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::track) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::ul) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name == HTML::TagNames::video) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (lowercase_tag_name.is_one_of( HTML::TagNames::article, HTML::TagNames::section, HTML::TagNames::nav, HTML::TagNames::aside, HTML::TagNames::hgroup, HTML::TagNames::header, HTML::TagNames::footer, HTML::TagNames::address, HTML::TagNames::dt, HTML::TagNames::dd, HTML::TagNames::figure, HTML::TagNames::figcaption, HTML::TagNames::main, HTML::TagNames::em, HTML::TagNames::strong, HTML::TagNames::small, HTML::TagNames::s, HTML::TagNames::cite, HTML::TagNames::dfn, HTML::TagNames::abbr, HTML::TagNames::ruby, HTML::TagNames::rt, HTML::TagNames::rp, HTML::TagNames::code, HTML::TagNames::var, HTML::TagNames::samp, HTML::TagNames::kbd, HTML::TagNames::sub, HTML::TagNames::sup, HTML::TagNames::i, HTML::TagNames::b, HTML::TagNames::u, HTML::TagNames::mark, HTML::TagNames::bdi, HTML::TagNames::bdo, HTML::TagNames::wbr, HTML::TagNames::summary, HTML::TagNames::noscript, // Obsolete HTML::TagNames::acronym, HTML::TagNames::basefont, HTML::TagNames::big, HTML::TagNames::center, HTML::TagNames::nobr, HTML::TagNames::noembed, HTML::TagNames::noframes, HTML::TagNames::plaintext, HTML::TagNames::rb, HTML::TagNames::rtc, HTML::TagNames::strike, HTML::TagNames::tt)) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (HTML::is_valid_custom_element_name(qualified_name.local_name())) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); } static WebIDL::ExceptionOr> create_svg_element(JS::Realm& realm, Document& document, QualifiedName qualified_name) { auto const& local_name = qualified_name.local_name(); if (local_name == SVG::TagNames::svg) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); // FIXME: Support SVG's mixedCase tag names properly. if (local_name.equals_ignoring_ascii_case(SVG::TagNames::clipPath)) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (local_name == SVG::TagNames::circle) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (local_name.equals_ignoring_ascii_case(SVG::TagNames::defs)) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (local_name == SVG::TagNames::ellipse) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (local_name.equals_ignoring_ascii_case(SVG::TagNames::foreignObject)) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (local_name == SVG::TagNames::line) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (local_name == SVG::TagNames::linearGradient) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (local_name == SVG::TagNames::path) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (local_name == SVG::TagNames::polygon) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (local_name == SVG::TagNames::polyline) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (local_name == SVG::TagNames::radialGradient) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (local_name == SVG::TagNames::rect) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (local_name == SVG::TagNames::g) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (local_name == SVG::TagNames::stop) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); if (local_name == SVG::TagNames::text) return MUST_OR_THROW_OOM(realm.heap().allocate(realm, document, move(qualified_name))); return nullptr; } // https://dom.spec.whatwg.org/#concept-create-element WebIDL::ExceptionOr> create_element(Document& document, DeprecatedFlyString local_name, DeprecatedFlyString namespace_, DeprecatedFlyString prefix, Optional is_value, bool synchronous_custom_elements_flag) { auto& realm = document.realm(); // 1. If prefix was not given, let prefix be null. // NOTE: This is already taken care of by `prefix` having a default value. // 2. If is was not given, let is be null. // NOTE: This is already taken care of by `is` having a default value. // 3. Let result be null. // NOTE: We collapse this into just returning an element where necessary. // 4. Let definition be the result of looking up a custom element definition given document, namespace, localName, and is. auto definition = document.lookup_custom_element_definition(namespace_, local_name, is_value); // 5. If definition is non-null, and definition’s name is not equal to its local name (i.e., definition represents a customized built-in element), then: if (definition && definition->name() != definition->local_name()) { // 1. Let interface be the element interface for localName and the HTML namespace. // 2. Set result to a new element that implements interface, with no attributes, namespace set to the HTML namespace, // namespace prefix set to prefix, local name set to localName, custom element state set to "undefined", custom element definition set to null, // is value set to is, and node document set to document. auto element = TRY(create_html_element(realm, document, QualifiedName { local_name, prefix, Namespace::HTML })); // 3. If the synchronous custom elements flag is set, then run this step while catching any exceptions: if (synchronous_custom_elements_flag) { // 1. Upgrade element using definition. auto upgrade_result = element->upgrade_element(*definition); // If this step threw an exception, then: if (upgrade_result.is_throw_completion()) { // 1. Report the exception. HTML::report_exception(upgrade_result, realm); // 2. Set result’s custom element state to "failed". element->set_custom_element_state(CustomElementState::Failed); } } // 4. Otherwise, enqueue a custom element upgrade reaction given result and definition. else { element->enqueue_a_custom_element_upgrade_reaction(*definition); } return element; } // 6. Otherwise, if definition is non-null, then: if (definition) { // 1. If the synchronous custom elements flag is set, then run these steps while catching any exceptions: if (synchronous_custom_elements_flag) { auto synchronously_upgrade_custom_element = [&]() -> JS::ThrowCompletionOr> { auto& vm = document.vm(); // 1. Let C be definition’s constructor. auto& constructor = definition->constructor(); // 2. Set result to the result of constructing C, with no arguments. auto result = TRY(WebIDL::construct(constructor)); // FIXME: 3. Assert: result’s custom element state and custom element definition are initialized. // FIXME: 4. Assert: result’s namespace is the HTML namespace. // Spec Note: IDL enforces that result is an HTMLElement object, which all use the HTML namespace. // IDL does not currently convert the object for us, so we will have to do it here. if (!result.has_value() || !result->is_object() || !is(result->as_object())) return vm.throw_completion(JS::ErrorType::NotAnObjectOfType, "HTMLElement"sv); JS::NonnullGCPtr element = verify_cast(result->as_object()); // 5. If result’s attribute list is not empty, then throw a "NotSupportedError" DOMException. if (element->has_attributes()) return JS::throw_completion(WebIDL::NotSupportedError::create(realm, "Synchronously created custom element cannot have attributes"sv)); // 6. If result has children, then throw a "NotSupportedError" DOMException. if (element->has_children()) return JS::throw_completion(WebIDL::NotSupportedError::create(realm, "Synchronously created custom element cannot have children"sv)); // 7. If result’s parent is not null, then throw a "NotSupportedError" DOMException. if (element->parent()) return JS::throw_completion(WebIDL::NotSupportedError::create(realm, "Synchronously created custom element cannot have a parent"sv)); // 8. If result’s node document is not document, then throw a "NotSupportedError" DOMException. if (&element->document() != &document) return JS::throw_completion(WebIDL::NotSupportedError::create(realm, "Synchronously created custom element must be in the same document that element creation was invoked in"sv)); // 9. If result’s local name is not equal to localName, then throw a "NotSupportedError" DOMException. if (element->local_name() != local_name) return JS::throw_completion(WebIDL::NotSupportedError::create(realm, "Synchronously created custom element must have the same local name that element creation was invoked with"sv)); // 10. Set result’s namespace prefix to prefix. element->set_prefix(prefix); // 11. Set result’s is value to null. element->set_is_value(Optional {}); return element; }; auto result = synchronously_upgrade_custom_element(); // If any of these steps threw an exception, then: if (result.is_throw_completion()) { // 1. Report the exception. HTML::report_exception(result, realm); // 2. Set result to a new element that implements the HTMLUnknownElement interface, with no attributes, namespace set to the HTML namespace, namespace prefix set to prefix, // local name set to localName, custom element state set to "failed", custom element definition set to null, is value set to null, and node document set to document. JS::NonnullGCPtr element = realm.heap().allocate(realm, document, QualifiedName { local_name, prefix, Namespace::HTML }).release_allocated_value_but_fixme_should_propagate_errors(); element->set_custom_element_state(CustomElementState::Failed); return element; } return result.release_value(); } // 2. Otherwise: // 1. Set result to a new element that implements the HTMLElement interface, with no attributes, namespace set to the HTML namespace, namespace prefix set to prefix, // local name set to localName, custom element state set to "undefined", custom element definition set to null, is value set to null, and node document set to document. JS::NonnullGCPtr element = realm.heap().allocate(realm, document, QualifiedName { local_name, prefix, Namespace::HTML }).release_allocated_value_but_fixme_should_propagate_errors(); element->set_custom_element_state(CustomElementState::Undefined); // 2. Enqueue a custom element upgrade reaction given result and definition. element->enqueue_a_custom_element_upgrade_reaction(*definition); return element; } // 7. Otherwise: // 1. Let interface be the element interface for localName and namespace. // 2. Set result to a new element that implements interface, with no attributes, namespace set to namespace, namespace prefix set to prefix, // local name set to localName, custom element state set to "uncustomized", custom element definition set to null, is value set to is, // and node document set to document. auto qualified_name = QualifiedName { local_name, prefix, namespace_ }; if (namespace_ == Namespace::HTML) { auto element = TRY(create_html_element(realm, document, move(qualified_name))); element->set_is_value(move(is_value)); element->set_custom_element_state(CustomElementState::Uncustomized); // 3. If namespace is the HTML namespace, and either localName is a valid custom element name or is is non-null, // then set result’s custom element state to "undefined". if (HTML::is_valid_custom_element_name(local_name) || is_value.has_value()) element->set_custom_element_state(CustomElementState::Undefined); return element; } if (namespace_ == Namespace::SVG) { auto element = TRY(create_svg_element(realm, document, qualified_name)); if (element) { element->set_is_value(move(is_value)); element->set_custom_element_state(CustomElementState::Uncustomized); return JS::NonnullGCPtr { *element }; } } // 8. Return result. // NOTE: See step 3. // https://dom.spec.whatwg.org/#concept-element-interface // The element interface for any name and namespace is Element, unless stated otherwise. dbgln("Potential FIXME: Creating unknown generic element '{}' in namespace '{}'", local_name, namespace_); JS::NonnullGCPtr element = realm.heap().allocate(realm, document, move(qualified_name)).release_allocated_value_but_fixme_should_propagate_errors(); element->set_is_value(move(is_value)); element->set_custom_element_state(CustomElementState::Uncustomized); return element; } }