CalculatedOr.h 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186
  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/Flex.h>
  10. #include <LibWeb/CSS/Frequency.h>
  11. #include <LibWeb/CSS/Length.h>
  12. #include <LibWeb/CSS/Percentage.h>
  13. #include <LibWeb/CSS/Resolution.h>
  14. #include <LibWeb/CSS/StyleValues/CalculatedStyleValue.h>
  15. #include <LibWeb/CSS/Time.h>
  16. namespace Web::CSS {
  17. template<typename T>
  18. class CalculatedOr {
  19. public:
  20. CalculatedOr(T t)
  21. : m_value(move(t))
  22. {
  23. }
  24. CalculatedOr(NonnullRefPtr<CalculatedStyleValue> calculated)
  25. : m_value(move(calculated))
  26. {
  27. }
  28. virtual ~CalculatedOr() = default;
  29. bool is_calculated() const { return m_value.template has<NonnullRefPtr<CalculatedStyleValue>>(); }
  30. T const& value() const
  31. {
  32. VERIFY(!is_calculated());
  33. return m_value.template get<T>();
  34. }
  35. NonnullRefPtr<CalculatedStyleValue> const& calculated() const
  36. {
  37. VERIFY(is_calculated());
  38. return m_value.template get<NonnullRefPtr<CalculatedStyleValue>>();
  39. }
  40. virtual T resolve_calculated(NonnullRefPtr<CalculatedStyleValue> const&, Layout::Node const&) const = 0;
  41. T resolved(Layout::Node const& layout_node) const
  42. {
  43. return m_value.visit(
  44. [&](T const& t) {
  45. return t;
  46. },
  47. [&](NonnullRefPtr<CalculatedStyleValue> const& calculated) {
  48. return resolve_calculated(calculated, layout_node);
  49. });
  50. }
  51. String to_string() const
  52. {
  53. if (is_calculated())
  54. return m_value.template get<NonnullRefPtr<CalculatedStyleValue>>()->to_string();
  55. return m_value.template get<T>().to_string();
  56. }
  57. bool operator==(CalculatedOr<T> const& other) const
  58. {
  59. if (is_calculated() || other.is_calculated())
  60. return false;
  61. return (m_value.template get<T>() == other.m_value.template get<T>());
  62. }
  63. private:
  64. Variant<T, NonnullRefPtr<CalculatedStyleValue>> m_value;
  65. };
  66. class AngleOrCalculated : public CalculatedOr<Angle> {
  67. public:
  68. using CalculatedOr<Angle>::CalculatedOr;
  69. Angle resolve_calculated(NonnullRefPtr<CalculatedStyleValue> const&, Layout::Node const&) const override;
  70. };
  71. class FlexOrCalculated : public CalculatedOr<Flex> {
  72. public:
  73. using CalculatedOr<Flex>::CalculatedOr;
  74. Flex resolve_calculated(NonnullRefPtr<CalculatedStyleValue> const&, Layout::Node const&) const override;
  75. };
  76. class FrequencyOrCalculated : public CalculatedOr<Frequency> {
  77. public:
  78. using CalculatedOr<Frequency>::CalculatedOr;
  79. Frequency resolve_calculated(NonnullRefPtr<CalculatedStyleValue> const&, Layout::Node const&) const override;
  80. };
  81. class IntegerOrCalculated : public CalculatedOr<i64> {
  82. public:
  83. using CalculatedOr<i64>::CalculatedOr;
  84. i64 resolve_calculated(NonnullRefPtr<CalculatedStyleValue> const&, Layout::Node const&) const override;
  85. };
  86. class LengthOrCalculated : public CalculatedOr<Length> {
  87. public:
  88. using CalculatedOr<Length>::CalculatedOr;
  89. Length resolve_calculated(NonnullRefPtr<CalculatedStyleValue> const&, Layout::Node const&) const override;
  90. [[nodiscard]] Length resolved(Length::ResolutionContext const&) const;
  91. };
  92. class NumberOrCalculated : public CalculatedOr<double> {
  93. public:
  94. using CalculatedOr<double>::CalculatedOr;
  95. double resolve_calculated(NonnullRefPtr<CalculatedStyleValue> const&, Layout::Node const&) const override;
  96. };
  97. class PercentageOrCalculated : public CalculatedOr<Percentage> {
  98. public:
  99. using CalculatedOr<Percentage>::CalculatedOr;
  100. Percentage resolve_calculated(NonnullRefPtr<CalculatedStyleValue> const&, Layout::Node const&) const override;
  101. };
  102. class ResolutionOrCalculated : public CalculatedOr<Resolution> {
  103. public:
  104. using CalculatedOr<Resolution>::CalculatedOr;
  105. Resolution resolve_calculated(NonnullRefPtr<CalculatedStyleValue> const&, Layout::Node const&) const override;
  106. };
  107. class TimeOrCalculated : public CalculatedOr<Time> {
  108. public:
  109. using CalculatedOr<Time>::CalculatedOr;
  110. Time resolve_calculated(NonnullRefPtr<CalculatedStyleValue> const&, Layout::Node const&) const override;
  111. };
  112. }
  113. template<>
  114. struct AK::Formatter<Web::CSS::AngleOrCalculated> : Formatter<StringView> {
  115. ErrorOr<void> format(FormatBuilder& builder, Web::CSS::AngleOrCalculated const& calculated_or)
  116. {
  117. return Formatter<StringView>::format(builder, calculated_or.to_string());
  118. }
  119. };
  120. template<>
  121. struct AK::Formatter<Web::CSS::FrequencyOrCalculated> : Formatter<StringView> {
  122. ErrorOr<void> format(FormatBuilder& builder, Web::CSS::FrequencyOrCalculated const& calculated_or)
  123. {
  124. return Formatter<StringView>::format(builder, calculated_or.to_string());
  125. }
  126. };
  127. template<>
  128. struct AK::Formatter<Web::CSS::LengthOrCalculated> : Formatter<StringView> {
  129. ErrorOr<void> format(FormatBuilder& builder, Web::CSS::LengthOrCalculated const& calculated_or)
  130. {
  131. return Formatter<StringView>::format(builder, calculated_or.to_string());
  132. }
  133. };
  134. template<>
  135. struct AK::Formatter<Web::CSS::PercentageOrCalculated> : Formatter<StringView> {
  136. ErrorOr<void> format(FormatBuilder& builder, Web::CSS::PercentageOrCalculated const& calculated_or)
  137. {
  138. return Formatter<StringView>::format(builder, calculated_or.to_string());
  139. }
  140. };
  141. template<>
  142. struct AK::Formatter<Web::CSS::TimeOrCalculated> : Formatter<StringView> {
  143. ErrorOr<void> format(FormatBuilder& builder, Web::CSS::TimeOrCalculated const& calculated_or)
  144. {
  145. return Formatter<StringView>::format(builder, calculated_or.to_string());
  146. }
  147. };