HTMLProgressElement.cpp 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114
  1. /*
  2. * Copyright (c) 2020-2022, the SerenityOS developers.
  3. * Copyright (c) 2022, MacDue <macdue@dueutil.tech>
  4. *
  5. * SPDX-License-Identifier: BSD-2-Clause
  6. */
  7. #include <LibWeb/DOM/Document.h>
  8. #include <LibWeb/DOM/ShadowRoot.h>
  9. #include <LibWeb/HTML/HTMLProgressElement.h>
  10. #include <LibWeb/Layout/BlockContainer.h>
  11. #include <LibWeb/Layout/Node.h>
  12. #include <LibWeb/Layout/Progress.h>
  13. namespace Web::HTML {
  14. HTMLProgressElement::HTMLProgressElement(DOM::Document& document, DOM::QualifiedName qualified_name)
  15. : HTMLElement(document, move(qualified_name))
  16. {
  17. }
  18. HTMLProgressElement::~HTMLProgressElement() = default;
  19. void HTMLProgressElement::initialize(JS::Realm& realm)
  20. {
  21. Base::initialize(realm);
  22. set_prototype(&Bindings::ensure_web_prototype<Bindings::HTMLProgressElementPrototype>(realm, "HTMLProgressElement"));
  23. }
  24. JS::GCPtr<Layout::Node> HTMLProgressElement::create_layout_node(NonnullRefPtr<CSS::StyleProperties> style)
  25. {
  26. if (style->appearance().value_or(CSS::Appearance::Auto) == CSS::Appearance::None)
  27. return HTMLElement::create_layout_node(style);
  28. return heap().allocate_without_realm<Layout::Progress>(document(), *this, move(style));
  29. }
  30. bool HTMLProgressElement::using_system_appearance() const
  31. {
  32. if (layout_node())
  33. return is<Layout::Progress>(*layout_node());
  34. return false;
  35. }
  36. void HTMLProgressElement::progress_position_updated()
  37. {
  38. if (using_system_appearance())
  39. layout_node()->set_needs_display();
  40. else
  41. document().invalidate_layout();
  42. }
  43. double HTMLProgressElement::value() const
  44. {
  45. auto const& value_characters = deprecated_attribute(HTML::AttributeNames::value);
  46. if (value_characters == nullptr)
  47. return 0;
  48. // https://html.spec.whatwg.org/multipage/common-microsyntaxes.html#rules-for-parsing-floating-point-number-values
  49. // 6. Skip ASCII whitespace within input given position.
  50. auto maybe_double = value_characters.to_double(AK::TrimWhitespace::Yes);
  51. if (!maybe_double.has_value())
  52. return 0;
  53. if (!isfinite(maybe_double.value()) || maybe_double.value() < 0)
  54. return 0;
  55. return min(maybe_double.value(), max());
  56. }
  57. WebIDL::ExceptionOr<void> HTMLProgressElement::set_value(double value)
  58. {
  59. if (value < 0)
  60. return {};
  61. TRY(set_attribute(HTML::AttributeNames::value, DeprecatedString::number(value)));
  62. progress_position_updated();
  63. return {};
  64. }
  65. double HTMLProgressElement::max() const
  66. {
  67. auto const& max_characters = deprecated_attribute(HTML::AttributeNames::max);
  68. if (max_characters == nullptr)
  69. return 1;
  70. // https://html.spec.whatwg.org/multipage/common-microsyntaxes.html#rules-for-parsing-floating-point-number-values
  71. // 6. Skip ASCII whitespace within input given position.
  72. auto double_or_none = max_characters.to_double(AK::TrimWhitespace::Yes);
  73. if (!double_or_none.has_value())
  74. return 1;
  75. if (!isfinite(double_or_none.value()) || double_or_none.value() <= 0)
  76. return 1;
  77. return double_or_none.value();
  78. }
  79. WebIDL::ExceptionOr<void> HTMLProgressElement::set_max(double value)
  80. {
  81. if (value <= 0)
  82. return {};
  83. TRY(set_attribute(HTML::AttributeNames::max, DeprecatedString::number(value)));
  84. progress_position_updated();
  85. return {};
  86. }
  87. double HTMLProgressElement::position() const
  88. {
  89. if (!is_determinate())
  90. return -1;
  91. return value() / max();
  92. }
  93. }