summaryrefslogtreecommitdiff
path: root/Userland/Libraries/LibWeb/HTML/HTMLInputElement.cpp
diff options
context:
space:
mode:
authorAndreas Kling <kling@serenityos.org>2021-02-10 18:26:07 +0100
committerAndreas Kling <kling@serenityos.org>2021-02-10 19:06:20 +0100
commit29a2aac89ab4eb9d369d4094d7a851ca5c2faee8 (patch)
tree7a0d00e77259ab9f5db9d61e60a8acae35c15e32 /Userland/Libraries/LibWeb/HTML/HTMLInputElement.cpp
parente4e325ff61ba4346eff92e9e55ba1a133f02aabe (diff)
downloadserenity-29a2aac89ab4eb9d369d4094d7a851ca5c2faee8.zip
LibWeb: Start implementing <input type=text> using a shadow DOM
Text <input> fields will now generate a basic shadow DOM and attach it to the input element. The shadow DOM contains a <div> with some inline style, and an always- editable text node inside it. Accessing the "value" attribute on such an input element will get/set the value from that text node. This is really cool, although not super stable since HTML editing is not super stable. But it's a start! :^)
Diffstat (limited to 'Userland/Libraries/LibWeb/HTML/HTMLInputElement.cpp')
-rw-r--r--Userland/Libraries/LibWeb/HTML/HTMLInputElement.cpp44
1 files changed, 42 insertions, 2 deletions
diff --git a/Userland/Libraries/LibWeb/HTML/HTMLInputElement.cpp b/Userland/Libraries/LibWeb/HTML/HTMLInputElement.cpp
index 27db0a94c9..c46e841723 100644
--- a/Userland/Libraries/LibWeb/HTML/HTMLInputElement.cpp
+++ b/Userland/Libraries/LibWeb/HTML/HTMLInputElement.cpp
@@ -27,10 +27,13 @@
#include <LibGfx/FontDatabase.h>
#include <LibWeb/DOM/Document.h>
#include <LibWeb/DOM/Event.h>
+#include <LibWeb/DOM/ShadowRoot.h>
+#include <LibWeb/DOM/Text.h>
#include <LibWeb/HTML/EventNames.h>
#include <LibWeb/HTML/HTMLFormElement.h>
#include <LibWeb/HTML/HTMLInputElement.h>
#include <LibWeb/InProcessWebView.h>
+#include <LibWeb/Layout/BlockBox.h>
#include <LibWeb/Layout/ButtonBox.h>
#include <LibWeb/Layout/CheckBox.h>
#include <LibWeb/Page/Frame.h>
@@ -74,8 +77,10 @@ RefPtr<Layout::Node> HTMLInputElement::create_layout_node()
if (type() == "checkbox")
return adopt(*new Layout::CheckBox(document(), *this, move(style)));
- // FIXME: Implement <input type=text> in terms of LibWeb primitives.
- return nullptr;
+ create_shadow_tree_if_needed();
+ auto layout_node = adopt(*new Layout::BlockBox(document(), this, move(style)));
+ layout_node->set_inline(true);
+ return layout_node;
}
void HTMLInputElement::set_checked(bool checked)
@@ -94,4 +99,39 @@ bool HTMLInputElement::enabled() const
return !has_attribute(HTML::AttributeNames::disabled);
}
+String HTMLInputElement::value() const
+{
+ if (m_text_node)
+ return m_text_node->data();
+ return default_value();
+}
+
+void HTMLInputElement::set_value(String value)
+{
+ if (m_text_node) {
+ m_text_node->set_data(move(value));
+ return;
+ }
+ set_attribute(HTML::AttributeNames::value, move(value));
+}
+
+void HTMLInputElement::create_shadow_tree_if_needed()
+{
+ if (shadow_root())
+ return;
+
+ // FIXME: This assumes that we want a text box. Is that always true?
+ auto shadow_root = adopt(*new DOM::ShadowRoot(document(), *this));
+ auto initial_value = attribute(HTML::AttributeNames::value);
+ if (initial_value.is_null())
+ initial_value = String::empty();
+ auto element = document().create_element(HTML::TagNames::div);
+ element->set_attribute(HTML::AttributeNames::style, "white-space: pre");
+ m_text_node = adopt(*new DOM::Text(document(), initial_value));
+ m_text_node->set_always_editable(true);
+ element->append_child(*m_text_node);
+ shadow_root->append_child(move(element));
+ set_shadow_root(move(shadow_root));
+}
+
}