/* * Copyright (c) 2023, Aliaksandr Kalenik * Copyright (c) 2012-2023, Apple Inc. All rights reserved. * Copyright (c) 2022, Sam Atkins * * SPDX-License-Identifier: BSD-2-Clause */ #pragma once #include #include #include #include #include #include #include #include #include namespace Web { /// DevicePixels: A position or length on the physical display. AK_TYPEDEF_DISTINCT_NUMERIC_GENERAL(int, DevicePixels, Arithmetic, CastToUnderlying, Comparison, Increment); template constexpr bool operator==(DevicePixels left, T right) { return left.value() == right; } template constexpr bool operator!=(DevicePixels left, T right) { return left.value() != right; } template constexpr bool operator>(DevicePixels left, T right) { return left.value() > right; } template constexpr bool operator<(DevicePixels left, T right) { return left.value() < right; } template constexpr bool operator>=(DevicePixels left, T right) { return left.value() >= right; } template constexpr bool operator<=(DevicePixels left, T right) { return left.value() <= right; } template constexpr DevicePixels operator*(DevicePixels left, T right) { return left.value() * right; } template constexpr DevicePixels operator*(T left, DevicePixels right) { return right * left; } template constexpr DevicePixels operator/(DevicePixels left, T right) { return left.value() / right; } template constexpr DevicePixels operator%(DevicePixels left, T right) { return left.value() % right; } class CSSPixelFraction; /// CSSPixels: A position or length in CSS "reference pixels", independent of zoom or screen DPI. /// See https://www.w3.org/TR/css-values-3/#reference-pixel class CSSPixels { public: static constexpr i32 fractional_bits = 6; static constexpr i32 fixed_point_denominator = 1 << fractional_bits; static constexpr i32 radix_mask = fixed_point_denominator - 1; static constexpr i32 max_integer_value = NumericLimits::max() >> fractional_bits; static constexpr i32 min_integer_value = NumericLimits::min() >> fractional_bits; constexpr CSSPixels() = default; template constexpr CSSPixels(I value) { if (value > max_integer_value) [[unlikely]] m_value = NumericLimits::max(); else if (value < min_integer_value) [[unlikely]] m_value = NumericLimits::min(); else m_value = static_cast(value) << fractional_bits; } template explicit CSSPixels(F value) { *this = nearest_value_for(value); } template static CSSPixels nearest_value_for(F value) { i32 raw_value = 0; if (!isnan(value)) raw_value = AK::clamp_to(value * fixed_point_denominator); // Note: The resolution of CSSPixels is 0.015625, so care must be taken when converting // floats/doubles to CSSPixels as small values (such as scale factors) can underflow to zero, // or otherwise produce inaccurate results (when scaled back up). if (raw_value == 0 && value != 0) dbgln_if(LIBWEB_CSS_DEBUG, "CSSPixels: Conversion from float or double underflowed to zero"); return from_raw(raw_value); } template static CSSPixels floored_value_for(F value) { i32 raw_value = 0; if (!isnan(value)) raw_value = AK::clamp_to(floor(value * fixed_point_denominator)); return from_raw(raw_value); } template constexpr CSSPixels(U value) { if (value > max_integer_value) [[unlikely]] m_value = NumericLimits::max(); else m_value = static_cast(value) << fractional_bits; } static constexpr CSSPixels from_raw(int value) { CSSPixels res; res.set_raw_value(value); return res; } static constexpr CSSPixels min() { return from_raw(NumericLimits::min()); } static constexpr CSSPixels max() { return from_raw(NumericLimits::max()); } static constexpr CSSPixels smallest_positive_value() { return from_raw(1); } float to_float() const; double to_double() const; int to_int() const; constexpr int raw_value() const { return m_value; } constexpr void set_raw_value(int value) { m_value = value; } constexpr bool might_be_saturated() const { return raw_value() == NumericLimits::max() || raw_value() == NumericLimits::min(); } constexpr bool operator==(CSSPixels const& other) const = default; explicit operator double() const { return to_double(); } explicit operator float() const { return to_float(); } explicit operator int() const { return to_int(); } constexpr CSSPixels& operator++() { m_value = Checked::saturating_add(m_value, fixed_point_denominator); return *this; } constexpr CSSPixels& operator--() { m_value = Checked::saturating_sub(m_value, fixed_point_denominator); return *this; } constexpr int operator<=>(CSSPixels const& other) const { return raw_value() > other.raw_value() ? 1 : raw_value() < other.raw_value() ? -1 : 0; } constexpr CSSPixels operator+() const { return from_raw(+raw_value()); } constexpr CSSPixels operator-() const { return from_raw(-raw_value()); } constexpr CSSPixels operator+(CSSPixels const& other) const { return from_raw(Checked::saturating_add(raw_value(), other.raw_value())); } constexpr CSSPixels operator-(CSSPixels const& other) const { return from_raw(Checked::saturating_sub(raw_value(), other.raw_value())); } constexpr CSSPixels operator*(CSSPixels const& other) const { i64 value = raw_value(); value *= other.raw_value(); int int_value = AK::clamp_to(value >> fractional_bits); // Rounding: // If last bit cut off was 1: if (value & (1u << (fractional_bits - 1))) { // If any bit after was 1 as well if (value & (radix_mask >> 1u)) { // We need to round away from 0 int_value = Checked::saturating_add(int_value, 1); } else { // Otherwise we round to the next even value // Which means we add the least significant bit of the raw integer value int_value = Checked::saturating_add(int_value, int_value & 1); } } return from_raw(int_value); } constexpr CSSPixels operator*(CSSPixelFraction const& other) const; constexpr CSSPixelFraction operator/(CSSPixels const& other) const; constexpr CSSPixels operator/(CSSPixelFraction const& other) const; constexpr CSSPixels& operator+=(CSSPixels const& other) { *this = *this + other; return *this; } constexpr CSSPixels& operator-=(CSSPixels const& other) { *this = *this - other; return *this; } constexpr CSSPixels& operator*=(CSSPixels const& other) { *this = *this * other; return *this; } constexpr CSSPixels& operator*=(CSSPixelFraction const& other) { *this = *this * other; return *this; } constexpr CSSPixels& operator/=(CSSPixels const& other) { *this = *this * other; return *this; } constexpr CSSPixels abs() const { return from_raw(::abs(m_value)); } CSSPixels& scale_by(float value) { *this = CSSPixels(to_float() * value); return *this; } CSSPixels& scale_by(double value) { *this = CSSPixels(to_double() * value); return *this; } CSSPixels scaled(float value) const { auto result = *this; result.scale_by(value); return result; } CSSPixels scaled(double value) const { auto result = *this; result.scale_by(value); return result; } private: i32 m_value { 0 }; }; template constexpr bool operator==(CSSPixels left, T right) { return left == CSSPixels(right); } inline bool operator==(CSSPixels left, float right) { return left.to_float() == right; } inline bool operator==(CSSPixels left, double right) { return left.to_double() == right; } template constexpr bool operator>(CSSPixels left, T right) { return left > CSSPixels(right); } inline bool operator>(CSSPixels left, float right) { return left.to_float() > right; } inline bool operator>(CSSPixels left, double right) { return left.to_double() > right; } template constexpr bool operator<(CSSPixels left, T right) { return left < CSSPixels(right); } inline bool operator<(CSSPixels left, float right) { return left.to_float() < right; } inline bool operator<(CSSPixels left, double right) { return left.to_double() < right; } template constexpr CSSPixels operator*(CSSPixels left, T right) { return left * CSSPixels(right); } inline float operator*(CSSPixels left, float right) { return left.to_float() * right; } inline double operator*(CSSPixels left, double right) { return left.to_double() * right; } template constexpr CSSPixels operator*(T left, CSSPixels right) { return CSSPixels(left) * right; } inline float operator*(float left, CSSPixels right) { return right.to_float() * left; } inline double operator*(double left, CSSPixels right) { return right.to_double() * left; } class CSSPixelFraction { public: constexpr CSSPixelFraction(CSSPixels numerator, CSSPixels denominator) : m_numerator(numerator) , m_denominator(denominator) { VERIFY(denominator != 0); } explicit constexpr CSSPixelFraction(CSSPixels value) : m_numerator(value) , m_denominator(1) { } template constexpr CSSPixelFraction(I numerator, I denominator = 1) : m_numerator(numerator) , m_denominator(denominator) { VERIFY(denominator != 0); } template constexpr CSSPixelFraction(F numerator, F denominator = 1) { if (CSSPixels::nearest_value_for(denominator) == 0) { numerator = numerator / denominator; denominator = 1; } m_numerator = CSSPixels(numerator); m_denominator = CSSPixels(denominator); VERIFY(denominator != 0); } constexpr operator CSSPixels() const { i64 wide_value = m_numerator.raw_value(); wide_value <<= CSSPixels::fractional_bits; wide_value /= m_denominator.raw_value(); return CSSPixels::from_raw(AK::clamp_to(wide_value)); } constexpr CSSPixels operator-(CSSPixels const& other) const { return CSSPixels(*this) - other; } constexpr CSSPixels operator+(CSSPixels const& other) const { return CSSPixels(*this) + other; } constexpr CSSPixelFraction operator-() const { return CSSPixelFraction(-numerator(), denominator()); } constexpr int operator<=>(CSSPixelFraction const& other) const { auto left = static_cast(m_numerator.raw_value()) * other.m_denominator.raw_value(); auto right = static_cast(other.m_numerator.raw_value()) * m_denominator.raw_value(); if (left > right) return 1; if (left < right) return -1; return 0; } template constexpr int operator<=>(I const& other) const { return *this <=> CSSPixelFraction(other); } constexpr CSSPixels numerator() const { return m_numerator; } constexpr CSSPixels denominator() const { return m_denominator; } float to_float() const { return CSSPixels(*this).to_float(); } double to_double() const { return CSSPixels(*this).to_double(); } int to_int() const { return CSSPixels(*this).to_int(); } bool might_be_saturated() const { return CSSPixels(*this).might_be_saturated(); } private: CSSPixels m_numerator; CSSPixels m_denominator; }; constexpr CSSPixels CSSPixels::operator*(CSSPixelFraction const& other) const { i64 wide_value = raw_value(); wide_value *= other.numerator().raw_value(); wide_value /= other.denominator().raw_value(); return CSSPixels::from_raw(AK::clamp_to(wide_value)); } constexpr CSSPixelFraction CSSPixels::operator/(CSSPixels const& other) const { return CSSPixelFraction(*this, other); } constexpr CSSPixels CSSPixels::operator/(CSSPixelFraction const& other) const { i64 wide_value = raw_value(); wide_value *= other.denominator().raw_value(); wide_value /= other.numerator().raw_value(); return CSSPixels::from_raw(AK::clamp_to(wide_value)); } template constexpr CSSPixelFraction operator/(CSSPixels left, T right) { return left / CSSPixels(right); } inline float operator/(CSSPixels left, float right) { return left.to_float() / right; } inline double operator/(CSSPixels left, double right) { return left.to_double() / right; } using CSSPixelLine = Gfx::Line; using CSSPixelPoint = Gfx::Point; using CSSPixelRect = Gfx::Rect; using CSSPixelSize = Gfx::Size; using DevicePixelLine = Gfx::Line; using DevicePixelPoint = Gfx::Point; using DevicePixelRect = Gfx::Rect; using DevicePixelSize = Gfx::Size; } constexpr Web::CSSPixels abs(Web::CSSPixels const& value) { return value.abs(); } constexpr Web::CSSPixels floor(Web::CSSPixels const& value) { return Web::CSSPixels::from_raw(value.raw_value() & ~Web::CSSPixels::radix_mask); } constexpr Web::CSSPixels ceil(Web::CSSPixels const& value) { auto floor_value = value.raw_value() & ~Web::CSSPixels::radix_mask; auto ceil_value = floor_value + (value.raw_value() & Web::CSSPixels::radix_mask ? Web::CSSPixels::fixed_point_denominator : 0); return Web::CSSPixels::from_raw(ceil_value); } constexpr Web::CSSPixels round(Web::CSSPixels const& value) { // FIXME: Maybe do this with bit-fiddling instead if (value > 0) return floor(value + Web::CSSPixels::from_raw(Web::CSSPixels::fixed_point_denominator >> 1 /* 0.5 */)); return ceil(value - Web::CSSPixels::from_raw(Web::CSSPixels::fixed_point_denominator >> 1 /* 0.5 */)); } inline Web::CSSPixels sqrt(Web::CSSPixels const& value) { return Web::CSSPixels::nearest_value_for(AK::sqrt(value.to_float())); } constexpr Web::DevicePixels abs(Web::DevicePixels const& value) { return AK::abs(value.value()); } constexpr Web::CSSPixels square_distance_between(Web::CSSPixelPoint const& a, Web::CSSPixelPoint const& b) { auto delta_x = abs(a.x() - b.x()); auto delta_y = abs(a.y() - b.y()); return delta_x * delta_x + delta_y * delta_y; } template<> template<> [[nodiscard]] ALWAYS_INLINE Web::CSSPixelRect Web::CSSPixelRect::to_rounded() const { return { round(x()), round(y()), round(width()), round(height()), }; } namespace AK { template<> struct Traits : public DefaultTraits { static unsigned hash(Web::CSSPixels const& key) { return Traits::hash(key.raw_value()); } static bool equals(Web::CSSPixels const& a, Web::CSSPixels const& b) { return a == b; } }; template<> struct Traits : public DefaultTraits { static unsigned hash(Web::DevicePixels const& key) { return Traits::hash(key.value()); } static bool equals(Web::DevicePixels const& a, Web::DevicePixels const& b) { return a == b; } }; template<> struct Formatter : Formatter { ErrorOr format(FormatBuilder& builder, Web::CSSPixels const& value) { return Formatter::format(builder, value.to_double()); } }; template<> struct Formatter : Formatter { ErrorOr format(FormatBuilder& builder, Web::DevicePixels const& value) { return Formatter::format(builder, value.value()); } }; } namespace IPC { template<> ErrorOr encode(Encoder& encoder, Web::DevicePixels const& value); template<> ErrorOr decode(Decoder& decoder); template<> ErrorOr encode(Encoder& encoder, Web::DevicePixelPoint const& value); template<> ErrorOr decode(Decoder& decoder); template<> ErrorOr encode(Encoder& encoder, Web::DevicePixelSize const& value); template<> ErrorOr decode(Decoder& decoder); template<> ErrorOr encode(Encoder& encoder, Web::DevicePixelRect const& value); template<> ErrorOr decode(Decoder& decoder); }