mirror of
				https://github.com/LadybirdBrowser/ladybird.git
				synced 2025-10-31 13:20:59 +00:00 
			
		
		
		
	 03a4ecce19
			
		
	
	
		03a4ecce19
		
	
	
	
	
		
			
			As with ScaleStyleValue, the serialization is the only unique part of this class, and we can just move it over.
		
			
				
	
	
		
			152 lines
		
	
	
	
		
			6.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			152 lines
		
	
	
	
		
			6.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2018-2024, 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 "TransformationStyleValue.h"
 | |
| #include <AK/StringBuilder.h>
 | |
| #include <LibWeb/CSS/StyleValues/AngleStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/LengthStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/NumberStyleValue.h>
 | |
| #include <LibWeb/CSS/StyleValues/PercentageStyleValue.h>
 | |
| #include <LibWeb/CSS/Transformation.h>
 | |
| 
 | |
| namespace Web::CSS {
 | |
| 
 | |
| Transformation TransformationStyleValue::to_transformation() const
 | |
| {
 | |
|     auto function_metadata = transform_function_metadata(m_properties.transform_function);
 | |
|     Vector<TransformValue> values;
 | |
|     size_t argument_index = 0;
 | |
|     for (auto& transformation_value : m_properties.values) {
 | |
|         if (transformation_value->is_calculated()) {
 | |
|             auto& calculated = transformation_value->as_calculated();
 | |
|             if (calculated.resolves_to_length_percentage()) {
 | |
|                 values.append(LengthPercentage { calculated });
 | |
|             } else if (calculated.resolves_to_percentage()) {
 | |
|                 // FIXME: Maybe transform this for loop to always check the metadata for the correct types
 | |
|                 if (function_metadata.parameters[argument_index].type == TransformFunctionParameterType::NumberPercentage) {
 | |
|                     values.append(NumberPercentage { calculated.resolve_percentage().value() });
 | |
|                 } else {
 | |
|                     values.append(LengthPercentage { calculated.resolve_percentage().value() });
 | |
|                 }
 | |
|             } else if (calculated.resolves_to_number()) {
 | |
|                 values.append({ Number(Number::Type::Number, calculated.resolve_number().value()) });
 | |
|             } else if (calculated.resolves_to_angle()) {
 | |
|                 values.append({ calculated.resolve_angle().value() });
 | |
|             } else {
 | |
|                 dbgln("FIXME: Unsupported calc value in transform! {}", calculated.to_string(SerializationMode::Normal));
 | |
|             }
 | |
|         } else if (transformation_value->is_length()) {
 | |
|             values.append({ transformation_value->as_length().length() });
 | |
|         } else if (transformation_value->is_percentage()) {
 | |
|             if (function_metadata.parameters[argument_index].type == TransformFunctionParameterType::NumberPercentage) {
 | |
|                 values.append(NumberPercentage { transformation_value->as_percentage().percentage() });
 | |
|             } else {
 | |
|                 values.append(LengthPercentage { transformation_value->as_percentage().percentage() });
 | |
|             }
 | |
|         } else if (transformation_value->is_number()) {
 | |
|             values.append({ Number(Number::Type::Number, transformation_value->as_number().number()) });
 | |
|         } else if (transformation_value->is_angle()) {
 | |
|             values.append({ transformation_value->as_angle().angle() });
 | |
|         } else {
 | |
|             dbgln("FIXME: Unsupported value in transform! {}", transformation_value->to_string(SerializationMode::Normal));
 | |
|         }
 | |
|         argument_index++;
 | |
|     }
 | |
|     return Transformation { m_properties.transform_function, move(values) };
 | |
| }
 | |
| 
 | |
| String TransformationStyleValue::to_string(SerializationMode mode) const
 | |
| {
 | |
|     // https://drafts.csswg.org/css-transforms-2/#individual-transform-serialization
 | |
|     if (m_properties.property == PropertyID::Scale) {
 | |
|         auto resolve_to_string = [mode](CSSStyleValue const& value) -> String {
 | |
|             if (value.is_number()) {
 | |
|                 return MUST(String::formatted("{}", value.as_number().number()));
 | |
|             }
 | |
|             if (value.is_percentage()) {
 | |
|                 return MUST(String::formatted("{}", value.as_percentage().percentage().as_fraction()));
 | |
|             }
 | |
|             return value.to_string(mode);
 | |
|         };
 | |
| 
 | |
|         auto x_value = resolve_to_string(m_properties.values[0]);
 | |
|         auto y_value = resolve_to_string(m_properties.values[1]);
 | |
|         // FIXME: 3D scaling
 | |
| 
 | |
|         StringBuilder builder;
 | |
|         builder.append(x_value);
 | |
|         if (x_value != y_value) {
 | |
|             builder.append(" "sv);
 | |
|             builder.append(y_value);
 | |
|         }
 | |
|         return builder.to_string_without_validation();
 | |
|     }
 | |
|     if (m_properties.property == PropertyID::Translate) {
 | |
|         auto resolve_to_string = [mode](CSSStyleValue const& value) -> Optional<String> {
 | |
|             if (value.is_length()) {
 | |
|                 if (value.as_length().length().raw_value() == 0)
 | |
|                     return {};
 | |
|             }
 | |
|             if (value.is_percentage()) {
 | |
|                 if (value.as_percentage().percentage().value() == 0)
 | |
|                     return {};
 | |
|             }
 | |
|             return value.to_string(mode);
 | |
|         };
 | |
| 
 | |
|         auto x_value = resolve_to_string(m_properties.values[0]);
 | |
|         auto y_value = resolve_to_string(m_properties.values[1]);
 | |
|         // FIXME: 3D translation
 | |
| 
 | |
|         StringBuilder builder;
 | |
|         builder.append(x_value.value_or("0px"_string));
 | |
|         if (y_value.has_value()) {
 | |
|             builder.append(" "sv);
 | |
|             builder.append(y_value.value());
 | |
|         }
 | |
| 
 | |
|         return builder.to_string_without_validation();
 | |
|     }
 | |
| 
 | |
|     StringBuilder builder;
 | |
|     builder.append(CSS::to_string(m_properties.transform_function));
 | |
|     builder.append('(');
 | |
|     for (size_t i = 0; i < m_properties.values.size(); ++i) {
 | |
|         auto const& value = m_properties.values[i];
 | |
| 
 | |
|         // https://www.w3.org/TR/css-transforms-2/#individual-transforms
 | |
|         // A <percentage> is equivalent to a <number>, for example scale: 100% is equivalent to scale: 1.
 | |
|         // Numbers are used during serialization of specified and computed values.
 | |
|         if ((m_properties.transform_function == CSS::TransformFunction::Scale
 | |
|                 || m_properties.transform_function == CSS::TransformFunction::Scale3d
 | |
|                 || m_properties.transform_function == CSS::TransformFunction::ScaleX
 | |
|                 || m_properties.transform_function == CSS::TransformFunction::ScaleY
 | |
|                 || m_properties.transform_function == CSS::TransformFunction::ScaleZ)
 | |
|             && value->is_percentage()) {
 | |
|             builder.append(String::number(value->as_percentage().percentage().as_fraction()));
 | |
|         } else {
 | |
|             builder.append(value->to_string(mode));
 | |
|         }
 | |
| 
 | |
|         if (i != m_properties.values.size() - 1)
 | |
|             builder.append(", "sv);
 | |
|     }
 | |
|     builder.append(')');
 | |
| 
 | |
|     return MUST(builder.to_string());
 | |
| }
 | |
| 
 | |
| bool TransformationStyleValue::Properties::operator==(Properties const& other) const
 | |
| {
 | |
|     return property == other.property
 | |
|         && transform_function == other.transform_function
 | |
|         && values.span() == other.values.span();
 | |
| }
 | |
| 
 | |
| }
 |