mirror of
				https://github.com/LadybirdBrowser/ladybird.git
				synced 2025-10-25 18:34:14 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			189 lines
		
	
	
	
		
			7.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			189 lines
		
	
	
	
		
			7.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | ||
|  * Copyright (c) 2025, Sam Atkins <sam@ladybird.org>
 | ||
|  *
 | ||
|  * SPDX-License-Identifier: BSD-2-Clause
 | ||
|  */
 | ||
| 
 | ||
| #include "CSSUnitValue.h"
 | ||
| #include <LibWeb/Bindings/CSSUnitValuePrototype.h>
 | ||
| #include <LibWeb/Bindings/Intrinsics.h>
 | ||
| #include <LibWeb/CSS/Serialize.h>
 | ||
| #include <LibWeb/CSS/Units.h>
 | ||
| #include <LibWeb/WebIDL/ExceptionOr.h>
 | ||
| 
 | ||
| namespace Web::CSS {
 | ||
| 
 | ||
| GC_DEFINE_ALLOCATOR(CSSUnitValue);
 | ||
| 
 | ||
| GC::Ref<CSSUnitValue> CSSUnitValue::create(JS::Realm& realm, double value, FlyString unit)
 | ||
| {
 | ||
|     // The type of a CSSUnitValue is the result of creating a type from its unit internal slot.
 | ||
|     // https://drafts.css-houdini.org/css-typed-om-1/#type-of-a-cssunitvalue
 | ||
|     auto numeric_type = NumericType::create_from_unit(unit);
 | ||
|     return realm.create<CSSUnitValue>(realm, value, move(unit), numeric_type.release_value());
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.css-houdini.org/css-typed-om-1/#dom-cssunitvalue-cssunitvalue
 | ||
| WebIDL::ExceptionOr<GC::Ref<CSSUnitValue>> CSSUnitValue::construct_impl(JS::Realm& realm, double value, FlyString unit)
 | ||
| {
 | ||
|     // 1. If creating a type from unit returns failure, throw a TypeError and abort this algorithm.
 | ||
|     auto numeric_type = NumericType::create_from_unit(unit);
 | ||
|     if (!numeric_type.has_value())
 | ||
|         return WebIDL::SimpleException { WebIDL::SimpleExceptionType::TypeError, MUST(String::formatted("Cannot create CSSUnitValue with unrecognized unit '{}'", unit)) };
 | ||
| 
 | ||
|     // 2. Return a new CSSUnitValue with its value internal slot set to value and its unit set to unit.
 | ||
|     return realm.create<CSSUnitValue>(realm, value, move(unit), numeric_type.release_value());
 | ||
| }
 | ||
| 
 | ||
| CSSUnitValue::CSSUnitValue(JS::Realm& realm, double value, FlyString unit, NumericType type)
 | ||
|     : CSSNumericValue(realm, move(type))
 | ||
|     , m_value(value)
 | ||
|     // AD-HOC: WPT expects the unit to be lowercase but this doesn't seem to be specified anywhere.
 | ||
|     , m_unit(unit.to_ascii_lowercase())
 | ||
| {
 | ||
| }
 | ||
| 
 | ||
| void CSSUnitValue::initialize(JS::Realm& realm)
 | ||
| {
 | ||
|     WEB_SET_PROTOTYPE_FOR_INTERFACE(CSSUnitValue);
 | ||
|     Base::initialize(realm);
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.css-houdini.org/css-typed-om-1/#dom-cssunitvalue-value
 | ||
| void CSSUnitValue::set_value(double value)
 | ||
| {
 | ||
|     // AD-HOC: No definition: https://github.com/w3c/css-houdini-drafts/issues/1146
 | ||
|     m_value = value;
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.css-houdini.org/css-typed-om-1/#serialize-a-cssunitvalue
 | ||
| String CSSUnitValue::serialize_unit_value(Optional<double> minimum, Optional<double> maximum) const
 | ||
| {
 | ||
|     // To serialize a CSSUnitValue this, with optional arguments minimum, a numeric value, and maximum, a numeric value:
 | ||
| 
 | ||
|     // 1. Let value and unit be this‘s value and unit internal slots.
 | ||
| 
 | ||
|     // 2. Set s to the result of serializing a <number> from value, per CSSOM §6.7.2 Serializing CSS Values.
 | ||
|     StringBuilder s;
 | ||
|     serialize_a_number(s, m_value);
 | ||
| 
 | ||
|     // 3. If unit is:
 | ||
|     // -> "number"
 | ||
|     if (m_unit == "number"_fly_string) {
 | ||
|         // Do nothing.
 | ||
|     }
 | ||
|     // -> "percent"
 | ||
|     else if (m_unit == "percent"_fly_string) {
 | ||
|         // Append "%" to s.
 | ||
|         s.append("%"sv);
 | ||
|     }
 | ||
|     // -> anything else
 | ||
|     else {
 | ||
|         // Append unit to s.
 | ||
|         s.append(m_unit.to_ascii_lowercase());
 | ||
|     }
 | ||
| 
 | ||
|     // 4. If minimum was passed and this is less than minimum, or if maximum was passed and this is greater than
 | ||
|     //    maximum, or either minimum and/or maximum were passed and the relative size of this and minimum/maximum can’t
 | ||
|     //    be determined with the available information at this time, prepend "calc(" to s, then append ")" to s.
 | ||
|     if ((minimum.has_value() && m_value < minimum.value())
 | ||
|         || (maximum.has_value() && m_value > maximum.value())) {
 | ||
|         // FIXME: "or either minimum and/or maximum were passed and the relative size of this and minimum/maximum can’t be determined with the available information at this time"
 | ||
|         return MUST(String::formatted("calc({})", s.string_view()));
 | ||
|     }
 | ||
| 
 | ||
|     // 5. Return s.
 | ||
|     return s.to_string_without_validation();
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.css-houdini.org/css-typed-om-1/#convert-a-cssunitvalue
 | ||
| GC::Ptr<CSSUnitValue> CSSUnitValue::converted_to_unit(FlyString const& unit) const
 | ||
| {
 | ||
|     // 1. Let old unit be the value of this’s unit internal slot, and old value be the value of this’s value internal
 | ||
|     //    slot.
 | ||
|     auto old_unit = m_unit;
 | ||
|     auto old_value = m_value;
 | ||
| 
 | ||
|     // 2. If old unit and unit are not compatible units, return failure.
 | ||
|     double ratio = 1.0;
 | ||
|     // NB: If the units are identical, they're always compatible. That also covers cases of `number` and `percent`
 | ||
|     //     which aren't actually units.
 | ||
|     if (old_unit != unit) {
 | ||
|         auto old_dimension_type = dimension_for_unit(old_unit);
 | ||
|         auto new_dimension_type = dimension_for_unit(unit);
 | ||
|         if (!new_dimension_type.has_value() || old_dimension_type != new_dimension_type)
 | ||
|             return {};
 | ||
| 
 | ||
|         switch (*new_dimension_type) {
 | ||
|         case DimensionType::Angle: {
 | ||
|             auto from = string_to_angle_unit(old_unit).release_value();
 | ||
|             auto to = string_to_angle_unit(unit).release_value();
 | ||
|             if (!units_are_compatible(from, to))
 | ||
|                 return {};
 | ||
|             ratio = ratio_between_units(from, to);
 | ||
|             break;
 | ||
|         }
 | ||
|         case DimensionType::Flex: {
 | ||
|             auto from = string_to_angle_unit(old_unit).release_value();
 | ||
|             auto to = string_to_angle_unit(unit).release_value();
 | ||
|             if (!units_are_compatible(from, to))
 | ||
|                 return {};
 | ||
|             ratio = ratio_between_units(from, to);
 | ||
|             break;
 | ||
|         }
 | ||
|         case DimensionType::Frequency: {
 | ||
|             auto from = string_to_frequency_unit(old_unit).release_value();
 | ||
|             auto to = string_to_frequency_unit(unit).release_value();
 | ||
|             if (!units_are_compatible(from, to))
 | ||
|                 return {};
 | ||
|             ratio = ratio_between_units(from, to);
 | ||
|             break;
 | ||
|         }
 | ||
|         case DimensionType::Length: {
 | ||
|             auto from = string_to_length_unit(old_unit).release_value();
 | ||
|             auto to = string_to_length_unit(unit).release_value();
 | ||
|             if (!units_are_compatible(from, to))
 | ||
|                 return {};
 | ||
|             ratio = ratio_between_units(from, to);
 | ||
|             break;
 | ||
|         }
 | ||
|         case DimensionType::Resolution: {
 | ||
|             auto from = string_to_resolution_unit(old_unit).release_value();
 | ||
|             auto to = string_to_resolution_unit(unit).release_value();
 | ||
|             if (!units_are_compatible(from, to))
 | ||
|                 return {};
 | ||
|             ratio = ratio_between_units(from, to);
 | ||
|             break;
 | ||
|         }
 | ||
|         case DimensionType::Time: {
 | ||
|             auto from = string_to_time_unit(old_unit).release_value();
 | ||
|             auto to = string_to_time_unit(unit).release_value();
 | ||
|             if (!units_are_compatible(from, to))
 | ||
|                 return {};
 | ||
|             ratio = ratio_between_units(from, to);
 | ||
|             break;
 | ||
|         }
 | ||
|         }
 | ||
|     }
 | ||
| 
 | ||
|     // 3. Return a new CSSUnitValue whose unit internal slot is set to unit, and whose value internal slot is set to
 | ||
|     //    old value multiplied by the conversation ratio between old unit and unit.
 | ||
|     return CSSUnitValue::create(realm(), old_value * ratio, unit);
 | ||
| }
 | ||
| 
 | ||
| // https://drafts.css-houdini.org/css-typed-om-1/#equal-numeric-value
 | ||
| bool CSSUnitValue::is_equal_numeric_value(GC::Ref<CSSNumericValue> other) const
 | ||
| {
 | ||
|     // NB: Only steps 1 and 2 are relevant.
 | ||
|     // 1. If value1 and value2 are not members of the same interface, return false.
 | ||
|     auto* other_unit_value = as_if<CSSUnitValue>(*other);
 | ||
|     if (!other_unit_value)
 | ||
|         return false;
 | ||
| 
 | ||
|     // 2. If value1 and value2 are both CSSUnitValues, return true if they have equal unit and value internal slots,
 | ||
|     //    or false otherwise.
 | ||
|     return m_unit == other_unit_value->m_unit
 | ||
|         && m_value == other_unit_value->m_value;
 | ||
| }
 | ||
| 
 | ||
| }
 | 
