CalculatedOr.h 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155
  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. ErrorOr<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. };
  79. class PercentageOrCalculated : public CalculatedOr<Percentage> {
  80. public:
  81. using CalculatedOr<Percentage>::CalculatedOr;
  82. Percentage resolve_calculated(NonnullRefPtr<CalculatedStyleValue> const&, Layout::Node const&) const override;
  83. };
  84. class TimeOrCalculated : public CalculatedOr<Time> {
  85. public:
  86. using CalculatedOr<Time>::CalculatedOr;
  87. Time resolve_calculated(NonnullRefPtr<CalculatedStyleValue> const&, Layout::Node const&) const override;
  88. };
  89. }
  90. template<>
  91. struct AK::Formatter<Web::CSS::AngleOrCalculated> : Formatter<StringView> {
  92. ErrorOr<void> format(FormatBuilder& builder, Web::CSS::AngleOrCalculated const& calculated_or)
  93. {
  94. return Formatter<StringView>::format(builder, TRY(calculated_or.to_string()));
  95. }
  96. };
  97. template<>
  98. struct AK::Formatter<Web::CSS::FrequencyOrCalculated> : Formatter<StringView> {
  99. ErrorOr<void> format(FormatBuilder& builder, Web::CSS::FrequencyOrCalculated const& calculated_or)
  100. {
  101. return Formatter<StringView>::format(builder, TRY(calculated_or.to_string()));
  102. }
  103. };
  104. template<>
  105. struct AK::Formatter<Web::CSS::LengthOrCalculated> : Formatter<StringView> {
  106. ErrorOr<void> format(FormatBuilder& builder, Web::CSS::LengthOrCalculated const& calculated_or)
  107. {
  108. return Formatter<StringView>::format(builder, TRY(calculated_or.to_string()));
  109. }
  110. };
  111. template<>
  112. struct AK::Formatter<Web::CSS::PercentageOrCalculated> : Formatter<StringView> {
  113. ErrorOr<void> format(FormatBuilder& builder, Web::CSS::PercentageOrCalculated const& calculated_or)
  114. {
  115. return Formatter<StringView>::format(builder, TRY(calculated_or.to_string()));
  116. }
  117. };
  118. template<>
  119. struct AK::Formatter<Web::CSS::TimeOrCalculated> : Formatter<StringView> {
  120. ErrorOr<void> format(FormatBuilder& builder, Web::CSS::TimeOrCalculated const& calculated_or)
  121. {
  122. return Formatter<StringView>::format(builder, TRY(calculated_or.to_string()));
  123. }
  124. };