2022-02-21 17:43:30 +00:00
|
|
|
/*
|
2025-05-16 20:02:16 +01:00
|
|
|
* Copyright (c) 2022-2025, Sam Atkins <sam@ladybird.org>
|
2022-02-21 17:43:30 +00:00
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <AK/Math.h>
|
2025-05-16 20:02:16 +01:00
|
|
|
#include <LibWeb/CSS/Angle.h>
|
2023-03-30 15:33:37 +01:00
|
|
|
#include <LibWeb/CSS/Percentage.h>
|
2024-12-11 15:05:56 +00:00
|
|
|
#include <LibWeb/CSS/StyleValues/CalculatedStyleValue.h>
|
2022-02-21 17:43:30 +00:00
|
|
|
|
|
|
|
namespace Web::CSS {
|
|
|
|
|
2023-05-27 21:10:21 +02:00
|
|
|
Angle::Angle(double value, Type type)
|
2022-02-21 17:43:30 +00:00
|
|
|
: m_type(type)
|
|
|
|
, m_value(value)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2023-05-27 21:10:21 +02:00
|
|
|
Angle Angle::make_degrees(double value)
|
2022-02-21 17:43:30 +00:00
|
|
|
{
|
|
|
|
return { value, Type::Deg };
|
|
|
|
}
|
|
|
|
|
|
|
|
Angle Angle::percentage_of(Percentage const& percentage) const
|
|
|
|
{
|
|
|
|
return Angle { percentage.as_fraction() * m_value, m_type };
|
|
|
|
}
|
|
|
|
|
2025-05-16 20:02:16 +01:00
|
|
|
String Angle::to_string(SerializationMode serialization_mode) const
|
2022-02-21 17:43:30 +00:00
|
|
|
{
|
2025-05-16 20:02:16 +01:00
|
|
|
if (serialization_mode == SerializationMode::ResolvedValue)
|
|
|
|
return MUST(String::formatted("{}deg", to_degrees()));
|
2025-03-19 17:04:15 +00:00
|
|
|
return MUST(String::formatted("{}{}", raw_value(), unit_name()));
|
2022-02-21 17:43:30 +00:00
|
|
|
}
|
|
|
|
|
2023-05-27 21:10:21 +02:00
|
|
|
double Angle::to_degrees() const
|
2022-02-21 17:43:30 +00:00
|
|
|
{
|
|
|
|
switch (m_type) {
|
|
|
|
case Type::Deg:
|
|
|
|
return m_value;
|
|
|
|
case Type::Grad:
|
2023-05-27 21:10:21 +02:00
|
|
|
return m_value * (360.0 / 400.0);
|
2022-02-21 17:43:30 +00:00
|
|
|
case Type::Rad:
|
2023-09-09 14:43:39 +02:00
|
|
|
return AK::to_degrees(m_value);
|
2022-02-21 17:43:30 +00:00
|
|
|
case Type::Turn:
|
2023-05-27 21:10:21 +02:00
|
|
|
return m_value * 360.0;
|
2022-02-21 17:43:30 +00:00
|
|
|
}
|
|
|
|
VERIFY_NOT_REACHED();
|
|
|
|
}
|
|
|
|
|
2023-05-27 23:50:33 +02:00
|
|
|
double Angle::to_radians() const
|
|
|
|
{
|
2023-09-09 14:43:39 +02:00
|
|
|
return AK::to_radians(to_degrees());
|
2023-05-27 23:50:33 +02:00
|
|
|
}
|
|
|
|
|
2022-02-21 17:43:30 +00:00
|
|
|
StringView Angle::unit_name() const
|
|
|
|
{
|
|
|
|
switch (m_type) {
|
|
|
|
case Type::Deg:
|
|
|
|
return "deg"sv;
|
|
|
|
case Type::Grad:
|
|
|
|
return "grad"sv;
|
|
|
|
case Type::Rad:
|
|
|
|
return "rad"sv;
|
|
|
|
case Type::Turn:
|
|
|
|
return "turn"sv;
|
|
|
|
}
|
|
|
|
VERIFY_NOT_REACHED();
|
|
|
|
}
|
|
|
|
|
|
|
|
Optional<Angle::Type> Angle::unit_from_name(StringView name)
|
|
|
|
{
|
2023-03-10 08:48:54 +01:00
|
|
|
if (name.equals_ignoring_ascii_case("deg"sv)) {
|
2022-02-21 17:43:30 +00:00
|
|
|
return Type::Deg;
|
2023-03-10 08:48:54 +01:00
|
|
|
}
|
|
|
|
if (name.equals_ignoring_ascii_case("grad"sv)) {
|
2022-02-21 17:43:30 +00:00
|
|
|
return Type::Grad;
|
2023-03-10 08:48:54 +01:00
|
|
|
}
|
|
|
|
if (name.equals_ignoring_ascii_case("rad"sv)) {
|
2022-02-21 17:43:30 +00:00
|
|
|
return Type::Rad;
|
2023-03-10 08:48:54 +01:00
|
|
|
}
|
|
|
|
if (name.equals_ignoring_ascii_case("turn"sv)) {
|
2022-02-21 17:43:30 +00:00
|
|
|
return Type::Turn;
|
|
|
|
}
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2025-04-15 15:18:27 -06:00
|
|
|
Angle Angle::resolve_calculated(NonnullRefPtr<CalculatedStyleValue const> const& calculated, Layout::Node const& layout_node, Angle const& reference_value)
|
2024-08-02 14:28:24 +02:00
|
|
|
{
|
LibWeb/CSS: Wrap calc()-resolution data in a struct
Initially I added this to the existing CalculationContext, but in
reality, we have some data at parse-time and different data at
resolve-time, so it made more sense to keep those separate.
Instead of needing a variety of methods for resolving a Foo, depending
on whether we have a Layout::Node available, or a percentage basis, or
a length resolution context... put those in a
CalculationResolutionContext, and just pass that one thing to these
methods. This also removes the need for separate resolve_*_percentage()
methods, because we can just pass the percentage basis in to the regular
resolve_foo() method.
This also corrects the issue that *any* calculation may need to resolve
lengths, but we previously only passed a length resolution context to
specific types in some situations. Now, they can all have one available,
though it's up to the caller to provide it.
2025-01-22 16:05:32 +00:00
|
|
|
return calculated->resolve_angle(
|
|
|
|
{
|
|
|
|
.percentage_basis = reference_value,
|
|
|
|
.length_resolution_context = Length::ResolutionContext::for_layout_node(layout_node),
|
|
|
|
})
|
|
|
|
.value();
|
2024-08-02 14:28:24 +02:00
|
|
|
}
|
|
|
|
|
2022-02-21 17:43:30 +00:00
|
|
|
}
|