Browse Source

LibWeb: Add basic parse floating point number function

Bastiaan van der Plaat 1 year ago
parent
commit
761d824b72

+ 12 - 7
Userland/Libraries/LibWeb/HTML/HTMLInputElement.cpp

@@ -953,10 +953,8 @@ String HTMLInputElement::value_sanitization_algorithm(String const& value) const
         }
         }
     } else if (type_state() == HTMLInputElement::TypeAttributeState::Number) {
     } else if (type_state() == HTMLInputElement::TypeAttributeState::Number) {
         // If the value of the element is not a valid floating-point number, then set it to the empty string instead.
         // If the value of the element is not a valid floating-point number, then set it to the empty string instead.
-        // https://html.spec.whatwg.org/multipage/common-microsyntaxes.html#rules-for-parsing-floating-point-number-values
-        // 6. Skip ASCII whitespace within input given position.
-        auto maybe_double = value.bytes_as_string_view().to_double(TrimWhitespace::Yes);
-        if (!maybe_double.has_value() || !isfinite(maybe_double.value()))
+        auto maybe_value = parse_floating_point_number(value);
+        if (!maybe_value.has_value() || !isfinite(maybe_value.value()))
             return String {};
             return String {};
     } else if (type_state() == HTMLInputElement::TypeAttributeState::Date) {
     } else if (type_state() == HTMLInputElement::TypeAttributeState::Date) {
         // https://html.spec.whatwg.org/multipage/input.html#date-state-(type=date):value-sanitization-algorithm
         // https://html.spec.whatwg.org/multipage/input.html#date-state-(type=date):value-sanitization-algorithm
@@ -981,9 +979,16 @@ String HTMLInputElement::value_sanitization_algorithm(String const& value) const
         return String {};
         return String {};
     } else if (type_state() == HTMLInputElement::TypeAttributeState::Range) {
     } else if (type_state() == HTMLInputElement::TypeAttributeState::Range) {
         // https://html.spec.whatwg.org/multipage/input.html#range-state-(type=range):value-sanitization-algorithm
         // https://html.spec.whatwg.org/multipage/input.html#range-state-(type=range):value-sanitization-algorithm
-        auto maybe_double = value.bytes_as_string_view().to_double(TrimWhitespace::Yes);
-        if (!maybe_double.has_value() || !isfinite(maybe_double.value()))
-            return JS::number_to_string(maybe_double.value_or(0));
+        // If the value of the element is not a valid floating-point number, then set it to the best representation, as a floating-point number, of the default value.
+        auto maybe_value = parse_floating_point_number(value);
+        if (!maybe_value.has_value() || !isfinite(maybe_value.value())) {
+            // The default value is the minimum plus half the difference between the minimum and the maximum, unless the maximum is less than the minimum, in which case the default value is the minimum.
+            auto min = parse_floating_point_number(get_attribute(HTML::AttributeNames::min).value_or("0"_string)).value_or(0);
+            auto max = parse_floating_point_number(get_attribute(HTML::AttributeNames::max).value_or("1"_string)).value_or(1);
+            if (max > min)
+                return JS::number_to_string(min);
+            return JS::number_to_string(min + (max - min) / 2);
+        }
     } else if (type_state() == HTMLInputElement::TypeAttributeState::Color) {
     } else if (type_state() == HTMLInputElement::TypeAttributeState::Color) {
         // https://html.spec.whatwg.org/multipage/input.html#color-state-(type=color):value-sanitization-algorithm
         // https://html.spec.whatwg.org/multipage/input.html#color-state-(type=color):value-sanitization-algorithm
         // If the value of the element is a valid simple color, then set it to the value of the element converted to ASCII lowercase;
         // If the value of the element is a valid simple color, then set it to the value of the element converted to ASCII lowercase;

+ 14 - 23
Userland/Libraries/LibWeb/HTML/HTMLProgressElement.cpp

@@ -8,6 +8,7 @@
 #include <LibWeb/DOM/Document.h>
 #include <LibWeb/DOM/Document.h>
 #include <LibWeb/DOM/ShadowRoot.h>
 #include <LibWeb/DOM/ShadowRoot.h>
 #include <LibWeb/HTML/HTMLProgressElement.h>
 #include <LibWeb/HTML/HTMLProgressElement.h>
+#include <LibWeb/HTML/Numbers.h>
 #include <LibWeb/Layout/BlockContainer.h>
 #include <LibWeb/Layout/BlockContainer.h>
 #include <LibWeb/Layout/Node.h>
 #include <LibWeb/Layout/Node.h>
 #include <LibWeb/Layout/Progress.h>
 #include <LibWeb/Layout/Progress.h>
@@ -51,26 +52,21 @@ void HTMLProgressElement::progress_position_updated()
         document().invalidate_layout();
         document().invalidate_layout();
 }
 }
 
 
+// https://html.spec.whatwg.org/multipage/form-elements.html#dom-progress-value
 double HTMLProgressElement::value() const
 double HTMLProgressElement::value() const
 {
 {
-    auto const& value_characters = deprecated_attribute(HTML::AttributeNames::value);
-    if (value_characters == nullptr)
+    auto maybe_value_string = get_attribute(HTML::AttributeNames::value);
+    if (!maybe_value_string.has_value())
         return 0;
         return 0;
-
-    // https://html.spec.whatwg.org/multipage/common-microsyntaxes.html#rules-for-parsing-floating-point-number-values
-    // 6. Skip ASCII whitespace within input given position.
-    auto maybe_double = value_characters.to_double(AK::TrimWhitespace::Yes);
-    if (!maybe_double.has_value())
-        return 0;
-    if (!isfinite(maybe_double.value()) || maybe_double.value() < 0)
+    auto maybe_value = parse_floating_point_number(maybe_value_string.value());
+    if (!maybe_value.has_value())
         return 0;
         return 0;
-
-    return min(maybe_double.value(), max());
+    return clamp(maybe_value.value(), 0, max());
 }
 }
 
 
 WebIDL::ExceptionOr<void> HTMLProgressElement::set_value(double value)
 WebIDL::ExceptionOr<void> HTMLProgressElement::set_value(double value)
 {
 {
-    if (value < 0)
+    if (value < 0 || value > max())
         return {};
         return {};
 
 
     TRY(set_attribute(HTML::AttributeNames::value, MUST(String::number(value))));
     TRY(set_attribute(HTML::AttributeNames::value, MUST(String::number(value))));
@@ -78,21 +74,16 @@ WebIDL::ExceptionOr<void> HTMLProgressElement::set_value(double value)
     return {};
     return {};
 }
 }
 
 
+// https://html.spec.whatwg.org/multipage/form-elements.html#dom-progress-max
 double HTMLProgressElement::max() const
 double HTMLProgressElement::max() const
 {
 {
-    auto const& max_characters = deprecated_attribute(HTML::AttributeNames::max);
-    if (max_characters == nullptr)
+    auto maybe_max_string = get_attribute(HTML::AttributeNames::max);
+    if (!maybe_max_string.has_value())
         return 1;
         return 1;
-
-    // https://html.spec.whatwg.org/multipage/common-microsyntaxes.html#rules-for-parsing-floating-point-number-values
-    // 6. Skip ASCII whitespace within input given position.
-    auto double_or_none = max_characters.to_double(AK::TrimWhitespace::Yes);
-    if (!double_or_none.has_value())
-        return 1;
-    if (!isfinite(double_or_none.value()) || double_or_none.value() <= 0)
+    auto maybe_max = parse_floating_point_number(maybe_max_string.value());
+    if (!maybe_max.has_value())
         return 1;
         return 1;
-
-    return double_or_none.value();
+    return AK::max(maybe_max.value(), 0);
 }
 }
 
 
 WebIDL::ExceptionOr<void> HTMLProgressElement::set_max(double value)
 WebIDL::ExceptionOr<void> HTMLProgressElement::set_max(double value)

+ 13 - 0
Userland/Libraries/LibWeb/HTML/Numbers.cpp

@@ -7,6 +7,7 @@
 #include <AK/GenericLexer.h>
 #include <AK/GenericLexer.h>
 #include <LibWeb/HTML/Numbers.h>
 #include <LibWeb/HTML/Numbers.h>
 #include <LibWeb/Infra/CharacterTypes.h>
 #include <LibWeb/Infra/CharacterTypes.h>
+#include <math.h>
 
 
 namespace Web::HTML {
 namespace Web::HTML {
 
 
@@ -79,4 +80,16 @@ Optional<u32> parse_non_negative_integer(StringView string)
     return static_cast<u32>(optional_value.value());
     return static_cast<u32>(optional_value.value());
 }
 }
 
 
+// https://html.spec.whatwg.org/multipage/common-microsyntaxes.html#rules-for-parsing-floating-point-number-values
+Optional<double> parse_floating_point_number(StringView string)
+{
+    // FIXME: Implement spec compliant floating point number parsing
+    auto maybe_double = MUST(String::from_utf8(string)).to_number<double>(TrimWhitespace::Yes);
+    if (!maybe_double.has_value())
+        return {};
+    if (!isfinite(maybe_double.value()))
+        return {};
+    return maybe_double.value();
+}
+
 }
 }

+ 2 - 0
Userland/Libraries/LibWeb/HTML/Numbers.h

@@ -15,4 +15,6 @@ Optional<i32> parse_integer(StringView string);
 
 
 Optional<u32> parse_non_negative_integer(StringView string);
 Optional<u32> parse_non_negative_integer(StringView string);
 
 
+Optional<double> parse_floating_point_number(StringView string);
+
 }
 }

+ 1 - 2
Userland/Libraries/LibWeb/Layout/TreeBuilder.cpp

@@ -402,8 +402,7 @@ ErrorOr<void> TreeBuilder::create_layout_tree(DOM::Node& dom_node, TreeBuilder::
             bar_style->set_property(CSS::PropertyID::Display, CSS::DisplayStyleValue::create(CSS::Display::from_short(CSS::Display::Short::FlowRoot)));
             bar_style->set_property(CSS::PropertyID::Display, CSS::DisplayStyleValue::create(CSS::Display::from_short(CSS::Display::Short::FlowRoot)));
             auto value_style = TRY(style_computer.compute_style(progress, CSS::Selector::PseudoElement::ProgressValue));
             auto value_style = TRY(style_computer.compute_style(progress, CSS::Selector::PseudoElement::ProgressValue));
             value_style->set_property(CSS::PropertyID::Display, CSS::DisplayStyleValue::create(CSS::Display::from_short(CSS::Display::Short::Block)));
             value_style->set_property(CSS::PropertyID::Display, CSS::DisplayStyleValue::create(CSS::Display::from_short(CSS::Display::Short::Block)));
-            auto position = progress.position();
-            value_style->set_property(CSS::PropertyID::Width, CSS::PercentageStyleValue::create(CSS::Percentage(position >= 0 ? round_to<int>(100 * position) : 0)));
+            value_style->set_property(CSS::PropertyID::Width, CSS::PercentageStyleValue::create(CSS::Percentage(progress.position() * 100)));
             auto bar_display = bar_style->display();
             auto bar_display = bar_style->display();
             auto value_display = value_style->display();
             auto value_display = value_style->display();
             auto progress_bar = DOM::Element::create_layout_node_for_display_type(document, bar_display, bar_style, nullptr);
             auto progress_bar = DOM::Element::create_layout_node_for_display_type(document, bar_display, bar_style, nullptr);