SVGGradientElement.cpp 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141
  1. /*
  2. * Copyright (c) 2023, MacDue <macdue@dueutil.tech>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <LibWeb/Bindings/Intrinsics.h>
  7. #include <LibWeb/DOM/Document.h>
  8. #include <LibWeb/SVG/AttributeNames.h>
  9. #include <LibWeb/SVG/SVGGradientElement.h>
  10. #include <LibWeb/SVG/SVGGraphicsElement.h>
  11. namespace Web::SVG {
  12. SVGGradientElement::SVGGradientElement(DOM::Document& document, DOM::QualifiedName qualified_name)
  13. : SVGElement(document, move(qualified_name))
  14. {
  15. }
  16. void SVGGradientElement::attribute_changed(FlyString const& name, Optional<String> const& value)
  17. {
  18. SVGElement::attribute_changed(name, value);
  19. if (name == AttributeNames::gradientUnits) {
  20. m_gradient_units = AttributeParser::parse_units(value.value_or(String {}));
  21. } else if (name == AttributeNames::spreadMethod) {
  22. m_spread_method = AttributeParser::parse_spread_method(value.value_or(String {}));
  23. } else if (name == AttributeNames::gradientTransform) {
  24. if (auto transform_list = AttributeParser::parse_transform(value.value_or(String {})); transform_list.has_value()) {
  25. m_gradient_transform = transform_from_transform_list(*transform_list);
  26. } else {
  27. m_gradient_transform = {};
  28. }
  29. }
  30. }
  31. GradientUnits SVGGradientElement::gradient_units() const
  32. {
  33. HashTable<SVGGradientElement const*> seen_gradients;
  34. return gradient_units_impl(seen_gradients);
  35. }
  36. GradientUnits SVGGradientElement::gradient_units_impl(HashTable<SVGGradientElement const*>& seen_gradients) const
  37. {
  38. if (m_gradient_units.has_value())
  39. return *m_gradient_units;
  40. if (auto gradient = linked_gradient(seen_gradients))
  41. return gradient->gradient_units_impl(seen_gradients);
  42. return GradientUnits::ObjectBoundingBox;
  43. }
  44. SpreadMethod SVGGradientElement::spread_method() const
  45. {
  46. HashTable<SVGGradientElement const*> seen_gradients;
  47. return spread_method_impl(seen_gradients);
  48. }
  49. SpreadMethod SVGGradientElement::spread_method_impl(HashTable<SVGGradientElement const*>& seen_gradients) const
  50. {
  51. if (m_spread_method.has_value())
  52. return *m_spread_method;
  53. if (auto gradient = linked_gradient(seen_gradients))
  54. return gradient->spread_method_impl(seen_gradients);
  55. return SpreadMethod::Pad;
  56. }
  57. Optional<Gfx::AffineTransform> SVGGradientElement::gradient_transform() const
  58. {
  59. HashTable<SVGGradientElement const*> seen_gradients;
  60. return gradient_transform_impl(seen_gradients);
  61. }
  62. Optional<Gfx::AffineTransform> SVGGradientElement::gradient_transform_impl(HashTable<SVGGradientElement const*>& seen_gradients) const
  63. {
  64. if (m_gradient_transform.has_value())
  65. return m_gradient_transform;
  66. if (auto gradient = linked_gradient(seen_gradients))
  67. return gradient->gradient_transform_impl(seen_gradients);
  68. return {};
  69. }
  70. // The gradient transform, appropriately scaled and combined with the paint transform.
  71. Gfx::AffineTransform SVGGradientElement::gradient_paint_transform(SVGPaintContext const& paint_context) const
  72. {
  73. auto transform = gradient_transform().value_or(Gfx::AffineTransform {});
  74. if (gradient_units() == GradientUnits::ObjectBoundingBox) {
  75. // Adjust transform to take place in the coordinate system defined by the bounding box:
  76. return Gfx::AffineTransform { paint_context.transform }
  77. .translate(paint_context.path_bounding_box.location())
  78. .scale(paint_context.path_bounding_box.width(), paint_context.path_bounding_box.height())
  79. .multiply(transform);
  80. }
  81. return Gfx::AffineTransform { paint_context.transform }.multiply(transform);
  82. }
  83. void SVGGradientElement::add_color_stops(Gfx::SVGGradientPaintStyle& paint_style) const
  84. {
  85. for_each_color_stop([&](auto& stop) {
  86. // https://svgwg.org/svg2-draft/pservers.html#StopNotes
  87. // Gradient offset values less than 0 (or less than 0%) are rounded up to 0%.
  88. // Gradient offset values greater than 1 (or greater than 100%) are rounded down to 100%.
  89. float stop_offset = AK::clamp(stop.stop_offset().value(), 0.0f, 1.0f);
  90. // FIXME: Each gradient offset value is required to be equal to or greater than the previous gradient
  91. // stop's offset value. If a given gradient stop's offset value is not equal to or greater than all
  92. // previous offset values, then the offset value is adjusted to be equal to the largest of all previous
  93. // offset values.
  94. paint_style.add_color_stop(stop_offset, stop.stop_color().with_opacity(stop.stop_opacity())).release_value_but_fixme_should_propagate_errors();
  95. });
  96. }
  97. JS::GCPtr<SVGGradientElement const> SVGGradientElement::linked_gradient(HashTable<SVGGradientElement const*>& seen_gradients) const
  98. {
  99. // FIXME: This entire function is an ad-hoc hack!
  100. // It can only resolve #<ids> in the same document.
  101. auto link = has_attribute(AttributeNames::href) ? get_attribute(AttributeNames::href) : get_attribute("xlink:href"_fly_string);
  102. if (auto href = link; href.has_value() && !link->is_empty()) {
  103. auto url = document().parse_url(*href);
  104. auto id = url.fragment();
  105. if (!id.has_value() || id->is_empty())
  106. return {};
  107. auto element = document().get_element_by_id(id.value());
  108. if (!element)
  109. return {};
  110. if (element == this)
  111. return {};
  112. if (!is<SVGGradientElement>(*element))
  113. return {};
  114. if (seen_gradients.set(&verify_cast<SVGGradientElement>(*element)) != AK::HashSetResult::InsertedNewEntry)
  115. return {};
  116. return &verify_cast<SVGGradientElement>(*element);
  117. }
  118. return {};
  119. }
  120. void SVGGradientElement::initialize(JS::Realm& realm)
  121. {
  122. Base::initialize(realm);
  123. WEB_SET_PROTOTYPE_FOR_INTERFACE(SVGGradientElement);
  124. }
  125. }