mirror of
				https://github.com/LadybirdBrowser/ladybird.git
				synced 2025-11-04 07:10:57 +00:00 
			
		
		
		
	For us, that's KeywordStyleValue and CustomIdentStyleValue. CustomIdentStyleValue now has a .cpp file to reduce the number of includes in its header file.
		
			
				
	
	
		
			358 lines
		
	
	
	
		
			14 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			358 lines
		
	
	
	
		
			14 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/*
 | 
						|
 * Copyright (c) 2018-2020, Andreas Kling <andreas@ladybird.org>
 | 
						|
 * Copyright (c) 2021, Tobias Christiansen <tobyase@serenityos.org>
 | 
						|
 * Copyright (c) 2021-2025, Sam Atkins <sam@ladybird.org>
 | 
						|
 * Copyright (c) 2022-2023, MacDue <macdue@dueutil.tech>
 | 
						|
 *
 | 
						|
 * SPDX-License-Identifier: BSD-2-Clause
 | 
						|
 */
 | 
						|
 | 
						|
#include "KeywordStyleValue.h"
 | 
						|
#include <LibGfx/Palette.h>
 | 
						|
#include <LibWeb/CSS/CSSKeywordValue.h>
 | 
						|
#include <LibWeb/CSS/SystemColor.h>
 | 
						|
#include <LibWeb/DOM/Document.h>
 | 
						|
#include <LibWeb/Layout/Node.h>
 | 
						|
#include <LibWeb/Page/Page.h>
 | 
						|
 | 
						|
namespace Web::CSS {
 | 
						|
 | 
						|
String KeywordStyleValue::to_string(SerializationMode) const
 | 
						|
{
 | 
						|
    return MUST(String::from_utf8(string_from_keyword(keyword())));
 | 
						|
}
 | 
						|
 | 
						|
bool KeywordStyleValue::is_color(Keyword keyword)
 | 
						|
{
 | 
						|
    switch (keyword) {
 | 
						|
    case Keyword::Accentcolor:
 | 
						|
    case Keyword::Accentcolortext:
 | 
						|
    case Keyword::Activeborder:
 | 
						|
    case Keyword::Activecaption:
 | 
						|
    case Keyword::Activetext:
 | 
						|
    case Keyword::Appworkspace:
 | 
						|
    case Keyword::Background:
 | 
						|
    case Keyword::Buttonborder:
 | 
						|
    case Keyword::Buttonface:
 | 
						|
    case Keyword::Buttonhighlight:
 | 
						|
    case Keyword::Buttonshadow:
 | 
						|
    case Keyword::Buttontext:
 | 
						|
    case Keyword::Canvas:
 | 
						|
    case Keyword::Canvastext:
 | 
						|
    case Keyword::Captiontext:
 | 
						|
    case Keyword::Currentcolor:
 | 
						|
    case Keyword::Field:
 | 
						|
    case Keyword::Fieldtext:
 | 
						|
    case Keyword::Graytext:
 | 
						|
    case Keyword::Highlight:
 | 
						|
    case Keyword::Highlighttext:
 | 
						|
    case Keyword::Inactiveborder:
 | 
						|
    case Keyword::Inactivecaption:
 | 
						|
    case Keyword::Inactivecaptiontext:
 | 
						|
    case Keyword::Infobackground:
 | 
						|
    case Keyword::Infotext:
 | 
						|
    case Keyword::LibwebButtonfacedisabled:
 | 
						|
    case Keyword::LibwebButtonfacehover:
 | 
						|
    case Keyword::LibwebLink:
 | 
						|
    case Keyword::LibwebPaletteActiveLink:
 | 
						|
    case Keyword::LibwebPaletteActiveWindowBorder1:
 | 
						|
    case Keyword::LibwebPaletteActiveWindowBorder2:
 | 
						|
    case Keyword::LibwebPaletteActiveWindowTitle:
 | 
						|
    case Keyword::LibwebPaletteBase:
 | 
						|
    case Keyword::LibwebPaletteBaseText:
 | 
						|
    case Keyword::LibwebPaletteButton:
 | 
						|
    case Keyword::LibwebPaletteButtonText:
 | 
						|
    case Keyword::LibwebPaletteDesktopBackground:
 | 
						|
    case Keyword::LibwebPaletteFocusOutline:
 | 
						|
    case Keyword::LibwebPaletteHighlightWindowBorder1:
 | 
						|
    case Keyword::LibwebPaletteHighlightWindowBorder2:
 | 
						|
    case Keyword::LibwebPaletteHighlightWindowTitle:
 | 
						|
    case Keyword::LibwebPaletteHoverHighlight:
 | 
						|
    case Keyword::LibwebPaletteInactiveSelection:
 | 
						|
    case Keyword::LibwebPaletteInactiveSelectionText:
 | 
						|
    case Keyword::LibwebPaletteInactiveWindowBorder1:
 | 
						|
    case Keyword::LibwebPaletteInactiveWindowBorder2:
 | 
						|
    case Keyword::LibwebPaletteInactiveWindowTitle:
 | 
						|
    case Keyword::LibwebPaletteLink:
 | 
						|
    case Keyword::LibwebPaletteMenuBase:
 | 
						|
    case Keyword::LibwebPaletteMenuBaseText:
 | 
						|
    case Keyword::LibwebPaletteMenuSelection:
 | 
						|
    case Keyword::LibwebPaletteMenuSelectionText:
 | 
						|
    case Keyword::LibwebPaletteMenuStripe:
 | 
						|
    case Keyword::LibwebPaletteMovingWindowBorder1:
 | 
						|
    case Keyword::LibwebPaletteMovingWindowBorder2:
 | 
						|
    case Keyword::LibwebPaletteMovingWindowTitle:
 | 
						|
    case Keyword::LibwebPaletteRubberBandBorder:
 | 
						|
    case Keyword::LibwebPaletteRubberBandFill:
 | 
						|
    case Keyword::LibwebPaletteRuler:
 | 
						|
    case Keyword::LibwebPaletteRulerActiveText:
 | 
						|
    case Keyword::LibwebPaletteRulerBorder:
 | 
						|
    case Keyword::LibwebPaletteRulerInactiveText:
 | 
						|
    case Keyword::LibwebPaletteSelection:
 | 
						|
    case Keyword::LibwebPaletteSelectionText:
 | 
						|
    case Keyword::LibwebPaletteSyntaxComment:
 | 
						|
    case Keyword::LibwebPaletteSyntaxControlKeyword:
 | 
						|
    case Keyword::LibwebPaletteSyntaxIdentifier:
 | 
						|
    case Keyword::LibwebPaletteSyntaxKeyword:
 | 
						|
    case Keyword::LibwebPaletteSyntaxNumber:
 | 
						|
    case Keyword::LibwebPaletteSyntaxOperator:
 | 
						|
    case Keyword::LibwebPaletteSyntaxPreprocessorStatement:
 | 
						|
    case Keyword::LibwebPaletteSyntaxPreprocessorValue:
 | 
						|
    case Keyword::LibwebPaletteSyntaxPunctuation:
 | 
						|
    case Keyword::LibwebPaletteSyntaxString:
 | 
						|
    case Keyword::LibwebPaletteSyntaxType:
 | 
						|
    case Keyword::LibwebPaletteTextCursor:
 | 
						|
    case Keyword::LibwebPaletteThreedHighlight:
 | 
						|
    case Keyword::LibwebPaletteThreedShadow1:
 | 
						|
    case Keyword::LibwebPaletteThreedShadow2:
 | 
						|
    case Keyword::LibwebPaletteVisitedLink:
 | 
						|
    case Keyword::LibwebPaletteWindow:
 | 
						|
    case Keyword::LibwebPaletteWindowText:
 | 
						|
    case Keyword::Linktext:
 | 
						|
    case Keyword::Mark:
 | 
						|
    case Keyword::Marktext:
 | 
						|
    case Keyword::Menu:
 | 
						|
    case Keyword::Menutext:
 | 
						|
    case Keyword::Scrollbar:
 | 
						|
    case Keyword::Selecteditem:
 | 
						|
    case Keyword::Selecteditemtext:
 | 
						|
    case Keyword::Threeddarkshadow:
 | 
						|
    case Keyword::Threedface:
 | 
						|
    case Keyword::Threedhighlight:
 | 
						|
    case Keyword::Threedlightshadow:
 | 
						|
    case Keyword::Threedshadow:
 | 
						|
    case Keyword::Visitedtext:
 | 
						|
    case Keyword::Window:
 | 
						|
    case Keyword::Windowframe:
 | 
						|
    case Keyword::Windowtext:
 | 
						|
        return true;
 | 
						|
    default:
 | 
						|
        return false;
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
bool KeywordStyleValue::has_color() const
 | 
						|
{
 | 
						|
    return is_color(keyword());
 | 
						|
}
 | 
						|
 | 
						|
Optional<Color> KeywordStyleValue::to_color(ColorResolutionContext color_resolution_context) const
 | 
						|
{
 | 
						|
    if (keyword() == Keyword::Currentcolor) {
 | 
						|
        return color_resolution_context.current_color.value_or(Color::Black);
 | 
						|
    }
 | 
						|
 | 
						|
    PreferredColorScheme scheme = color_resolution_context.color_scheme.value_or(PreferredColorScheme::Light);
 | 
						|
 | 
						|
    // First, handle <system-color>s, since they don't strictly require a node.
 | 
						|
    // https://www.w3.org/TR/css-color-4/#css-system-colors
 | 
						|
    // https://www.w3.org/TR/css-color-4/#deprecated-system-colors
 | 
						|
    switch (keyword()) {
 | 
						|
    case Keyword::Accentcolor:
 | 
						|
        return SystemColor::accent_color(scheme);
 | 
						|
    case Keyword::Accentcolortext:
 | 
						|
        return SystemColor::accent_color_text(scheme);
 | 
						|
    case Keyword::Buttonborder:
 | 
						|
    case Keyword::Activeborder:
 | 
						|
    case Keyword::Inactiveborder:
 | 
						|
    case Keyword::Threeddarkshadow:
 | 
						|
    case Keyword::Threedhighlight:
 | 
						|
    case Keyword::Threedlightshadow:
 | 
						|
    case Keyword::Threedshadow:
 | 
						|
    case Keyword::Windowframe:
 | 
						|
        return SystemColor::button_border(scheme);
 | 
						|
    case Keyword::Buttonface:
 | 
						|
    case Keyword::Buttonhighlight:
 | 
						|
    case Keyword::Buttonshadow:
 | 
						|
    case Keyword::Threedface:
 | 
						|
        return SystemColor::button_face(scheme);
 | 
						|
    case Keyword::Buttontext:
 | 
						|
        return SystemColor::button_text(scheme);
 | 
						|
    case Keyword::Canvas:
 | 
						|
    case Keyword::Appworkspace:
 | 
						|
    case Keyword::Background:
 | 
						|
    case Keyword::Inactivecaption:
 | 
						|
    case Keyword::Infobackground:
 | 
						|
    case Keyword::Menu:
 | 
						|
    case Keyword::Scrollbar:
 | 
						|
    case Keyword::Window:
 | 
						|
        return SystemColor::canvas(scheme);
 | 
						|
    case Keyword::Canvastext:
 | 
						|
    case Keyword::Activecaption:
 | 
						|
    case Keyword::Captiontext:
 | 
						|
    case Keyword::Infotext:
 | 
						|
    case Keyword::Menutext:
 | 
						|
    case Keyword::Windowtext:
 | 
						|
        return SystemColor::canvas_text(scheme);
 | 
						|
    case Keyword::Field:
 | 
						|
        return SystemColor::field(scheme);
 | 
						|
    case Keyword::Fieldtext:
 | 
						|
        return SystemColor::field_text(scheme);
 | 
						|
    case Keyword::Graytext:
 | 
						|
    case Keyword::Inactivecaptiontext:
 | 
						|
        return SystemColor::gray_text(scheme);
 | 
						|
    case Keyword::Highlight:
 | 
						|
        return SystemColor::highlight(scheme);
 | 
						|
    case Keyword::Highlighttext:
 | 
						|
        return SystemColor::highlight_text(scheme);
 | 
						|
    case Keyword::Mark:
 | 
						|
        return SystemColor::mark(scheme);
 | 
						|
    case Keyword::Marktext:
 | 
						|
        return SystemColor::mark_text(scheme);
 | 
						|
    case Keyword::Selecteditem:
 | 
						|
        return SystemColor::selected_item(scheme);
 | 
						|
    case Keyword::Selecteditemtext:
 | 
						|
        return SystemColor::selected_item_text(scheme);
 | 
						|
    case Keyword::LibwebButtonfacedisabled:
 | 
						|
        return SystemColor::button_face(scheme).with_alpha(128);
 | 
						|
    case Keyword::LibwebButtonfacehover:
 | 
						|
        return SystemColor::button_face(scheme).darkened(0.8f);
 | 
						|
    default:
 | 
						|
        break;
 | 
						|
    }
 | 
						|
 | 
						|
    if (!color_resolution_context.document) {
 | 
						|
        // FIXME: Can't resolve palette colors without a document.
 | 
						|
        return Color::Black;
 | 
						|
    }
 | 
						|
 | 
						|
    switch (keyword()) {
 | 
						|
    case Keyword::LibwebLink:
 | 
						|
    case Keyword::Linktext:
 | 
						|
        return color_resolution_context.document->normal_link_color().value_or(SystemColor::link_text(scheme));
 | 
						|
    case Keyword::Visitedtext:
 | 
						|
        return color_resolution_context.document->visited_link_color().value_or(SystemColor::visited_text(scheme));
 | 
						|
    case Keyword::Activetext:
 | 
						|
        return color_resolution_context.document->active_link_color().value_or(SystemColor::active_text(scheme));
 | 
						|
    default:
 | 
						|
        break;
 | 
						|
    }
 | 
						|
 | 
						|
    auto palette = color_resolution_context.document->page().palette();
 | 
						|
    switch (keyword()) {
 | 
						|
    case Keyword::LibwebPaletteDesktopBackground:
 | 
						|
        return palette.color(ColorRole::DesktopBackground);
 | 
						|
    case Keyword::LibwebPaletteActiveWindowBorder1:
 | 
						|
        return palette.color(ColorRole::ActiveWindowBorder1);
 | 
						|
    case Keyword::LibwebPaletteActiveWindowBorder2:
 | 
						|
        return palette.color(ColorRole::ActiveWindowBorder2);
 | 
						|
    case Keyword::LibwebPaletteActiveWindowTitle:
 | 
						|
        return palette.color(ColorRole::ActiveWindowTitle);
 | 
						|
    case Keyword::LibwebPaletteInactiveWindowBorder1:
 | 
						|
        return palette.color(ColorRole::InactiveWindowBorder1);
 | 
						|
    case Keyword::LibwebPaletteInactiveWindowBorder2:
 | 
						|
        return palette.color(ColorRole::InactiveWindowBorder2);
 | 
						|
    case Keyword::LibwebPaletteInactiveWindowTitle:
 | 
						|
        return palette.color(ColorRole::InactiveWindowTitle);
 | 
						|
    case Keyword::LibwebPaletteMovingWindowBorder1:
 | 
						|
        return palette.color(ColorRole::MovingWindowBorder1);
 | 
						|
    case Keyword::LibwebPaletteMovingWindowBorder2:
 | 
						|
        return palette.color(ColorRole::MovingWindowBorder2);
 | 
						|
    case Keyword::LibwebPaletteMovingWindowTitle:
 | 
						|
        return palette.color(ColorRole::MovingWindowTitle);
 | 
						|
    case Keyword::LibwebPaletteHighlightWindowBorder1:
 | 
						|
        return palette.color(ColorRole::HighlightWindowBorder1);
 | 
						|
    case Keyword::LibwebPaletteHighlightWindowBorder2:
 | 
						|
        return palette.color(ColorRole::HighlightWindowBorder2);
 | 
						|
    case Keyword::LibwebPaletteHighlightWindowTitle:
 | 
						|
        return palette.color(ColorRole::HighlightWindowTitle);
 | 
						|
    case Keyword::LibwebPaletteMenuStripe:
 | 
						|
        return palette.color(ColorRole::MenuStripe);
 | 
						|
    case Keyword::LibwebPaletteMenuBase:
 | 
						|
        return palette.color(ColorRole::MenuBase);
 | 
						|
    case Keyword::LibwebPaletteMenuBaseText:
 | 
						|
        return palette.color(ColorRole::MenuBaseText);
 | 
						|
    case Keyword::LibwebPaletteMenuSelection:
 | 
						|
        return palette.color(ColorRole::MenuSelection);
 | 
						|
    case Keyword::LibwebPaletteMenuSelectionText:
 | 
						|
        return palette.color(ColorRole::MenuSelectionText);
 | 
						|
    case Keyword::LibwebPaletteWindow:
 | 
						|
        return palette.color(ColorRole::Window);
 | 
						|
    case Keyword::LibwebPaletteWindowText:
 | 
						|
        return palette.color(ColorRole::WindowText);
 | 
						|
    case Keyword::LibwebPaletteButton:
 | 
						|
        return palette.color(ColorRole::Button);
 | 
						|
    case Keyword::LibwebPaletteButtonText:
 | 
						|
        return palette.color(ColorRole::ButtonText);
 | 
						|
    case Keyword::LibwebPaletteBase:
 | 
						|
        return palette.color(ColorRole::Base);
 | 
						|
    case Keyword::LibwebPaletteBaseText:
 | 
						|
        return palette.color(ColorRole::BaseText);
 | 
						|
    case Keyword::LibwebPaletteThreedHighlight:
 | 
						|
        return palette.color(ColorRole::ThreedHighlight);
 | 
						|
    case Keyword::LibwebPaletteThreedShadow1:
 | 
						|
        return palette.color(ColorRole::ThreedShadow1);
 | 
						|
    case Keyword::LibwebPaletteThreedShadow2:
 | 
						|
        return palette.color(ColorRole::ThreedShadow2);
 | 
						|
    case Keyword::LibwebPaletteHoverHighlight:
 | 
						|
        return palette.color(ColorRole::HoverHighlight);
 | 
						|
    case Keyword::LibwebPaletteSelection:
 | 
						|
        return palette.color(ColorRole::Selection);
 | 
						|
    case Keyword::LibwebPaletteSelectionText:
 | 
						|
        return palette.color(ColorRole::SelectionText);
 | 
						|
    case Keyword::LibwebPaletteInactiveSelection:
 | 
						|
        return palette.color(ColorRole::InactiveSelection);
 | 
						|
    case Keyword::LibwebPaletteInactiveSelectionText:
 | 
						|
        return palette.color(ColorRole::InactiveSelectionText);
 | 
						|
    case Keyword::LibwebPaletteRubberBandFill:
 | 
						|
        return palette.color(ColorRole::RubberBandFill);
 | 
						|
    case Keyword::LibwebPaletteRubberBandBorder:
 | 
						|
        return palette.color(ColorRole::RubberBandBorder);
 | 
						|
    case Keyword::LibwebPaletteLink:
 | 
						|
        return palette.color(ColorRole::Link);
 | 
						|
    case Keyword::LibwebPaletteActiveLink:
 | 
						|
        return palette.color(ColorRole::ActiveLink);
 | 
						|
    case Keyword::LibwebPaletteVisitedLink:
 | 
						|
        return palette.color(ColorRole::VisitedLink);
 | 
						|
    case Keyword::LibwebPaletteRuler:
 | 
						|
        return palette.color(ColorRole::Ruler);
 | 
						|
    case Keyword::LibwebPaletteRulerBorder:
 | 
						|
        return palette.color(ColorRole::RulerBorder);
 | 
						|
    case Keyword::LibwebPaletteRulerActiveText:
 | 
						|
        return palette.color(ColorRole::RulerActiveText);
 | 
						|
    case Keyword::LibwebPaletteRulerInactiveText:
 | 
						|
        return palette.color(ColorRole::RulerInactiveText);
 | 
						|
    case Keyword::LibwebPaletteTextCursor:
 | 
						|
        return palette.color(ColorRole::TextCursor);
 | 
						|
    case Keyword::LibwebPaletteFocusOutline:
 | 
						|
        return palette.color(ColorRole::FocusOutline);
 | 
						|
    case Keyword::LibwebPaletteSyntaxComment:
 | 
						|
        return palette.color(ColorRole::SyntaxComment);
 | 
						|
    case Keyword::LibwebPaletteSyntaxNumber:
 | 
						|
        return palette.color(ColorRole::SyntaxNumber);
 | 
						|
    case Keyword::LibwebPaletteSyntaxString:
 | 
						|
        return palette.color(ColorRole::SyntaxString);
 | 
						|
    case Keyword::LibwebPaletteSyntaxType:
 | 
						|
        return palette.color(ColorRole::SyntaxType);
 | 
						|
    case Keyword::LibwebPaletteSyntaxPunctuation:
 | 
						|
        return palette.color(ColorRole::SyntaxPunctuation);
 | 
						|
    case Keyword::LibwebPaletteSyntaxOperator:
 | 
						|
        return palette.color(ColorRole::SyntaxOperator);
 | 
						|
    case Keyword::LibwebPaletteSyntaxKeyword:
 | 
						|
        return palette.color(ColorRole::SyntaxKeyword);
 | 
						|
    case Keyword::LibwebPaletteSyntaxControlKeyword:
 | 
						|
        return palette.color(ColorRole::SyntaxControlKeyword);
 | 
						|
    case Keyword::LibwebPaletteSyntaxIdentifier:
 | 
						|
        return palette.color(ColorRole::SyntaxIdentifier);
 | 
						|
    case Keyword::LibwebPaletteSyntaxPreprocessorStatement:
 | 
						|
        return palette.color(ColorRole::SyntaxPreprocessorStatement);
 | 
						|
    case Keyword::LibwebPaletteSyntaxPreprocessorValue:
 | 
						|
        return palette.color(ColorRole::SyntaxPreprocessorValue);
 | 
						|
    default:
 | 
						|
        return {};
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
Vector<Parser::ComponentValue> KeywordStyleValue::tokenize() const
 | 
						|
{
 | 
						|
    return { Parser::Token::create_ident(FlyString::from_utf8_without_validation(string_from_keyword(m_keyword).bytes())) };
 | 
						|
}
 | 
						|
 | 
						|
// https://drafts.css-houdini.org/css-typed-om-1/#reify-ident
 | 
						|
GC::Ref<CSSStyleValue> KeywordStyleValue::reify(JS::Realm& realm, String const&) const
 | 
						|
{
 | 
						|
    // 1. Return a new CSSKeywordValue with its value internal slot set to the serialization of ident.
 | 
						|
    return CSSKeywordValue::create(realm, FlyString::from_utf8_without_validation(string_from_keyword(m_keyword).bytes()));
 | 
						|
}
 | 
						|
 | 
						|
}
 |