/* * Copyright (c) 2023-2024, Shannon Booth * Copyright (c) 2024, Pavel Shliak * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include #include #include #include #include namespace Web::SVG { GC_DEFINE_ALLOCATOR(SVGScriptElement); SVGScriptElement::SVGScriptElement(DOM::Document& document, DOM::QualifiedName qualified_name) : SVGElement(document, move(qualified_name)) { } void SVGScriptElement::initialize(JS::Realm& realm) { Base::initialize(realm); WEB_SET_PROTOTYPE_FOR_INTERFACE(SVGScriptElement); } void SVGScriptElement::visit_edges(Cell::Visitor& visitor) { Base::visit_edges(visitor); SVGURIReferenceMixin::visit_edges(visitor); visitor.visit(m_script); } // https://www.w3.org/TR/SVGMobile12/script.html#ScriptContentProcessing void SVGScriptElement::process_the_script_element() { // 1. If the 'script' element's "already processed" flag is true or if the element is not in the // document tree, then no action is performed and these steps are ended. if (m_already_processed || !in_a_document_tree()) return; IGNORE_USE_IN_ESCAPING_LAMBDA String script_content; auto script_url = m_document->url(); // 2. If the 'script' element references external script content, then the external script content // using the current value of the 'xlink:href' attribute is fetched. Further processing of the // 'script' element is dependent on the external script content, and will block here until the // resource has been fetched or is determined to be an invalid IRI reference. if (has_attribute(SVG::AttributeNames::href) || has_attribute_ns(Namespace::XLink.to_string(), SVG::AttributeNames::href)) { auto href_value = href()->base_val(); script_url = document().parse_url(href_value); if (!script_url.is_valid()) { dbgln("Invalid script URL: {}", href_value); return; } auto& vm = realm().vm(); auto request = Fetch::Infrastructure::Request::create(vm); request->set_url(script_url); request->set_destination(Fetch::Infrastructure::Request::Destination::Script); // FIXME: Use CORS state specified by the ‘crossorigin’ attribute. request->set_mode(Fetch::Infrastructure::Request::Mode::NoCORS); request->set_credentials_mode(Fetch::Infrastructure::Request::CredentialsMode::SameOrigin); request->set_client(&document().relevant_settings_object()); IGNORE_USE_IN_ESCAPING_LAMBDA bool fetch_done = false; Fetch::Infrastructure::FetchAlgorithms::Input fetch_algorithms_input {}; fetch_algorithms_input.process_response = [this, &script_content, &fetch_done](GC::Ref response) { if (response->is_network_error()) { dbgln("Failed to fetch SVG external script."); fetch_done = true; return; } auto& realm = this->realm(); auto& global = document().realm().global_object(); auto on_data_read = GC::create_function(realm.heap(), [&script_content, &fetch_done](ByteBuffer data) { auto content_or_error = String::from_utf8(data); if (content_or_error.is_error()) { dbgln("Failed to decode script content as UTF-8"); } else { script_content = content_or_error.release_value(); } fetch_done = true; }); auto on_error = GC::create_function(realm.heap(), [&fetch_done](JS::Value) { dbgln("Error occurred while reading script data."); fetch_done = true; }); VERIFY(response->body()); response->body()->fully_read(realm, on_data_read, on_error, GC::Ref { global }); }; auto fetch_promise = Fetch::Fetching::fetch(realm(), request, Fetch::Infrastructure::FetchAlgorithms::create(vm, move(fetch_algorithms_input))); // Block until the resource has been fetched or determined invalid HTML::main_thread_event_loop().spin_until(GC::create_function(heap(), [&] { return fetch_done; })); if (script_content.is_empty()) { // Failed to fetch or decode return; } } else { // Inline script content script_content = child_text_content(); } // 3. The 'script' element's "already processed" flag is set to true. m_already_processed = true; // 4. If the script content is inline, or if it is external and was fetched successfully, then the // script is executed. Note that at this point, these steps may be re-entrant if the execution // of the script results in further 'script' elements being inserted into the document. // https://html.spec.whatwg.org/multipage/document-lifecycle.html#read-html // Before any script execution occurs, the user agent must wait for scripts may run for the newly-created document to be true for document. if (!m_document->ready_to_run_scripts()) HTML::main_thread_event_loop().spin_until(GC::create_function(heap(), [&] { return m_document->ready_to_run_scripts(); })); m_script = HTML::ClassicScript::create(script_url.basename(), script_content, realm(), m_document->base_url(), m_source_line_number); // FIXME: Note that a load event is dispatched on a 'script' element once it has been processed, // unless it referenced external script content with an invalid IRI reference and 'externalResourcesRequired' was set to 'true'. (void)m_script->run(); } }