mirror of
https://github.com/LadybirdBrowser/ladybird.git
synced 2025-07-30 04:39:06 +00:00
LibWeb: Implement PopoverInvokerElement
attribute change steps
PopoverInvokerElement's explicitly set attr-element should be set to null whenever the value of the `popovertarget` content attribute is changed.
This commit is contained in:
parent
40cb36607c
commit
022141647a
Notes:
github-actions[bot]
2024-12-14 20:12:21 +00:00
Author: https://github.com/tcl3
Commit: 022141647a
Pull-request: https://github.com/LadybirdBrowser/ladybird/pull/2891
7 changed files with 101 additions and 4 deletions
|
@ -441,6 +441,7 @@ set(SOURCES
|
||||||
HTML/Numbers.cpp
|
HTML/Numbers.cpp
|
||||||
HTML/PageTransitionEvent.cpp
|
HTML/PageTransitionEvent.cpp
|
||||||
HTML/PolicyContainers.cpp
|
HTML/PolicyContainers.cpp
|
||||||
|
HTML/PopoverInvokerElement.cpp
|
||||||
HTML/PopStateEvent.cpp
|
HTML/PopStateEvent.cpp
|
||||||
HTML/Parser/Entities.cpp
|
HTML/Parser/Entities.cpp
|
||||||
HTML/Parser/HTMLEncodingDetection.cpp
|
HTML/Parser/HTMLEncodingDetection.cpp
|
||||||
|
|
|
@ -45,6 +45,11 @@ WebIDL::ExceptionOr<void> HTMLButtonElement::set_type(String const& type)
|
||||||
return set_attribute(HTML::AttributeNames::type, type);
|
return set_attribute(HTML::AttributeNames::type, type);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void HTMLButtonElement::form_associated_element_attribute_changed(FlyString const& name, Optional<String> const& value, Optional<FlyString> const& namespace_)
|
||||||
|
{
|
||||||
|
PopoverInvokerElement::associated_attribute_changed(name, value, namespace_);
|
||||||
|
}
|
||||||
|
|
||||||
void HTMLButtonElement::visit_edges(Visitor& visitor)
|
void HTMLButtonElement::visit_edges(Visitor& visitor)
|
||||||
{
|
{
|
||||||
Base::visit_edges(visitor);
|
Base::visit_edges(visitor);
|
||||||
|
|
|
@ -40,6 +40,8 @@ public:
|
||||||
TypeAttributeState type_state() const;
|
TypeAttributeState type_state() const;
|
||||||
WebIDL::ExceptionOr<void> set_type(String const&);
|
WebIDL::ExceptionOr<void> set_type(String const&);
|
||||||
|
|
||||||
|
virtual void form_associated_element_attribute_changed(FlyString const& name, Optional<String> const& value, Optional<FlyString> const& namespace_) override;
|
||||||
|
|
||||||
// ^EventTarget
|
// ^EventTarget
|
||||||
// https://html.spec.whatwg.org/multipage/interaction.html#the-tabindex-attribute:the-button-element
|
// https://html.spec.whatwg.org/multipage/interaction.html#the-tabindex-attribute:the-button-element
|
||||||
// https://html.spec.whatwg.org/multipage/interaction.html#focusable-area
|
// https://html.spec.whatwg.org/multipage/interaction.html#focusable-area
|
||||||
|
|
32
Libraries/LibWeb/HTML/PopoverInvokerElement.cpp
Normal file
32
Libraries/LibWeb/HTML/PopoverInvokerElement.cpp
Normal file
|
@ -0,0 +1,32 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2024, Tim Ledbetter <tim.ledbetter@ladybird.org>
|
||||||
|
*
|
||||||
|
* SPDX-License-Identifier: BSD-2-Clause
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <LibWeb/DOM/Element.h>
|
||||||
|
#include <LibWeb/HTML/AttributeNames.h>
|
||||||
|
#include <LibWeb/HTML/PopoverInvokerElement.h>
|
||||||
|
|
||||||
|
namespace Web::HTML {
|
||||||
|
|
||||||
|
void PopoverInvokerElement::associated_attribute_changed(FlyString const& name, Optional<String> const&, Optional<FlyString> const& namespace_)
|
||||||
|
{
|
||||||
|
// From: https://html.spec.whatwg.org/multipage/common-dom-interfaces.html#reflecting-content-attributes-in-idl-attributess
|
||||||
|
// For element reflected targets only: the following attribute change steps, given element, localName, oldValue, value, and namespace,
|
||||||
|
// are used to synchronize between the content attribute and the IDL attribute:
|
||||||
|
|
||||||
|
// 1. If localName is not attr or namespace is not null, then return.
|
||||||
|
if (name != HTML::AttributeNames::popovertarget || namespace_.has_value())
|
||||||
|
return;
|
||||||
|
|
||||||
|
// 2. Set element's explicitly set attr-elements to null.
|
||||||
|
m_popover_target_element = nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
void PopoverInvokerElement::visit_edges(JS::Cell::Visitor& visitor)
|
||||||
|
{
|
||||||
|
visitor.visit(m_popover_target_element);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -22,10 +22,8 @@ public:
|
||||||
void set_popover_target_element(GC::Ptr<DOM::Element> value) { m_popover_target_element = value; }
|
void set_popover_target_element(GC::Ptr<DOM::Element> value) { m_popover_target_element = value; }
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
void visit_edges(JS::Cell::Visitor& visitor)
|
void visit_edges(JS::Cell::Visitor&);
|
||||||
{
|
void associated_attribute_changed(FlyString const& name, Optional<String> const& value, Optional<FlyString> const& namespace_);
|
||||||
visitor.visit(m_popover_target_element);
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
GC::Ptr<DOM::Element> m_popover_target_element;
|
GC::Ptr<DOM::Element> m_popover_target_element;
|
||||||
|
|
|
@ -0,0 +1,6 @@
|
||||||
|
Harness status: OK
|
||||||
|
|
||||||
|
Found 1 tests
|
||||||
|
|
||||||
|
1 Pass
|
||||||
|
Pass Element attribute reflection of popoverTargetElement/popovertarget should be kept in sync.
|
|
@ -0,0 +1,53 @@
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<link rel=author href="mailto:jarhar@chromium.org">
|
||||||
|
<link rel=help href="https://bugs.chromium.org/p/chromium/issues/detail?id=1523410">
|
||||||
|
<link rel=help href="https://bugzilla.mozilla.org/show_bug.cgi?id=1879001">
|
||||||
|
<link rel=help href="https://html.spec.whatwg.org/multipage/common-dom-interfaces.html#reflecting-content-attributes-in-idl-attributes:element">
|
||||||
|
<script src="../../../resources/testharness.js"></script>
|
||||||
|
<script src="../../../resources/testharnessreport.js"></script>
|
||||||
|
|
||||||
|
<button id=mybutton popovertarget="mypopover">toggle popover</button>
|
||||||
|
<div id=mypopover popover=auto>popover</div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
test(() => {
|
||||||
|
assert_equals(mybutton.popoverTargetElement.id, "mypopover",
|
||||||
|
'Setting element.popoverTargetElement to a valid element should work');
|
||||||
|
|
||||||
|
mybutton.popoverTargetElement = null;
|
||||||
|
assert_false(mybutton.hasAttribute('popovertarget'),
|
||||||
|
'Setting element.popoverTargetElement to null should unset popovertarget attribute.');
|
||||||
|
assert_equals(mybutton.popoverTargetElement, null,
|
||||||
|
'Setting element.popoverTargetElement to null should remove the existing element from element.popoverTargetElement.');
|
||||||
|
|
||||||
|
mybutton.popoverTargetElement = mypopover;
|
||||||
|
assert_true(mybutton.hasAttribute('popovertarget'),
|
||||||
|
'Assigning to element.popoverTargetElement should set the popovertarget attribute.');
|
||||||
|
|
||||||
|
mybutton.removeAttribute('popovertarget');
|
||||||
|
assert_equals(mybutton.popoverTargetElement, null,
|
||||||
|
'Removing the popovertarget attribute should remove the element from element.popoverTargetElement.');
|
||||||
|
|
||||||
|
mybutton.popoverTargetElement = mypopover;
|
||||||
|
assert_true(mybutton.hasAttribute('popovertarget'),
|
||||||
|
'Assigning to element.popoverTargetElement should set the popovertarget attribute.');
|
||||||
|
|
||||||
|
mybutton.setAttribute("popovertarget", 'invalid');
|
||||||
|
assert_equals(mybutton.popoverTargetElement, null,
|
||||||
|
'Setting the popovertarget attribute to a localName that is not attr should remove the existing element from element.popoverTargetElement.');
|
||||||
|
|
||||||
|
mybutton.popoverTargetElement = mypopover;
|
||||||
|
mybutton.setAttribute("popovertarget", "");
|
||||||
|
assert_equals(mybutton.popoverTargetElement, null,
|
||||||
|
'Setting the popovertarget attribute to empty string right after setting explicit element does remove the explicit element.');
|
||||||
|
|
||||||
|
mybutton.setAttribute("popovertarget", "mypopover");
|
||||||
|
assert_equals(mybutton.popoverTargetElement.id, "mypopover",
|
||||||
|
'Setting the popovertarget attribute to a value should set the popover target element.');
|
||||||
|
mybutton.setAttribute("popovertarget", "");
|
||||||
|
assert_equals(mybutton.getAttribute('popovertarget'), "",
|
||||||
|
'Assigning to element.popoverTargetElement to empty string should update the attribute value to empty string.');
|
||||||
|
assert_equals(mybutton.popoverTargetElement, null,
|
||||||
|
'Setting the popovertarget attribute to empty string should remove the existing element from element.popoverTargetElement.');
|
||||||
|
}, 'Element attribute reflection of popoverTargetElement/popovertarget should be kept in sync.');
|
||||||
|
</script>
|
Loading…
Add table
Add a link
Reference in a new issue