/* * Copyright (c) 2018-2020, Andreas Kling * Copyright (c) 2021, Tobias Christiansen * Copyright (c) 2021-2023, Sam Atkins * Copyright (c) 2022-2023, MacDue * * SPDX-License-Identifier: BSD-2-Clause */ #include "TransformationStyleValue.h" #include #include #include #include #include #include namespace Web::CSS { Transformation TransformationStyleValue::to_transformation() const { auto function_metadata = transform_function_metadata(m_properties.transform_function); Vector 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 { 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 is equivalent to a , 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 transform_function == other.transform_function && values.span() == other.values.span(); } }