mirror of
https://github.com/LadybirdBrowser/ladybird.git
synced 2026-04-19 18:30:27 +00:00
As FontFaces are added or removed from a FontFaceSet, and as they load or fail, the FontFaceSet moves them between a few different lists, and updates its loading/loaded status. In the spec, this is how the FontFaceSet.[[ReadyPromise]] gets fulfilled: When the document has finished loading and FontFaceSet.[[LoadingFonts]] is empty, it resolves the promise. To support this, FontFace now keeps a set of FontFaceSets that it is contained in. This lets us remove the non-spec resolve_ready_promise() call in EventLoop which was sometimes triggering before any fonts had attempted to load. As noted, there's a spec issue with the ready promise: If nothing modifies the document's fonts, then it would never resolve. My ad-hoc fix is to also switch the FontFaceSet to the loaded state if it is empty, which appears to solve the issues but is not ideal.
80 lines
2.6 KiB
C++
80 lines
2.6 KiB
C++
/*
|
|
* Copyright (c) 2024, Andrew Kaster <akaster@serenityos.org>
|
|
* Copyright (c) 2024, Jamie Mansfield <jmansfield@cadixdev.org>
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include <LibJS/Runtime/Set.h>
|
|
#include <LibJS/Runtime/SetIterator.h>
|
|
#include <LibWeb/Bindings/FontFaceSetPrototype.h>
|
|
#include <LibWeb/Bindings/PlatformObject.h>
|
|
#include <LibWeb/CSS/FontFace.h>
|
|
#include <LibWeb/DOM/EventTarget.h>
|
|
|
|
namespace Web::CSS {
|
|
|
|
class FontFaceSet final : public DOM::EventTarget {
|
|
WEB_PLATFORM_OBJECT(FontFaceSet, DOM::EventTarget);
|
|
GC_DECLARE_ALLOCATOR(FontFaceSet);
|
|
|
|
public:
|
|
[[nodiscard]] static GC::Ref<FontFaceSet> create(JS::Realm&);
|
|
virtual ~FontFaceSet() override = default;
|
|
|
|
GC::Ref<JS::Set> set_entries() const { return m_set_entries; }
|
|
|
|
WebIDL::ExceptionOr<GC::Ref<FontFaceSet>> add(GC::Root<FontFace>);
|
|
bool delete_(GC::Root<FontFace>);
|
|
void clear();
|
|
|
|
void add_css_connected_font(GC::Ref<FontFace>);
|
|
|
|
void set_onloading(WebIDL::CallbackType*);
|
|
WebIDL::CallbackType* onloading();
|
|
void set_onloadingdone(WebIDL::CallbackType*);
|
|
WebIDL::CallbackType* onloadingdone();
|
|
void set_onloadingerror(WebIDL::CallbackType*);
|
|
WebIDL::CallbackType* onloadingerror();
|
|
|
|
JS::ThrowCompletionOr<GC::Ref<WebIDL::Promise>> load(String const& font, String const& text);
|
|
|
|
Vector<GC::Ref<FontFace>>& loading_fonts() { return m_loading_fonts; }
|
|
Vector<GC::Ref<FontFace>>& loaded_fonts() { return m_loaded_fonts; }
|
|
Vector<GC::Ref<FontFace>>& failed_fonts() { return m_failed_fonts; }
|
|
|
|
GC::Ref<WebIDL::Promise> ready() const;
|
|
Bindings::FontFaceSetLoadStatus status() const { return m_status; }
|
|
|
|
void on_set_modified_from_js(Badge<Bindings::FontFaceSetPrototype>) { }
|
|
|
|
void fire_a_font_load_event(FlyString name, Vector<GC::Ref<FontFace>> = {});
|
|
void set_is_pending_on_the_environment(bool);
|
|
|
|
void switch_to_loading();
|
|
void switch_to_loaded();
|
|
|
|
private:
|
|
explicit FontFaceSet(JS::Realm&);
|
|
|
|
virtual void initialize(JS::Realm&) override;
|
|
virtual void visit_edges(Cell::Visitor&) override;
|
|
|
|
GC::Ref<JS::Set> m_set_entries;
|
|
GC::Ref<WebIDL::Promise> m_ready_promise; // [[ReadyPromise]]
|
|
|
|
Vector<GC::Ref<FontFace>> m_loading_fonts {}; // [[LoadingFonts]]
|
|
Vector<GC::Ref<FontFace>> m_loaded_fonts {}; // [[LoadedFonts]]
|
|
Vector<GC::Ref<FontFace>> m_failed_fonts {}; // [[FailedFonts]]
|
|
|
|
Bindings::FontFaceSetLoadStatus m_status { Bindings::FontFaceSetLoadStatus::Loaded };
|
|
|
|
bool m_is_pending_on_the_environment { true };
|
|
|
|
// https://drafts.csswg.org/css-font-loading/#fontfaceset-stuck-on-the-environment
|
|
bool m_is_stuck_on_the_environment { false };
|
|
};
|
|
|
|
}
|