2020-08-01 03:04:26 +01:00
/*
2021-04-28 22:46:44 +02:00
* Copyright ( c ) 2020 , the SerenityOS developers .
2020-08-01 03:04:26 +01:00
*
2021-04-22 01:24:48 -07:00
* SPDX - License - Identifier : BSD - 2 - Clause
2020-08-01 03:04:26 +01:00
*/
2026-04-18 10:54:06 +02:00
# include <LibWeb/Bindings/HTMLFieldSetElement.h>
2022-09-30 17:16:16 -06:00
# include <LibWeb/Bindings/Intrinsics.h>
2023-12-10 10:42:41 +01:00
# include <LibWeb/HTML/HTMLButtonElement.h>
2020-08-01 03:04:26 +01:00
# include <LibWeb/HTML/HTMLFieldSetElement.h>
2023-12-10 10:42:41 +01:00
# include <LibWeb/HTML/HTMLInputElement.h>
2022-09-30 16:21:34 +01:00
# include <LibWeb/HTML/HTMLLegendElement.h>
2023-12-10 10:42:41 +01:00
# include <LibWeb/HTML/HTMLObjectElement.h>
# include <LibWeb/HTML/HTMLOutputElement.h>
# include <LibWeb/HTML/HTMLSelectElement.h>
# include <LibWeb/HTML/HTMLTextAreaElement.h>
2024-11-05 08:07:13 +00:00
# include <LibWeb/Layout/FieldSetBox.h>
2020-08-01 03:04:26 +01:00
namespace Web : : HTML {
2024-11-15 04:01:23 +13:00
GC_DEFINE_ALLOCATOR ( HTMLFieldSetElement ) ;
2023-11-19 19:47:52 +01:00
2022-02-18 21:00:52 +01:00
HTMLFieldSetElement : : HTMLFieldSetElement ( DOM : : Document & document , DOM : : QualifiedName qualified_name )
2022-03-23 18:55:54 -04:00
: HTMLElement ( document , move ( qualified_name ) )
2020-08-01 03:04:26 +01:00
{
}
2022-03-14 13:21:51 -06:00
HTMLFieldSetElement : : ~ HTMLFieldSetElement ( ) = default ;
2022-09-30 16:21:34 +01:00
2023-08-07 08:41:28 +02:00
void HTMLFieldSetElement : : initialize ( JS : : Realm & realm )
2023-01-10 06:28:20 -05:00
{
2024-03-16 13:13:08 +01:00
WEB_SET_PROTOTYPE_FOR_INTERFACE ( HTMLFieldSetElement ) ;
2025-04-20 16:22:57 +02:00
Base : : initialize ( realm ) ;
2023-01-10 06:28:20 -05:00
}
2023-12-10 10:42:41 +01:00
void HTMLFieldSetElement : : visit_edges ( Cell : : Visitor & visitor )
{
Base : : visit_edges ( visitor ) ;
visitor . visit ( m_elements ) ;
}
2022-09-30 16:21:34 +01:00
// https://html.spec.whatwg.org/multipage/form-elements.html#concept-fieldset-disabled
bool HTMLFieldSetElement : : is_disabled ( ) const
{
// A fieldset element is a disabled fieldset if it matches any of the following conditions:
// - Its disabled attribute is specified
if ( has_attribute ( AttributeNames : : disabled ) )
return true ;
// - It is a descendant of another fieldset element whose disabled attribute is specified, and is not a descendant of that fieldset element's first legend element child, if any.
for ( auto * fieldset_ancestor = first_ancestor_of_type < HTMLFieldSetElement > ( ) ; fieldset_ancestor ; fieldset_ancestor = fieldset_ancestor - > first_ancestor_of_type < HTMLFieldSetElement > ( ) ) {
if ( fieldset_ancestor - > has_attribute ( HTML : : AttributeNames : : disabled ) ) {
auto * first_legend_element_child = fieldset_ancestor - > first_child_of_type < HTMLLegendElement > ( ) ;
if ( ! first_legend_element_child | | ! is_descendant_of ( * first_legend_element_child ) )
return true ;
}
}
return false ;
}
2026-02-15 19:21:54 +00:00
void HTMLFieldSetElement : : attribute_changed ( FlyString const & name , Optional < String > const & old_value , Optional < String > const & value , Optional < FlyString > const & namespace_ )
{
Base : : attribute_changed ( name , old_value , value , namespace_ ) ;
if ( name = = HTML : : AttributeNames : : disabled ) {
for_each_in_subtree_of_type < HTMLElement > ( [ ] ( auto & element ) {
if ( element . is_form_associated_custom_element ( ) )
element . update_face_disabled_state ( ) ;
return TraversalDecision : : Continue ;
} ) ;
}
}
2023-12-10 10:42:41 +01:00
// https://html.spec.whatwg.org/multipage/form-elements.html#dom-fieldset-elements
2024-11-15 04:01:23 +13:00
GC : : Ptr < DOM : : HTMLCollection > const & HTMLFieldSetElement : : elements ( )
2023-12-10 10:42:41 +01:00
{
// The elements IDL attribute must return an HTMLCollection rooted at the fieldset element, whose filter matches listed elements.
if ( ! m_elements ) {
m_elements = DOM : : HTMLCollection : : create ( * this , DOM : : HTMLCollection : : Scope : : Descendants , [ ] ( DOM : : Element const & element ) {
2026-02-15 19:55:29 +00:00
if ( auto const * form_associated_element = as_if < FormAssociatedElement > ( element ) ; form_associated_element & & form_associated_element - > is_listed ( ) )
return true ;
return false ;
2023-12-10 10:42:41 +01:00
} ) ;
}
return m_elements ;
}
2024-11-05 08:07:13 +00:00
Layout : : FieldSetBox * HTMLFieldSetElement : : layout_node ( )
{
return static_cast < Layout : : FieldSetBox * > ( Node : : layout_node ( ) ) ;
}
LibWeb: Make layout nodes refcounted
Move the layout tree from GC allocation to refcounted ownership so
removed layout and paint subtrees are destroyed synchronously instead
of waiting for the next GC sweep. This dramatically reduces GC memory
usage peaks after layout tree churn and makes it easier for memory use
to fall back after large document updates.
Update layout factories, tree traversal, SVG layout node creation,
paintable back-pointers, and pseudo-element layout links to use RefPtr
ownership.
Make display: contents follow the same shape as Blink and WebKit: the
element itself does not create a layout node, and its children are
flattened into the nearest layout parent. Wrap direct non-whitespace
text in an anonymous inline node when the boxless element contributes
inherited style to that text.
Use an internal inline wrapper for display: contents pseudo-elements
so generated content can still participate in layout, painting, hit
testing, and pseudo-element queries. Keep CSSOM reporting the computed
display value from the pseudo style, not the internal wrapper.
Remove the retained out-of-tree layout node list and its testing hook,
since the flattened model does not need a side owner for boxless
elements. Add coverage for inherited text style, dynamic insertion
order, pseudo-element hit testing, and computed style queries.
2026-06-07 17:50:33 +02:00
RefPtr < Layout : : Node > HTMLFieldSetElement : : create_layout_node ( CSS : : ComputedProperties const & style )
2024-11-05 08:07:13 +00:00
{
LibWeb: Make layout nodes refcounted
Move the layout tree from GC allocation to refcounted ownership so
removed layout and paint subtrees are destroyed synchronously instead
of waiting for the next GC sweep. This dramatically reduces GC memory
usage peaks after layout tree churn and makes it easier for memory use
to fall back after large document updates.
Update layout factories, tree traversal, SVG layout node creation,
paintable back-pointers, and pseudo-element layout links to use RefPtr
ownership.
Make display: contents follow the same shape as Blink and WebKit: the
element itself does not create a layout node, and its children are
flattened into the nearest layout parent. Wrap direct non-whitespace
text in an anonymous inline node when the boxless element contributes
inherited style to that text.
Use an internal inline wrapper for display: contents pseudo-elements
so generated content can still participate in layout, painting, hit
testing, and pseudo-element queries. Keep CSSOM reporting the computed
display value from the pseudo style, not the internal wrapper.
Remove the retained out-of-tree layout node list and its testing hook,
since the flattened model does not need a side owner for boxless
elements. Add coverage for inherited text style, dynamic insertion
order, pseudo-element hit testing, and computed style queries.
2026-06-07 17:50:33 +02:00
return make_ref_counted < Layout : : FieldSetBox > ( document ( ) , * this , style ) ;
2024-11-05 08:07:13 +00:00
}
2020-08-01 03:04:26 +01:00
}