CalculatedOr.h 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156
  1. /*
  2. * Copyright (c) 2023, Sam Atkins <atkinssj@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #pragma once
  7. #include <AK/Variant.h>
  8. #include <LibWeb/CSS/Angle.h>
  9. #include <LibWeb/CSS/Frequency.h>
  10. #include <LibWeb/CSS/Length.h>
  11. #include <LibWeb/CSS/Percentage.h>
  12. #include <LibWeb/CSS/StyleValues/CalculatedStyleValue.h>
  13. #include <LibWeb/CSS/Time.h>
  14. namespace Web::CSS {
  15. template<typename T>
  16. class CalculatedOr {
  17. public:
  18. CalculatedOr(T t)
  19. : m_value(move(t))
  20. {
  21. }
  22. CalculatedOr(NonnullRefPtr<CalculatedStyleValue> calculated)
  23. : m_value(move(calculated))
  24. {
  25. }
  26. virtual ~CalculatedOr() = default;
  27. bool is_calculated() const { return m_value.template has<NonnullRefPtr<CalculatedStyleValue>>(); }
  28. T const& value() const
  29. {
  30. VERIFY(!is_calculated());
  31. return m_value.template get<T>();
  32. }
  33. NonnullRefPtr<CalculatedStyleValue> const& calculated() const
  34. {
  35. VERIFY(is_calculated());
  36. return m_value.template get<NonnullRefPtr<CalculatedStyleValue>>();
  37. }
  38. virtual T resolve_calculated(NonnullRefPtr<CalculatedStyleValue> const&, Layout::Node const&) const = 0;
  39. T resolved(Layout::Node const& layout_node) const
  40. {
  41. return m_value.visit(
  42. [&](T const& t) {
  43. return t;
  44. },
  45. [&](NonnullRefPtr<CalculatedStyleValue> const& calculated) {
  46. return resolve_calculated(calculated, layout_node);
  47. });
  48. }
  49. String to_string() const
  50. {
  51. if (is_calculated())
  52. return m_value.template get<NonnullRefPtr<CalculatedStyleValue>>()->to_string();
  53. return m_value.template get<T>().to_string();
  54. }
  55. bool operator==(CalculatedOr<T> const& other) const
  56. {
  57. if (is_calculated() || other.is_calculated())
  58. return false;
  59. return (m_value.template get<T>() == other.m_value.template get<T>());
  60. }
  61. private:
  62. Variant<T, NonnullRefPtr<CalculatedStyleValue>> m_value;
  63. };
  64. class AngleOrCalculated : public CalculatedOr<Angle> {
  65. public:
  66. using CalculatedOr<Angle>::CalculatedOr;
  67. Angle resolve_calculated(NonnullRefPtr<CalculatedStyleValue> const&, Layout::Node const&) const override;
  68. };
  69. class FrequencyOrCalculated : public CalculatedOr<Frequency> {
  70. public:
  71. using CalculatedOr<Frequency>::CalculatedOr;
  72. Frequency resolve_calculated(NonnullRefPtr<CalculatedStyleValue> const&, Layout::Node const&) const override;
  73. };
  74. class LengthOrCalculated : public CalculatedOr<Length> {
  75. public:
  76. using CalculatedOr<Length>::CalculatedOr;
  77. Length resolve_calculated(NonnullRefPtr<CalculatedStyleValue> const&, Layout::Node const&) const override;
  78. [[nodiscard]] Length resolved(Length::ResolutionContext const&) const;
  79. };
  80. class PercentageOrCalculated : public CalculatedOr<Percentage> {
  81. public:
  82. using CalculatedOr<Percentage>::CalculatedOr;
  83. Percentage resolve_calculated(NonnullRefPtr<CalculatedStyleValue> const&, Layout::Node const&) const override;
  84. };
  85. class TimeOrCalculated : public CalculatedOr<Time> {
  86. public:
  87. using CalculatedOr<Time>::CalculatedOr;
  88. Time resolve_calculated(NonnullRefPtr<CalculatedStyleValue> const&, Layout::Node const&) const override;
  89. };
  90. }
  91. template<>
  92. struct AK::Formatter<Web::CSS::AngleOrCalculated> : Formatter<StringView> {
  93. ErrorOr<void> format(FormatBuilder& builder, Web::CSS::AngleOrCalculated const& calculated_or)
  94. {
  95. return Formatter<StringView>::format(builder, calculated_or.to_string());
  96. }
  97. };
  98. template<>
  99. struct AK::Formatter<Web::CSS::FrequencyOrCalculated> : Formatter<StringView> {
  100. ErrorOr<void> format(FormatBuilder& builder, Web::CSS::FrequencyOrCalculated const& calculated_or)
  101. {
  102. return Formatter<StringView>::format(builder, calculated_or.to_string());
  103. }
  104. };
  105. template<>
  106. struct AK::Formatter<Web::CSS::LengthOrCalculated> : Formatter<StringView> {
  107. ErrorOr<void> format(FormatBuilder& builder, Web::CSS::LengthOrCalculated const& calculated_or)
  108. {
  109. return Formatter<StringView>::format(builder, calculated_or.to_string());
  110. }
  111. };
  112. template<>
  113. struct AK::Formatter<Web::CSS::PercentageOrCalculated> : Formatter<StringView> {
  114. ErrorOr<void> format(FormatBuilder& builder, Web::CSS::PercentageOrCalculated const& calculated_or)
  115. {
  116. return Formatter<StringView>::format(builder, calculated_or.to_string());
  117. }
  118. };
  119. template<>
  120. struct AK::Formatter<Web::CSS::TimeOrCalculated> : Formatter<StringView> {
  121. ErrorOr<void> format(FormatBuilder& builder, Web::CSS::TimeOrCalculated const& calculated_or)
  122. {
  123. return Formatter<StringView>::format(builder, calculated_or.to_string());
  124. }
  125. };