mirror of
				https://github.com/LadybirdBrowser/ladybird.git
				synced 2025-10-25 18:34:14 +00:00 
			
		
		
		
	 a6dfc74e93
			
		
	
	
		a6dfc74e93
		
	
	
	
	
		
			
			Before this change, we were going through the chain of base classes for each IDL interface object and having them set the prototype to their prototype. Instead of doing that, reorder things so that we set the right prototype immediately in Foo::initialize(), and then don't bother in all the base class overrides. This knocks off a ~1% profile item on Speedometer 3.
		
			
				
	
	
		
			294 lines
		
	
	
	
		
			12 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			294 lines
		
	
	
	
		
			12 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | ||
|  * Copyright (c) 2024, Andrew Kaster <akaster@serenityos.org>
 | ||
|  * Copyright (c) 2024, Jamie Mansfield <jmansfield@cadixdev.org>
 | ||
|  *
 | ||
|  * SPDX-License-Identifier: BSD-2-Clause
 | ||
|  */
 | ||
| 
 | ||
| #include <LibGC/Heap.h>
 | ||
| #include <LibJS/Runtime/Realm.h>
 | ||
| #include <LibJS/Runtime/Set.h>
 | ||
| #include <LibWeb/Bindings/ExceptionOrUtils.h>
 | ||
| #include <LibWeb/Bindings/FontFaceSetPrototype.h>
 | ||
| #include <LibWeb/Bindings/Intrinsics.h>
 | ||
| #include <LibWeb/CSS/FontFace.h>
 | ||
| #include <LibWeb/CSS/FontFaceSet.h>
 | ||
| #include <LibWeb/CSS/Parser/Parser.h>
 | ||
| #include <LibWeb/CSS/StyleValues/CSSKeywordValue.h>
 | ||
| #include <LibWeb/CSS/StyleValues/ShorthandStyleValue.h>
 | ||
| #include <LibWeb/CSS/StyleValues/StringStyleValue.h>
 | ||
| #include <LibWeb/CSS/StyleValues/StyleValueList.h>
 | ||
| #include <LibWeb/DOM/Document.h>
 | ||
| #include <LibWeb/HTML/EventNames.h>
 | ||
| #include <LibWeb/HTML/Scripting/TemporaryExecutionContext.h>
 | ||
| #include <LibWeb/Platform/EventLoopPlugin.h>
 | ||
| #include <LibWeb/WebIDL/Promise.h>
 | ||
| 
 | ||
| namespace Web::CSS {
 | ||
| 
 | ||
| GC_DEFINE_ALLOCATOR(FontFaceSet);
 | ||
| 
 | ||
| // https://drafts.csswg.org/css-font-loading/#dom-fontfaceset-fontfaceset
 | ||
| GC::Ref<FontFaceSet> FontFaceSet::construct_impl(JS::Realm& realm, Vector<GC::Root<FontFace>> const& initial_faces)
 | ||
| {
 | ||
|     auto ready_promise = WebIDL::create_promise(realm);
 | ||
|     auto set_entries = JS::Set::create(realm);
 | ||
| 
 | ||
|     // The FontFaceSet constructor, when called, must iterate its initialFaces argument and add each value to its set entries.
 | ||
|     for (auto const& face : initial_faces)
 | ||
|         set_entries->set_add(face);
 | ||
| 
 | ||
|     return realm.create<FontFaceSet>(realm, ready_promise, set_entries);
 | ||
| }
 | ||
| 
 | ||
| GC::Ref<FontFaceSet> FontFaceSet::create(JS::Realm& realm)
 | ||
| {
 | ||
|     return construct_impl(realm, {});
 | ||
| }
 | ||
| 
 | ||
| FontFaceSet::FontFaceSet(JS::Realm& realm, GC::Ref<WebIDL::Promise> ready_promise, GC::Ref<JS::Set> set_entries)
 | ||
|     : DOM::EventTarget(realm)
 | ||
|     , m_set_entries(set_entries)
 | ||
|     , m_ready_promise(ready_promise)
 | ||
|     , m_status(Bindings::FontFaceSetLoadStatus::Loaded)
 | ||
| {
 | ||
| }
 | ||
| 
 | ||
| void FontFaceSet::initialize(JS::Realm& realm)
 | ||
| {
 | ||
|     WEB_SET_PROTOTYPE_FOR_INTERFACE(FontFaceSet);
 | ||
|     Base::initialize(realm);
 | ||
| }
 | ||
| 
 | ||
| void FontFaceSet::visit_edges(Cell::Visitor& visitor)
 | ||
| {
 | ||
|     Base::visit_edges(visitor);
 | ||
|     visitor.visit(m_set_entries);
 | ||
|     visitor.visit(m_ready_promise);
 | ||
|     visitor.visit(m_loading_fonts);
 | ||
|     visitor.visit(m_loaded_fonts);
 | ||
|     visitor.visit(m_failed_fonts);
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.csswg.org/css-font-loading/#dom-fontfaceset-add
 | ||
| WebIDL::ExceptionOr<GC::Ref<FontFaceSet>>
 | ||
| FontFaceSet::add(GC::Root<FontFace> face)
 | ||
| {
 | ||
|     // 1. If font is already in the FontFaceSet’s set entries, skip to the last step of this algorithm immediately.
 | ||
|     if (m_set_entries->set_has(face))
 | ||
|         return GC::Ref<FontFaceSet>(*this);
 | ||
| 
 | ||
|     // 2. If font is CSS-connected, throw an InvalidModificationError exception and exit this algorithm immediately.
 | ||
|     if (face->is_css_connected()) {
 | ||
|         return WebIDL::InvalidModificationError::create(realm(), "Cannot add a CSS-connected FontFace to a FontFaceSet"_string);
 | ||
|     }
 | ||
| 
 | ||
|     // 3. Add the font argument to the FontFaceSet’s set entries.
 | ||
|     m_set_entries->set_add(face);
 | ||
| 
 | ||
|     // 4. If font’s status attribute is "loading"
 | ||
|     if (face->status() == Bindings::FontFaceLoadStatus::Loading) {
 | ||
| 
 | ||
|         // 1. If the FontFaceSet’s [[LoadingFonts]] list is empty, switch the FontFaceSet to loading.
 | ||
|         if (m_loading_fonts.is_empty()) {
 | ||
|             m_status = Bindings::FontFaceSetLoadStatus::Loading;
 | ||
|         }
 | ||
| 
 | ||
|         // 2. Append font to the FontFaceSet’s [[LoadingFonts]] list.
 | ||
|         m_loading_fonts.append(*face);
 | ||
|     }
 | ||
| 
 | ||
|     // 5. Return the FontFaceSet.
 | ||
|     return GC::Ref<FontFaceSet>(*this);
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.csswg.org/css-font-loading/#dom-fontfaceset-delete
 | ||
| bool FontFaceSet::delete_(GC::Root<FontFace> face)
 | ||
| {
 | ||
|     // 1. If font is CSS-connected, return false and exit this algorithm immediately.
 | ||
|     if (face->is_css_connected()) {
 | ||
|         return false;
 | ||
|     }
 | ||
| 
 | ||
|     // 2. Let deleted be the result of removing font from the FontFaceSet’s set entries.
 | ||
|     bool deleted = m_set_entries->set_remove(face);
 | ||
| 
 | ||
|     // 3. If font is present in the FontFaceSet’s [[LoadedFonts]], or [[FailedFonts]] lists, remove it.
 | ||
|     m_loaded_fonts.remove_all_matching([face](auto const& entry) { return entry == face; });
 | ||
|     m_failed_fonts.remove_all_matching([face](auto const& entry) { return entry == face; });
 | ||
| 
 | ||
|     // 4. If font is present in the FontFaceSet’s [[LoadingFonts]] list, remove it. If font was the last item in that list (and so the list is now empty), switch the FontFaceSet to loaded.
 | ||
|     m_loading_fonts.remove_all_matching([face](auto const& entry) { return entry == face; });
 | ||
| 
 | ||
|     if (m_loading_fonts.is_empty()) {
 | ||
|         m_status = Bindings::FontFaceSetLoadStatus::Loaded;
 | ||
|     }
 | ||
| 
 | ||
|     return deleted;
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.csswg.org/css-font-loading/#dom-fontfaceset-clear
 | ||
| void FontFaceSet::clear()
 | ||
| {
 | ||
|     // FIXME: Do the actual spec steps
 | ||
|     m_set_entries->set_clear();
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.csswg.org/css-font-loading/#dom-fontfaceset-onloading
 | ||
| void FontFaceSet::set_onloading(WebIDL::CallbackType* event_handler)
 | ||
| {
 | ||
|     set_event_handler_attribute(HTML::EventNames::loading, event_handler);
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.csswg.org/css-font-loading/#dom-fontfaceset-onloading
 | ||
| WebIDL::CallbackType* FontFaceSet::onloading()
 | ||
| {
 | ||
|     return event_handler_attribute(HTML::EventNames::loading);
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.csswg.org/css-font-loading/#dom-fontfaceset-onloadingdone
 | ||
| void FontFaceSet::set_onloadingdone(WebIDL::CallbackType* event_handler)
 | ||
| {
 | ||
|     set_event_handler_attribute(HTML::EventNames::loadingdone, event_handler);
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.csswg.org/css-font-loading/#dom-fontfaceset-onloadingdone
 | ||
| WebIDL::CallbackType* FontFaceSet::onloadingdone()
 | ||
| {
 | ||
|     return event_handler_attribute(HTML::EventNames::loadingdone);
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.csswg.org/css-font-loading/#dom-fontfaceset-onloadingerror
 | ||
| void FontFaceSet::set_onloadingerror(WebIDL::CallbackType* event_handler)
 | ||
| {
 | ||
|     set_event_handler_attribute(HTML::EventNames::loadingerror, event_handler);
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.csswg.org/css-font-loading/#dom-fontfaceset-onloadingerror
 | ||
| WebIDL::CallbackType* FontFaceSet::onloadingerror()
 | ||
| {
 | ||
|     return event_handler_attribute(HTML::EventNames::loadingerror);
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.csswg.org/css-font-loading/#find-the-matching-font-faces
 | ||
| static WebIDL::ExceptionOr<GC::Ref<JS::Set>> find_matching_font_faces(JS::Realm& realm, FontFaceSet& font_face_set, String const& font, String const&)
 | ||
| {
 | ||
|     // 1. Parse font using the CSS value syntax of the font property. If a syntax error occurs, return a syntax error.
 | ||
|     auto property = parse_css_value(CSS::Parser::ParsingParams(), font, PropertyID::Font);
 | ||
|     if (!property)
 | ||
|         return WebIDL::SyntaxError::create(realm, "Unable to parse font"_string);
 | ||
| 
 | ||
|     // If the parsed value is a CSS-wide keyword, return a syntax error.
 | ||
|     if (property->is_css_wide_keyword())
 | ||
|         return WebIDL::SyntaxError::create(realm, "Parsed font is a CSS-wide keyword"_string);
 | ||
| 
 | ||
|     // FIXME: Absolutize all relative lengths against the initial values of the corresponding properties. (For example, a
 | ||
|     //        relative font weight like bolder is evaluated against the initial value normal.)
 | ||
| 
 | ||
|     // FIXME: 2. If text was not explicitly provided, let it be a string containing a single space character (U+0020 SPACE).
 | ||
| 
 | ||
|     // 3. Let font family list be the list of font families parsed from font, and font style be the other font style
 | ||
|     //    attributes parsed from font.
 | ||
|     auto const& font_family_list = property->as_shorthand().longhand(PropertyID::FontFamily)->as_value_list();
 | ||
| 
 | ||
|     // 4. Let available font faces be the available font faces within source. If the allow system fonts flag is specified,
 | ||
|     //    add all system fonts to available font faces.
 | ||
|     auto available_font_faces = font_face_set.set_entries();
 | ||
| 
 | ||
|     // 5. Let matched font faces initially be an empty list.
 | ||
|     auto matched_font_faces = JS::Set::create(realm);
 | ||
| 
 | ||
|     // 6. For each family in font family list, use the font matching rules to select the font faces from available font
 | ||
|     //    faces that match the font style, and add them to matched font faces. The use of the unicodeRange attribute means
 | ||
|     //    that this may be more than just a single font face.
 | ||
|     for (auto const& font_family : font_family_list.values()) {
 | ||
|         // FIXME: The matching below is super basic. We currently just match font family names by their string value.
 | ||
|         if (!font_family->is_string())
 | ||
|             continue;
 | ||
| 
 | ||
|         auto const& font_family_name = font_family->as_string().string_value();
 | ||
| 
 | ||
|         for (auto font_face_value : *available_font_faces) {
 | ||
|             auto& font_face = as<FontFace>(font_face_value.key.as_object());
 | ||
|             if (font_face.family() != font_family_name)
 | ||
|                 continue;
 | ||
| 
 | ||
|             matched_font_faces->set_add(font_face_value.key);
 | ||
|         }
 | ||
|     }
 | ||
| 
 | ||
|     // FIXME: 7. If matched font faces is empty, set the found faces flag to false. Otherwise, set it to true.
 | ||
|     // FIXME: 8. For each font face in matched font faces, if its defined unicode-range does not include the codepoint of at
 | ||
|     //           least one character in text, remove it from the list.
 | ||
| 
 | ||
|     // 9. Return matched font faces and the found faces flag.
 | ||
|     return matched_font_faces;
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.csswg.org/css-font-loading/#dom-fontfaceset-load
 | ||
| JS::ThrowCompletionOr<GC::Ref<WebIDL::Promise>> FontFaceSet::load(String const& font, String const& text)
 | ||
| {
 | ||
|     auto& realm = this->realm();
 | ||
| 
 | ||
|     // 1. Let font face set be the FontFaceSet object this method was called on. Let promise be a newly-created promise object.
 | ||
|     GC::Ref font_face_set = *this;
 | ||
|     auto promise = WebIDL::create_promise(realm);
 | ||
| 
 | ||
|     Platform::EventLoopPlugin::the().deferred_invoke(GC::create_function(realm.heap(), [&realm, font_face_set, promise, font, text]() mutable {
 | ||
|         // 3. Find the matching font faces from font face set using the font and text arguments passed to the function,
 | ||
|         //    and let font face list be the return value (ignoring the found faces flag). If a syntax error was returned,
 | ||
|         //    reject promise with a SyntaxError exception and terminate these steps.
 | ||
|         auto result = find_matching_font_faces(realm, font_face_set, font, text);
 | ||
|         if (result.is_error()) {
 | ||
|             HTML::TemporaryExecutionContext execution_context { realm, HTML::TemporaryExecutionContext::CallbacksEnabled::Yes };
 | ||
|             WebIDL::reject_promise(realm, promise, Bindings::exception_to_throw_completion(realm.vm(), result.release_error()).release_value());
 | ||
|             return;
 | ||
|         }
 | ||
| 
 | ||
|         auto matched_font_faces = result.release_value();
 | ||
| 
 | ||
|         // 4. Queue a task to run the following steps synchronously:
 | ||
|         HTML::queue_a_task(HTML::Task::Source::FontLoading, nullptr, nullptr, GC::create_function(realm.heap(), [&realm, promise, matched_font_faces] {
 | ||
|             GC::RootVector<GC::Ref<WebIDL::Promise>> promises(realm.heap());
 | ||
| 
 | ||
|             // 1. For all of the font faces in the font face list, call their load() method.
 | ||
|             for (auto font_face_value : *matched_font_faces) {
 | ||
|                 auto& font_face = as<FontFace>(font_face_value.key.as_object());
 | ||
|                 font_face.load();
 | ||
| 
 | ||
|                 promises.append(font_face.font_status_promise());
 | ||
|             }
 | ||
| 
 | ||
|             // 2. Resolve promise with the result of waiting for all of the [[FontStatusPromise]]s of each font face in
 | ||
|             //    the font face list, in order.
 | ||
|             HTML::TemporaryExecutionContext execution_context { realm, HTML::TemporaryExecutionContext::CallbacksEnabled::Yes };
 | ||
| 
 | ||
|             WebIDL::wait_for_all(
 | ||
|                 realm, promises,
 | ||
|                 [&realm, promise](auto const&) {
 | ||
|                     HTML::TemporaryExecutionContext execution_context { realm, HTML::TemporaryExecutionContext::CallbacksEnabled::Yes };
 | ||
|                     WebIDL::resolve_promise(realm, promise);
 | ||
|                 },
 | ||
|                 [&realm, promise](auto error) {
 | ||
|                     HTML::TemporaryExecutionContext execution_context { realm, HTML::TemporaryExecutionContext::CallbacksEnabled::Yes };
 | ||
|                     WebIDL::reject_promise(realm, promise, error);
 | ||
|                 });
 | ||
|         }));
 | ||
|     }));
 | ||
| 
 | ||
|     // 2. Return promise. Complete the rest of these steps asynchronously.
 | ||
|     return promise;
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.csswg.org/css-font-loading/#font-face-set-ready
 | ||
| GC::Ref<WebIDL::Promise> FontFaceSet::ready() const
 | ||
| {
 | ||
|     return m_ready_promise;
 | ||
| }
 | ||
| 
 | ||
| void FontFaceSet::resolve_ready_promise()
 | ||
| {
 | ||
|     WebIDL::resolve_promise(realm(), m_ready_promise);
 | ||
| }
 | ||
| 
 | ||
| }
 |