2022-03-21 23:54:45 +03:00
|
|
|
/*
|
2023-05-31 16:55:18 +03:00
|
|
|
* Copyright (c) 2022-2023, Sam Atkins <atkinssj@serenityos.org>
|
2022-03-21 23:54:45 +03:00
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
2023-01-06 21:02:26 +03:00
|
|
|
#include <AK/String.h>
|
2022-03-21 23:54:45 +03:00
|
|
|
#include <AK/Types.h>
|
|
|
|
#include <math.h>
|
|
|
|
|
|
|
|
namespace Web::CSS {
|
|
|
|
|
|
|
|
class Number {
|
|
|
|
public:
|
|
|
|
enum class Type {
|
|
|
|
Number,
|
|
|
|
IntegerWithExplicitSign, // This only exists for the nightmarish An+B parsing algorithm
|
|
|
|
Integer
|
|
|
|
};
|
|
|
|
|
|
|
|
Number()
|
|
|
|
: m_value(0)
|
|
|
|
, m_type(Type::Number)
|
|
|
|
{
|
|
|
|
}
|
2023-05-27 22:10:21 +03:00
|
|
|
Number(Type type, double value)
|
2022-03-21 23:54:45 +03:00
|
|
|
: m_value(value)
|
|
|
|
, m_type(type)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2023-04-11 14:57:32 +03:00
|
|
|
Type type() const { return m_type; }
|
2023-05-27 22:10:21 +03:00
|
|
|
double value() const { return m_value; }
|
2022-03-21 23:54:45 +03:00
|
|
|
i64 integer_value() const
|
|
|
|
{
|
|
|
|
// https://www.w3.org/TR/css-values-4/#numeric-types
|
|
|
|
// When a value cannot be explicitly supported due to range/precision limitations, it must be converted
|
|
|
|
// to the closest value supported by the implementation, but how the implementation defines "closest"
|
|
|
|
// is explicitly undefined as well.
|
2023-05-27 22:10:21 +03:00
|
|
|
return llround(m_value);
|
2022-03-21 23:54:45 +03:00
|
|
|
}
|
|
|
|
bool is_integer() const { return m_type == Type::Integer || m_type == Type::IntegerWithExplicitSign; }
|
|
|
|
bool is_integer_with_explicit_sign() const { return m_type == Type::IntegerWithExplicitSign; }
|
|
|
|
|
|
|
|
Number operator+(Number const& other) const
|
|
|
|
{
|
|
|
|
if (is_integer() && other.is_integer())
|
|
|
|
return { Type::Integer, m_value + other.m_value };
|
|
|
|
return { Type::Number, m_value + other.m_value };
|
|
|
|
}
|
|
|
|
|
|
|
|
Number operator-(Number const& other) const
|
|
|
|
{
|
|
|
|
if (is_integer() && other.is_integer())
|
|
|
|
return { Type::Integer, m_value - other.m_value };
|
|
|
|
return { Type::Number, m_value - other.m_value };
|
|
|
|
}
|
|
|
|
|
|
|
|
Number operator*(Number const& other) const
|
|
|
|
{
|
|
|
|
if (is_integer() && other.is_integer())
|
|
|
|
return { Type::Integer, m_value * other.m_value };
|
|
|
|
return { Type::Number, m_value * other.m_value };
|
|
|
|
}
|
|
|
|
|
|
|
|
Number operator/(Number const& other) const
|
|
|
|
{
|
|
|
|
return { Type::Number, m_value / other.m_value };
|
|
|
|
}
|
|
|
|
|
2023-08-22 14:25:30 +03:00
|
|
|
String to_string() const
|
2022-07-27 13:41:31 +03:00
|
|
|
{
|
|
|
|
if (m_type == Type::IntegerWithExplicitSign)
|
2023-08-22 14:25:30 +03:00
|
|
|
return MUST(String::formatted("{:+}", m_value));
|
|
|
|
return MUST(String::number(m_value));
|
2022-07-27 13:41:31 +03:00
|
|
|
}
|
|
|
|
|
2022-09-15 10:31:12 +03:00
|
|
|
bool operator==(Number const& other) const
|
|
|
|
{
|
|
|
|
return m_type == other.m_type && m_value == other.m_value;
|
|
|
|
}
|
|
|
|
|
2023-05-31 16:55:18 +03:00
|
|
|
int operator<=>(Number const& other) const
|
|
|
|
{
|
|
|
|
if (m_value < other.m_value)
|
|
|
|
return -1;
|
|
|
|
if (m_value > other.m_value)
|
|
|
|
return 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2022-03-21 23:54:45 +03:00
|
|
|
private:
|
2023-05-27 22:10:21 +03:00
|
|
|
double m_value { 0 };
|
2022-03-21 23:54:45 +03:00
|
|
|
Type m_type;
|
|
|
|
};
|
|
|
|
}
|
2022-07-27 13:41:31 +03:00
|
|
|
|
|
|
|
template<>
|
|
|
|
struct AK::Formatter<Web::CSS::Number> : Formatter<StringView> {
|
|
|
|
ErrorOr<void> format(FormatBuilder& builder, Web::CSS::Number const& number)
|
|
|
|
{
|
2023-08-22 14:25:30 +03:00
|
|
|
return Formatter<StringView>::format(builder, number.to_string());
|
2022-07-27 13:41:31 +03:00
|
|
|
}
|
|
|
|
};
|