| 
									
										
										
										
											2022-03-21 20:54:45 +00:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2025-01-27 10:39:00 +00:00
										 |  |  |  * Copyright (c) 2022-2025, Sam Atkins <sam@ladybird.org> | 
					
						
							| 
									
										
										
										
											2022-03-21 20:54:45 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * SPDX-License-Identifier: BSD-2-Clause | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #pragma once
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-06 19:02:26 +01:00
										 |  |  | #include <AK/String.h>
 | 
					
						
							| 
									
										
										
										
											2022-03-21 20:54:45 +00:00
										 |  |  | #include <AK/Types.h>
 | 
					
						
							| 
									
										
										
										
											2025-08-06 22:19:09 +12:00
										 |  |  | #include <LibWeb/CSS/SerializationMode.h>
 | 
					
						
							| 
									
										
										
										
											2022-03-21 20:54:45 +00:00
										 |  |  | #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 21:10:21 +02:00
										 |  |  |     Number(Type type, double value) | 
					
						
							| 
									
										
										
										
											2022-03-21 20:54:45 +00:00
										 |  |  |         : m_value(value) | 
					
						
							|  |  |  |         , m_type(type) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-11 12:57:32 +01:00
										 |  |  |     Type type() const { return m_type; } | 
					
						
							| 
									
										
										
										
											2023-05-27 21:10:21 +02:00
										 |  |  |     double value() const { return m_value; } | 
					
						
							| 
									
										
										
										
											2022-03-21 20:54:45 +00: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 21:10:21 +02:00
										 |  |  |         return llround(m_value); | 
					
						
							| 
									
										
										
										
											2022-03-21 20:54:45 +00: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 }; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-06 22:19:09 +12:00
										 |  |  |     String to_string(SerializationMode = SerializationMode::Normal) const; | 
					
						
							| 
									
										
										
										
											2022-07-27 11:41:31 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-15 08:31:12 +01:00
										 |  |  |     bool operator==(Number const& other) const | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return m_type == other.m_type && m_value == other.m_value; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-31 14:55:18 +01: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 20:54:45 +00:00
										 |  |  | private: | 
					
						
							| 
									
										
										
										
											2023-05-27 21:10:21 +02:00
										 |  |  |     double m_value { 0 }; | 
					
						
							| 
									
										
										
										
											2022-03-21 20:54:45 +00:00
										 |  |  |     Type m_type; | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2025-05-13 07:06:33 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-21 20:54:45 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2022-07-27 11:41:31 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | template<> | 
					
						
							|  |  |  | struct AK::Formatter<Web::CSS::Number> : Formatter<StringView> { | 
					
						
							|  |  |  |     ErrorOr<void> format(FormatBuilder& builder, Web::CSS::Number const& number) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2023-08-22 12:25:30 +01:00
										 |  |  |         return Formatter<StringView>::format(builder, number.to_string()); | 
					
						
							| 
									
										
										
										
											2022-07-27 11:41:31 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | }; |