mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-31 03:42:44 +00:00 
			
		
		
		
	 0e2684b10f
			
		
	
	
		0e2684b10f
		
	
	
	
	
		
			
			- The check for `local()` sources wasn't working, and output `local()` every time. Since we don't parse `local()` yet, let's just always output a regular URL source. - Put a space between the URL and the `format()`. - Stop double-quoting the format string.
		
			
				
	
	
		
			143 lines
		
	
	
	
		
			6.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			143 lines
		
	
	
	
		
			6.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | ||
|  * Copyright (c) 2022-2023, Sam Atkins <atkinssj@serenityos.org>
 | ||
|  * Copyright (c) 2022-2023, Andreas Kling <kling@serenityos.org>
 | ||
|  *
 | ||
|  * SPDX-License-Identifier: BSD-2-Clause
 | ||
|  */
 | ||
| 
 | ||
| #include <LibGfx/Font/FontStyleMapping.h>
 | ||
| #include <LibWeb/Bindings/CSSFontFaceRulePrototype.h>
 | ||
| #include <LibWeb/Bindings/Intrinsics.h>
 | ||
| #include <LibWeb/CSS/CSSFontFaceRule.h>
 | ||
| #include <LibWeb/CSS/Serialize.h>
 | ||
| #include <LibWeb/WebIDL/ExceptionOr.h>
 | ||
| 
 | ||
| namespace Web::CSS {
 | ||
| 
 | ||
| WebIDL::ExceptionOr<JS::NonnullGCPtr<CSSFontFaceRule>> CSSFontFaceRule::create(JS::Realm& realm, FontFace&& font_face)
 | ||
| {
 | ||
|     return MUST_OR_THROW_OOM(realm.heap().allocate<CSSFontFaceRule>(realm, realm, move(font_face)));
 | ||
| }
 | ||
| 
 | ||
| CSSFontFaceRule::CSSFontFaceRule(JS::Realm& realm, FontFace&& font_face)
 | ||
|     : CSSRule(realm)
 | ||
|     , m_font_face(move(font_face))
 | ||
| {
 | ||
| }
 | ||
| 
 | ||
| JS::ThrowCompletionOr<void> CSSFontFaceRule::initialize(JS::Realm& realm)
 | ||
| {
 | ||
|     MUST_OR_THROW_OOM(Base::initialize(realm));
 | ||
|     set_prototype(&Bindings::ensure_web_prototype<Bindings::CSSFontFaceRulePrototype>(realm, "CSSFontFaceRule"));
 | ||
| 
 | ||
|     return {};
 | ||
| }
 | ||
| 
 | ||
| CSSStyleDeclaration* CSSFontFaceRule::style()
 | ||
| {
 | ||
|     // FIXME: Return a CSSStyleDeclaration subclass that directs changes to the FontFace.
 | ||
|     return nullptr;
 | ||
| }
 | ||
| 
 | ||
| // https://www.w3.org/TR/cssom/#ref-for-cssfontfacerule
 | ||
| DeprecatedString CSSFontFaceRule::serialized() const
 | ||
| {
 | ||
|     StringBuilder builder;
 | ||
|     // The result of concatenating the following:
 | ||
| 
 | ||
|     // 1. The string "@font-face {", followed by a single SPACE (U+0020).
 | ||
|     builder.append("@font-face { "sv);
 | ||
| 
 | ||
|     // 2. The string "font-family:", followed by a single SPACE (U+0020).
 | ||
|     builder.append("font-family: "sv);
 | ||
| 
 | ||
|     // 3. The result of performing serialize a string on the rule’s font family name.
 | ||
|     serialize_a_string(builder, m_font_face.font_family()).release_value_but_fixme_should_propagate_errors();
 | ||
| 
 | ||
|     // 4. The string ";", i.e., SEMICOLON (U+003B).
 | ||
|     builder.append(';');
 | ||
| 
 | ||
|     // 5. If the rule’s associated source list is not empty, follow these substeps:
 | ||
|     if (!m_font_face.sources().is_empty()) {
 | ||
|         // 1. A single SPACE (U+0020), followed by the string "src:", followed by a single SPACE (U+0020).
 | ||
|         builder.append(" src: "sv);
 | ||
| 
 | ||
|         // 2. The result of invoking serialize a comma-separated list on performing serialize a URL or serialize a LOCAL for each source on the source list.
 | ||
|         serialize_a_comma_separated_list(builder, m_font_face.sources(), [&](StringBuilder& builder, FontFace::Source source) -> ErrorOr<void> {
 | ||
|             // FIXME: Serialize locals once we support those
 | ||
|             TRY(serialize_a_url(builder, source.url.to_deprecated_string()));
 | ||
| 
 | ||
|             // NOTE: No spec currently exists for format()
 | ||
|             if (source.format.has_value()) {
 | ||
|                 TRY(builder.try_append(" format("sv));
 | ||
|                 TRY(serialize_a_string(builder, source.format.value()));
 | ||
|                 TRY(builder.try_append(")"sv));
 | ||
|             }
 | ||
|             return {};
 | ||
|         }).release_value_but_fixme_should_propagate_errors();
 | ||
| 
 | ||
|         // 3. The string ";", i.e., SEMICOLON (U+003B).
 | ||
|         builder.append(';');
 | ||
|     }
 | ||
| 
 | ||
|     // 6. If rule’s associated unicode-range descriptor is present, a single SPACE (U+0020), followed by the string "unicode-range:", followed by a single SPACE (U+0020), followed by the result of performing serialize a <'unicode-range'>, followed by the string ";", i.e., SEMICOLON (U+003B).
 | ||
|     builder.append(" unicode-range: "sv);
 | ||
|     serialize_unicode_ranges(builder, m_font_face.unicode_ranges()).release_value_but_fixme_should_propagate_errors();
 | ||
|     builder.append(';');
 | ||
| 
 | ||
|     // FIXME: 7. If rule’s associated font-variant descriptor is present, a single SPACE (U+0020),
 | ||
|     // followed by the string "font-variant:", followed by a single SPACE (U+0020),
 | ||
|     // followed by the result of performing serialize a <'font-variant'>,
 | ||
|     // followed by the string ";", i.e., SEMICOLON (U+003B).
 | ||
| 
 | ||
|     // FIXME: 8. If rule’s associated font-feature-settings descriptor is present, a single SPACE (U+0020),
 | ||
|     // followed by the string "font-feature-settings:", followed by a single SPACE (U+0020),
 | ||
|     // followed by the result of performing serialize a <'font-feature-settings'>,
 | ||
|     // followed by the string ";", i.e., SEMICOLON (U+003B).
 | ||
| 
 | ||
|     // FIXME: 9. If rule’s associated font-stretch descriptor is present, a single SPACE (U+0020),
 | ||
|     // followed by the string "font-stretch:", followed by a single SPACE (U+0020),
 | ||
|     // followed by the result of performing serialize a <'font-stretch'>,
 | ||
|     // followed by the string ";", i.e., SEMICOLON (U+003B).
 | ||
| 
 | ||
|     // 10. If rule’s associated font-weight descriptor is present, a single SPACE (U+0020),
 | ||
|     //     followed by the string "font-weight:", followed by a single SPACE (U+0020),
 | ||
|     //     followed by the result of performing serialize a <'font-weight'>,
 | ||
|     //     followed by the string ";", i.e., SEMICOLON (U+003B).
 | ||
|     if (m_font_face.weight().has_value()) {
 | ||
|         auto weight = m_font_face.weight().value();
 | ||
|         builder.append(" font-weight: "sv);
 | ||
|         if (weight == 400)
 | ||
|             builder.append("normal"sv);
 | ||
|         else if (weight == 700)
 | ||
|             builder.append("bold"sv);
 | ||
|         else
 | ||
|             builder.appendff("{}", weight);
 | ||
|         builder.append(";"sv);
 | ||
|     }
 | ||
| 
 | ||
|     // 11. If rule’s associated font-style descriptor is present, a single SPACE (U+0020),
 | ||
|     //     followed by the string "font-style:", followed by a single SPACE (U+0020),
 | ||
|     //     followed by the result of performing serialize a <'font-style'>,
 | ||
|     //     followed by the string ";", i.e., SEMICOLON (U+003B).
 | ||
|     if (m_font_face.slope().has_value()) {
 | ||
|         auto slope = m_font_face.slope().value();
 | ||
|         builder.append(" font-style: "sv);
 | ||
|         if (slope == Gfx::name_to_slope("Normal"sv))
 | ||
|             builder.append("normal"sv);
 | ||
|         else if (slope == Gfx::name_to_slope("Italic"sv))
 | ||
|             builder.append("italic"sv);
 | ||
|         else {
 | ||
|             dbgln("FIXME: CSSFontFaceRule::serialized() does not support slope {}", slope);
 | ||
|             builder.append("italic"sv);
 | ||
|         }
 | ||
|         builder.append(";"sv);
 | ||
|     }
 | ||
| 
 | ||
|     // 12. A single SPACE (U+0020), followed by the string "}", i.e., RIGHT CURLY BRACKET (U+007D).
 | ||
|     builder.append(" }"sv);
 | ||
| 
 | ||
|     return builder.to_deprecated_string();
 | ||
| }
 | ||
| 
 | ||
| }
 |