/* * Copyright (c) 2020, Andreas Kling * Copyright (c) 2021, Tobias Christiansen * Copyright (c) 2022, Sam Atkins * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include #include #include #include #include #include #include namespace Web::CSS { Length::Length() = default; Length::Length(int value, Type type) : m_type(type) , m_value(value) { } Length::Length(float value, Type type) : m_type(type) , m_value(value) { } Length Length::make_auto() { return Length(0, Type::Auto); } Length Length::make_px(float value) { return Length(value, Type::Px); } Length Length::make_calculated(NonnullRefPtr calculated_style_value) { Length length { 0, Type::Calculated }; length.m_calculated_style = move(calculated_style_value); return length; } Length Length::percentage_of(Percentage const& percentage) const { if (is_undefined_or_auto()) { dbgln("Attempting to get percentage of an undefined or auto length, this seems wrong? But for now we just return the original length."); return *this; } return Length { percentage.as_fraction() * raw_value(), m_type }; } Length Length::resolved(Length const& fallback_for_undefined, Layout::Node const& layout_node) const { if (is_undefined()) return fallback_for_undefined; if (is_calculated()) return m_calculated_style->resolve_length(layout_node).release_value(); if (is_relative()) return make_px(to_px(layout_node)); return *this; } Length Length::resolved_or_auto(Layout::Node const& layout_node) const { return resolved(make_auto(), layout_node); } Length Length::resolved_or_zero(Layout::Node const& layout_node) const { return resolved(make_px(0), layout_node); } float Length::relative_length_to_px(Gfx::IntRect const& viewport_rect, Gfx::FontMetrics const& font_metrics, float root_font_size) const { switch (m_type) { case Type::Ex: return m_value * font_metrics.x_height; case Type::Em: return m_value * font_metrics.size; case Type::Ch: // FIXME: Use layout_node.font().glyph_height() when writing-mode is not horizontal-tb (it has to be implemented first) return m_value * (font_metrics.glyph_width + font_metrics.glyph_spacing); case Type::Rem: return m_value * root_font_size; case Type::Vw: return viewport_rect.width() * (m_value / 100); case Type::Vh: return viewport_rect.height() * (m_value / 100); case Type::Vmin: return min(viewport_rect.width(), viewport_rect.height()) * (m_value / 100); case Type::Vmax: return max(viewport_rect.width(), viewport_rect.height()) * (m_value / 100); default: VERIFY_NOT_REACHED(); } } float Length::to_px(Layout::Node const& layout_node) const { if (!layout_node.document().browsing_context()) return 0; auto viewport_rect = layout_node.document().browsing_context()->viewport_rect(); auto* root_element = layout_node.document().document_element(); if (!root_element || !root_element->layout_node()) return 0; return to_px(viewport_rect, layout_node.font().metrics('M'), root_element->layout_node()->font().presentation_size()); } const char* Length::unit_name() const { switch (m_type) { case Type::Cm: return "cm"; case Type::In: return "in"; case Type::Px: return "px"; case Type::Pt: return "pt"; case Type::Mm: return "mm"; case Type::Q: return "Q"; case Type::Pc: return "pc"; case Type::Ex: return "ex"; case Type::Em: return "em"; case Type::Ch: return "ch"; case Type::Rem: return "rem"; case Type::Auto: return "auto"; case Type::Undefined: return "undefined"; case Type::Vh: return "vh"; case Type::Vw: return "vw"; case Type::Vmax: return "vmax"; case Type::Vmin: return "vmin"; case Type::Calculated: return "calculated"; } VERIFY_NOT_REACHED(); } }