| 
									
										
										
										
											2020-06-07 23:10:45 +02:00
										 |  |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2025-02-21 00:43:26 +01:00
										 |  |  |  |  * Copyright (c) 2020-2025, Andreas Kling <andreas@ladybird.org> | 
					
						
							| 
									
										
										
										
											2021-05-02 18:17:08 +01:00
										 |  |  |  |  * Copyright (c) 2021, Adam Hodgen <ant1441@gmail.com> | 
					
						
							| 
									
										
										
										
											2020-06-07 23:10:45 +02:00
										 |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2021-04-22 01:24:48 -07:00
										 |  |  |  |  * SPDX-License-Identifier: BSD-2-Clause | 
					
						
							| 
									
										
										
										
											2020-06-07 23:10:45 +02:00
										 |  |  |  |  */ | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-27 12:09:58 +12:00
										 |  |  |  | #include <LibWeb/Bindings/HTMLTableElementPrototype.h>
 | 
					
						
							| 
									
										
										
										
											2022-09-30 17:16:16 -06:00
										 |  |  |  | #include <LibWeb/Bindings/Intrinsics.h>
 | 
					
						
							| 
									
										
										
										
											2024-12-20 11:32:17 +01:00
										 |  |  |  | #include <LibWeb/CSS/ComputedProperties.h>
 | 
					
						
							| 
									
										
										
										
											2021-07-30 19:31:46 +01:00
										 |  |  |  | #include <LibWeb/CSS/Parser/Parser.h>
 | 
					
						
							| 
									
										
										
										
											2024-08-14 16:37:02 +01:00
										 |  |  |  | #include <LibWeb/CSS/StyleValues/CSSColorValue.h>
 | 
					
						
							| 
									
										
										
										
											2024-08-14 11:46:56 +01:00
										 |  |  |  | #include <LibWeb/CSS/StyleValues/CSSKeywordValue.h>
 | 
					
						
							| 
									
										
										
										
											2024-10-01 17:45:10 +01:00
										 |  |  |  | #include <LibWeb/CSS/StyleValues/ImageStyleValue.h>
 | 
					
						
							| 
									
										
										
										
											2023-08-21 23:44:07 +00:00
										 |  |  |  | #include <LibWeb/CSS/StyleValues/LengthStyleValue.h>
 | 
					
						
							| 
									
										
										
										
											2025-02-05 11:54:17 +00:00
										 |  |  |  | #include <LibWeb/DOM/Document.h>
 | 
					
						
							| 
									
										
										
										
											2021-04-26 16:36:40 +01:00
										 |  |  |  | #include <LibWeb/DOM/ElementFactory.h>
 | 
					
						
							|  |  |  |  | #include <LibWeb/DOM/HTMLCollection.h>
 | 
					
						
							| 
									
										
										
										
											2025-02-21 00:43:26 +01:00
										 |  |  |  | #include <LibWeb/HTML/HTMLTableCellElement.h>
 | 
					
						
							| 
									
										
										
										
											2021-05-02 18:03:47 +01:00
										 |  |  |  | #include <LibWeb/HTML/HTMLTableColElement.h>
 | 
					
						
							| 
									
										
										
										
											2020-07-26 15:08:16 +02:00
										 |  |  |  | #include <LibWeb/HTML/HTMLTableElement.h>
 | 
					
						
							| 
									
										
										
										
											2021-04-26 16:36:40 +01:00
										 |  |  |  | #include <LibWeb/HTML/HTMLTableRowElement.h>
 | 
					
						
							| 
									
										
										
										
											2023-11-19 22:19:32 +04:00
										 |  |  |  | #include <LibWeb/HTML/Numbers.h>
 | 
					
						
							| 
									
										
										
										
											2022-03-26 14:29:52 +01:00
										 |  |  |  | #include <LibWeb/HTML/Parser/HTMLParser.h>
 | 
					
						
							| 
									
										
										
										
											2021-04-26 16:36:40 +01:00
										 |  |  |  | #include <LibWeb/Namespace.h>
 | 
					
						
							| 
									
										
										
										
											2020-06-07 23:10:45 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-28 18:20:36 +02:00
										 |  |  |  | namespace Web::HTML { | 
					
						
							| 
									
										
										
										
											2020-06-07 23:10:45 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-15 04:01:23 +13:00
										 |  |  |  | GC_DEFINE_ALLOCATOR(HTMLTableElement); | 
					
						
							| 
									
										
										
										
											2023-11-19 19:47:52 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-18 21:00:52 +01:00
										 |  |  |  | HTMLTableElement::HTMLTableElement(DOM::Document& document, DOM::QualifiedName qualified_name) | 
					
						
							| 
									
										
										
										
											2021-02-07 11:20:15 +01:00
										 |  |  |  |     : HTMLElement(document, move(qualified_name)) | 
					
						
							| 
									
										
										
										
											2020-06-07 23:10:45 +02:00
										 |  |  |  | { | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-14 13:21:51 -06:00
										 |  |  |  | HTMLTableElement::~HTMLTableElement() = default; | 
					
						
							| 
									
										
										
										
											2020-06-07 23:10:45 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-07 08:41:28 +02:00
										 |  |  |  | void HTMLTableElement::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(HTMLTableElement); | 
					
						
							| 
									
										
										
										
											2025-04-20 16:22:57 +02:00
										 |  |  |  |     Base::initialize(realm); | 
					
						
							| 
									
										
										
										
											2023-01-10 06:28:20 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-24 19:04:47 +01:00
										 |  |  |  | void HTMLTableElement::visit_edges(Cell::Visitor& visitor) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     Base::visit_edges(visitor); | 
					
						
							|  |  |  |  |     visitor.visit(m_rows); | 
					
						
							|  |  |  |  |     visitor.visit(m_t_bodies); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-16 19:04:45 +01:00
										 |  |  |  | static unsigned parse_border(StringView value) | 
					
						
							| 
									
										
										
										
											2023-08-21 23:44:07 +00:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2023-12-23 15:59:14 +13:00
										 |  |  |  |     return value.to_number<unsigned>().value_or(0); | 
					
						
							| 
									
										
										
										
											2023-08-21 23:44:07 +00:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-12-23 17:51:10 +01:00
										 |  |  |  | bool HTMLTableElement::is_presentational_hint(FlyString const& name) const | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     if (Base::is_presentational_hint(name)) | 
					
						
							|  |  |  |  |         return true; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return first_is_one_of(name, | 
					
						
							|  |  |  |  |         HTML::AttributeNames::align, | 
					
						
							|  |  |  |  |         HTML::AttributeNames::background, | 
					
						
							|  |  |  |  |         HTML::AttributeNames::bgcolor, | 
					
						
							|  |  |  |  |         HTML::AttributeNames::border, | 
					
						
							| 
									
										
										
										
											2025-02-18 10:09:40 +00:00
										 |  |  |  |         HTML::AttributeNames::bordercolor, | 
					
						
							| 
									
										
										
										
											2024-12-23 17:51:10 +01:00
										 |  |  |  |         HTML::AttributeNames::cellpadding, | 
					
						
							|  |  |  |  |         HTML::AttributeNames::cellspacing, | 
					
						
							|  |  |  |  |         HTML::AttributeNames::height, | 
					
						
							|  |  |  |  |         HTML::AttributeNames::width); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												LibWeb: Split StyleComputer work into two phases with separate outputs
Before this change, StyleComputer would essentially take a DOM element,
find all the CSS rules that apply to it, and resolve the computed value
for each CSS property for that element.
This worked great, but it meant we had to do all the work of selector
matching and cascading every time.
To enable new optimizations, this change introduces a break in the
middle of this process where we've produced a "CascadedProperties".
This object contains the result of the cascade, before we've begun
turning cascaded values into computed values.
The cascaded properties are now stored with each element, which will
later allow us to do partial updates without re-running the full
StyleComputer machine. This will be particularly valuable for
re-implementing CSS inheritance, which is extremely heavy today.
Note that CSS animations and CSS transitions operate entirely on the
computed values, even though the cascade order would have you believe
they happen earlier. I'm not confident we have the right architecture
for this, but that's a separate issue.
											
										 
											2024-12-12 10:06:29 +01:00
										 |  |  |  | void HTMLTableElement::apply_presentational_hints(GC::Ref<CSS::CascadedProperties> cascaded_properties) const | 
					
						
							| 
									
										
										
										
											2020-06-12 22:52:38 +02:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     for_each_attribute([&](auto& name, auto& value) { | 
					
						
							|  |  |  |  |         if (name == HTML::AttributeNames::width) { | 
					
						
							| 
									
										
										
										
											2022-03-26 14:11:34 +01:00
										 |  |  |  |             if (auto parsed_value = parse_nonzero_dimension_value(value)) | 
					
						
							| 
									
										
											  
											
												LibWeb: Split StyleComputer work into two phases with separate outputs
Before this change, StyleComputer would essentially take a DOM element,
find all the CSS rules that apply to it, and resolve the computed value
for each CSS property for that element.
This worked great, but it meant we had to do all the work of selector
matching and cascading every time.
To enable new optimizations, this change introduces a break in the
middle of this process where we've produced a "CascadedProperties".
This object contains the result of the cascade, before we've begun
turning cascaded values into computed values.
The cascaded properties are now stored with each element, which will
later allow us to do partial updates without re-running the full
StyleComputer machine. This will be particularly valuable for
re-implementing CSS inheritance, which is extremely heavy today.
Note that CSS animations and CSS transitions operate entirely on the
computed values, even though the cascade order would have you believe
they happen earlier. I'm not confident we have the right architecture
for this, but that's a separate issue.
											
										 
											2024-12-12 10:06:29 +01:00
										 |  |  |  |                 cascaded_properties->set_property_from_presentational_hint(CSS::PropertyID::Width, parsed_value.release_nonnull()); | 
					
						
							| 
									
										
										
										
											2020-06-13 10:54:20 +02:00
										 |  |  |  |             return; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-08-26 21:16:29 +02:00
										 |  |  |  |         if (name == HTML::AttributeNames::height) { | 
					
						
							| 
									
										
										
										
											2024-10-01 17:45:10 +01:00
										 |  |  |  |             if (auto parsed_value = parse_dimension_value(value)) | 
					
						
							| 
									
										
											  
											
												LibWeb: Split StyleComputer work into two phases with separate outputs
Before this change, StyleComputer would essentially take a DOM element,
find all the CSS rules that apply to it, and resolve the computed value
for each CSS property for that element.
This worked great, but it meant we had to do all the work of selector
matching and cascading every time.
To enable new optimizations, this change introduces a break in the
middle of this process where we've produced a "CascadedProperties".
This object contains the result of the cascade, before we've begun
turning cascaded values into computed values.
The cascaded properties are now stored with each element, which will
later allow us to do partial updates without re-running the full
StyleComputer machine. This will be particularly valuable for
re-implementing CSS inheritance, which is extremely heavy today.
Note that CSS animations and CSS transitions operate entirely on the
computed values, even though the cascade order would have you believe
they happen earlier. I'm not confident we have the right architecture
for this, but that's a separate issue.
											
										 
											2024-12-12 10:06:29 +01:00
										 |  |  |  |                 cascaded_properties->set_property_from_presentational_hint(CSS::PropertyID::Height, parsed_value.release_nonnull()); | 
					
						
							| 
									
										
										
										
											2020-08-26 21:16:29 +02:00
										 |  |  |  |             return; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-11-19 12:22:49 +04:00
										 |  |  |  |         if (name == HTML::AttributeNames::align) { | 
					
						
							|  |  |  |  |             if (value.equals_ignoring_ascii_case("center"sv)) { | 
					
						
							| 
									
										
											  
											
												LibWeb: Split StyleComputer work into two phases with separate outputs
Before this change, StyleComputer would essentially take a DOM element,
find all the CSS rules that apply to it, and resolve the computed value
for each CSS property for that element.
This worked great, but it meant we had to do all the work of selector
matching and cascading every time.
To enable new optimizations, this change introduces a break in the
middle of this process where we've produced a "CascadedProperties".
This object contains the result of the cascade, before we've begun
turning cascaded values into computed values.
The cascaded properties are now stored with each element, which will
later allow us to do partial updates without re-running the full
StyleComputer machine. This will be particularly valuable for
re-implementing CSS inheritance, which is extremely heavy today.
Note that CSS animations and CSS transitions operate entirely on the
computed values, even though the cascade order would have you believe
they happen earlier. I'm not confident we have the right architecture
for this, but that's a separate issue.
											
										 
											2024-12-12 10:06:29 +01:00
										 |  |  |  |                 cascaded_properties->set_property_from_presentational_hint(CSS::PropertyID::MarginLeft, CSS::CSSKeywordValue::create(CSS::Keyword::Auto)); | 
					
						
							|  |  |  |  |                 cascaded_properties->set_property_from_presentational_hint(CSS::PropertyID::MarginRight, CSS::CSSKeywordValue::create(CSS::Keyword::Auto)); | 
					
						
							| 
									
										
										
										
											2025-02-05 12:08:27 +00:00
										 |  |  |  |             } else if (auto parsed_value = parse_css_value(CSS::Parser::ParsingParams { document() }, value, CSS::PropertyID::Float)) { | 
					
						
							| 
									
										
											  
											
												LibWeb: Split StyleComputer work into two phases with separate outputs
Before this change, StyleComputer would essentially take a DOM element,
find all the CSS rules that apply to it, and resolve the computed value
for each CSS property for that element.
This worked great, but it meant we had to do all the work of selector
matching and cascading every time.
To enable new optimizations, this change introduces a break in the
middle of this process where we've produced a "CascadedProperties".
This object contains the result of the cascade, before we've begun
turning cascaded values into computed values.
The cascaded properties are now stored with each element, which will
later allow us to do partial updates without re-running the full
StyleComputer machine. This will be particularly valuable for
re-implementing CSS inheritance, which is extremely heavy today.
Note that CSS animations and CSS transitions operate entirely on the
computed values, even though the cascade order would have you believe
they happen earlier. I'm not confident we have the right architecture
for this, but that's a separate issue.
											
										 
											2024-12-12 10:06:29 +01:00
										 |  |  |  |                 cascaded_properties->set_property_from_presentational_hint(CSS::PropertyID::Float, parsed_value.release_nonnull()); | 
					
						
							| 
									
										
										
										
											2023-11-19 12:22:49 +04:00
										 |  |  |  |             } | 
					
						
							|  |  |  |  |             return; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-10-01 17:45:10 +01:00
										 |  |  |  |         if (name == HTML::AttributeNames::background) { | 
					
						
							| 
									
										
										
										
											2024-12-06 16:24:08 -05:00
										 |  |  |  |             // https://html.spec.whatwg.org/multipage/rendering.html#tables-2:encoding-parsing-and-serializing-a-url
 | 
					
						
							| 
									
										
										
										
											2025-01-23 19:40:57 +13:00
										 |  |  |  |             if (auto parsed_value = document().encoding_parse_url(value); parsed_value.has_value()) | 
					
						
							|  |  |  |  |                 cascaded_properties->set_property_from_presentational_hint(CSS::PropertyID::BackgroundImage, CSS::ImageStyleValue::create(*parsed_value)); | 
					
						
							| 
									
										
										
										
											2024-10-01 17:45:10 +01:00
										 |  |  |  |             return; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-06-13 10:54:20 +02:00
										 |  |  |  |         if (name == HTML::AttributeNames::bgcolor) { | 
					
						
							| 
									
										
										
										
											2023-05-28 15:06:12 +12:00
										 |  |  |  |             // https://html.spec.whatwg.org/multipage/rendering.html#tables-2:rules-for-parsing-a-legacy-colour-value
 | 
					
						
							|  |  |  |  |             auto color = parse_legacy_color_value(value); | 
					
						
							| 
									
										
										
										
											2020-06-13 10:54:20 +02:00
										 |  |  |  |             if (color.has_value()) | 
					
						
							| 
									
										
										
										
											2025-02-19 21:02:12 +11:00
										 |  |  |  |                 cascaded_properties->set_property_from_presentational_hint(CSS::PropertyID::BackgroundColor, CSS::CSSColorValue::create_from_color(color.value(), CSS::ColorSyntax::Legacy)); | 
					
						
							| 
									
										
										
										
											2020-06-13 10:54:20 +02:00
										 |  |  |  |             return; | 
					
						
							| 
									
										
										
										
											2020-06-12 22:52:38 +02:00
										 |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-07-05 11:58:59 +02:00
										 |  |  |  |         if (name == HTML::AttributeNames::cellspacing) { | 
					
						
							|  |  |  |  |             if (auto parsed_value = parse_dimension_value(value)) | 
					
						
							| 
									
										
											  
											
												LibWeb: Split StyleComputer work into two phases with separate outputs
Before this change, StyleComputer would essentially take a DOM element,
find all the CSS rules that apply to it, and resolve the computed value
for each CSS property for that element.
This worked great, but it meant we had to do all the work of selector
matching and cascading every time.
To enable new optimizations, this change introduces a break in the
middle of this process where we've produced a "CascadedProperties".
This object contains the result of the cascade, before we've begun
turning cascaded values into computed values.
The cascaded properties are now stored with each element, which will
later allow us to do partial updates without re-running the full
StyleComputer machine. This will be particularly valuable for
re-implementing CSS inheritance, which is extremely heavy today.
Note that CSS animations and CSS transitions operate entirely on the
computed values, even though the cascade order would have you believe
they happen earlier. I'm not confident we have the right architecture
for this, but that's a separate issue.
											
										 
											2024-12-12 10:06:29 +01:00
										 |  |  |  |                 cascaded_properties->set_property_from_presentational_hint(CSS::PropertyID::BorderSpacing, parsed_value.release_nonnull()); | 
					
						
							| 
									
										
										
										
											2023-07-05 11:58:59 +02:00
										 |  |  |  |             return; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-08-21 23:44:07 +00:00
										 |  |  |  |         if (name == HTML::AttributeNames::border) { | 
					
						
							|  |  |  |  |             auto border = parse_border(value); | 
					
						
							|  |  |  |  |             if (!border) | 
					
						
							|  |  |  |  |                 return; | 
					
						
							| 
									
										
										
										
											2023-08-24 05:26:36 +00:00
										 |  |  |  |             auto apply_border_style = [&](CSS::PropertyID style_property, CSS::PropertyID width_property, CSS::PropertyID color_property) { | 
					
						
							| 
									
										
										
										
											2024-08-14 14:06:03 +01:00
										 |  |  |  |                 auto legacy_line_style = CSS::CSSKeywordValue::create(CSS::Keyword::Outset); | 
					
						
							| 
									
										
											  
											
												LibWeb: Split StyleComputer work into two phases with separate outputs
Before this change, StyleComputer would essentially take a DOM element,
find all the CSS rules that apply to it, and resolve the computed value
for each CSS property for that element.
This worked great, but it meant we had to do all the work of selector
matching and cascading every time.
To enable new optimizations, this change introduces a break in the
middle of this process where we've produced a "CascadedProperties".
This object contains the result of the cascade, before we've begun
turning cascaded values into computed values.
The cascaded properties are now stored with each element, which will
later allow us to do partial updates without re-running the full
StyleComputer machine. This will be particularly valuable for
re-implementing CSS inheritance, which is extremely heavy today.
Note that CSS animations and CSS transitions operate entirely on the
computed values, even though the cascade order would have you believe
they happen earlier. I'm not confident we have the right architecture
for this, but that's a separate issue.
											
										 
											2024-12-12 10:06:29 +01:00
										 |  |  |  |                 cascaded_properties->set_property_from_presentational_hint(style_property, legacy_line_style); | 
					
						
							|  |  |  |  |                 cascaded_properties->set_property_from_presentational_hint(width_property, CSS::LengthStyleValue::create(CSS::Length::make_px(border))); | 
					
						
							| 
									
										
										
										
											2025-02-19 21:02:12 +11:00
										 |  |  |  |                 cascaded_properties->set_property_from_presentational_hint(color_property, CSS::CSSColorValue::create_from_color(Color(128, 128, 128), CSS::ColorSyntax::Legacy)); | 
					
						
							| 
									
										
										
										
											2023-08-21 23:44:07 +00:00
										 |  |  |  |             }; | 
					
						
							| 
									
										
										
										
											2023-08-24 05:26:36 +00:00
										 |  |  |  |             apply_border_style(CSS::PropertyID::BorderLeftStyle, CSS::PropertyID::BorderLeftWidth, CSS::PropertyID::BorderLeftColor); | 
					
						
							|  |  |  |  |             apply_border_style(CSS::PropertyID::BorderTopStyle, CSS::PropertyID::BorderTopWidth, CSS::PropertyID::BorderTopColor); | 
					
						
							|  |  |  |  |             apply_border_style(CSS::PropertyID::BorderRightStyle, CSS::PropertyID::BorderRightWidth, CSS::PropertyID::BorderRightColor); | 
					
						
							|  |  |  |  |             apply_border_style(CSS::PropertyID::BorderBottomStyle, CSS::PropertyID::BorderBottomWidth, CSS::PropertyID::BorderBottomColor); | 
					
						
							| 
									
										
										
										
											2023-08-21 23:44:07 +00:00
										 |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2025-02-18 10:09:40 +00:00
										 |  |  |  |         if (name == HTML::AttributeNames::bordercolor) { | 
					
						
							|  |  |  |  |             // https://html.spec.whatwg.org/multipage/rendering.html#tables-2:attr-table-bordercolor
 | 
					
						
							|  |  |  |  |             // When a table element has a bordercolor attribute, its value is expected to be parsed using the rules for parsing a legacy color value,
 | 
					
						
							|  |  |  |  |             // and if that does not return failure, the user agent is expected to treat the attribute as a presentational hint setting the element's
 | 
					
						
							|  |  |  |  |             // 'border-top-color', 'border-right-color', 'border-bottom-color', and 'border-left-color' properties to the resulting color.
 | 
					
						
							|  |  |  |  |             if (auto parsed_color = parse_legacy_color_value(value); parsed_color.has_value()) { | 
					
						
							| 
									
										
										
										
											2025-02-19 21:02:12 +11:00
										 |  |  |  |                 auto color_value = CSS::CSSColorValue::create_from_color(parsed_color.value(), CSS::ColorSyntax::Legacy); | 
					
						
							| 
									
										
										
										
											2025-02-18 10:09:40 +00:00
										 |  |  |  |                 cascaded_properties->set_property_from_presentational_hint(CSS::PropertyID::BorderTopColor, color_value); | 
					
						
							|  |  |  |  |                 cascaded_properties->set_property_from_presentational_hint(CSS::PropertyID::BorderRightColor, color_value); | 
					
						
							|  |  |  |  |                 cascaded_properties->set_property_from_presentational_hint(CSS::PropertyID::BorderBottomColor, color_value); | 
					
						
							|  |  |  |  |                 cascaded_properties->set_property_from_presentational_hint(CSS::PropertyID::BorderLeftColor, color_value); | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-06-12 22:52:38 +02:00
										 |  |  |  |     }); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-14 08:14:16 -05:00
										 |  |  |  | void HTMLTableElement::attribute_changed(FlyString const& name, Optional<String> const& old_value, Optional<String> const& value, Optional<FlyString> const& namespace_) | 
					
						
							| 
									
										
										
										
											2023-11-19 22:19:32 +04:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2024-11-14 08:14:16 -05:00
										 |  |  |  |     Base::attribute_changed(name, old_value, value, namespace_); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-19 22:19:32 +04:00
										 |  |  |  |     if (name == HTML::AttributeNames::cellpadding) { | 
					
						
							| 
									
										
										
										
											2025-02-21 00:43:26 +01:00
										 |  |  |  |         auto old_cellpadding = m_cellpadding; | 
					
						
							| 
									
										
										
										
											2023-11-19 22:19:32 +04:00
										 |  |  |  |         if (value.has_value()) | 
					
						
							| 
									
										
										
										
											2025-02-21 00:43:26 +01:00
										 |  |  |  |             m_cellpadding = max(0, parse_integer(value.value()).value_or(0)); | 
					
						
							| 
									
										
										
										
											2023-11-19 22:19:32 +04:00
										 |  |  |  |         else | 
					
						
							| 
									
										
										
										
											2025-02-21 00:43:26 +01:00
										 |  |  |  |             m_cellpadding = 1; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         // NOTE: cellpadding is magical, it applies to the cells inside this table, not the table itself.
 | 
					
						
							|  |  |  |  |         //       When it changes, we need new style for the cells.
 | 
					
						
							|  |  |  |  |         if (old_cellpadding != m_cellpadding) { | 
					
						
							|  |  |  |  |             for_each_in_subtree_of_type<HTMLTableCellElement>([&](auto& cell) { | 
					
						
							|  |  |  |  |                 cell.set_needs_style_update(true); | 
					
						
							|  |  |  |  |                 return TraversalDecision::Continue; | 
					
						
							|  |  |  |  |             }); | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-11-19 22:19:32 +04:00
										 |  |  |  |         return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-15 19:48:22 +01:00
										 |  |  |  | // https://html.spec.whatwg.org/multipage/tables.html#dom-table-caption
 | 
					
						
							| 
									
										
										
										
											2024-11-15 04:01:23 +13:00
										 |  |  |  | GC::Ptr<HTMLTableCaptionElement> HTMLTableElement::caption() | 
					
						
							| 
									
										
										
										
											2021-05-02 17:43:42 +01:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2022-12-15 19:48:22 +01:00
										 |  |  |  |     // The caption IDL attribute must return, on getting, the first caption element child of the table element,
 | 
					
						
							|  |  |  |  |     // if any, or null otherwise.
 | 
					
						
							| 
									
										
										
										
											2021-05-02 17:43:42 +01:00
										 |  |  |  |     return first_child_of_type<HTMLTableCaptionElement>(); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-15 19:57:16 +01:00
										 |  |  |  | // https://html.spec.whatwg.org/multipage/tables.html#dom-table-caption
 | 
					
						
							| 
									
										
										
										
											2023-05-25 20:37:57 +01:00
										 |  |  |  | WebIDL::ExceptionOr<void> HTMLTableElement::set_caption(HTMLTableCaptionElement* caption) | 
					
						
							| 
									
										
										
										
											2021-05-02 17:43:42 +01:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2022-12-15 19:57:16 +01:00
										 |  |  |  |     // On setting, the first caption element child of the table element, if any, must be removed,
 | 
					
						
							|  |  |  |  |     // and the new value, if not null, must be inserted as the first node of the table element.
 | 
					
						
							| 
									
										
										
										
											2021-05-02 17:43:42 +01:00
										 |  |  |  |     delete_caption(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-15 19:57:16 +01:00
										 |  |  |  |     if (caption) | 
					
						
							| 
									
										
										
										
											2023-05-25 20:37:57 +01:00
										 |  |  |  |         TRY(pre_insert(*caption, first_child())); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return {}; | 
					
						
							| 
									
										
										
										
											2021-05-02 17:43:42 +01:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-15 20:11:01 +01:00
										 |  |  |  | // https://html.spec.whatwg.org/multipage/tables.html#dom-table-createcaption
 | 
					
						
							| 
									
										
										
										
											2024-11-15 04:01:23 +13:00
										 |  |  |  | GC::Ref<HTMLTableCaptionElement> HTMLTableElement::create_caption() | 
					
						
							| 
									
										
										
										
											2021-05-02 17:43:42 +01:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     auto maybe_caption = caption(); | 
					
						
							|  |  |  |  |     if (maybe_caption) { | 
					
						
							|  |  |  |  |         return *maybe_caption; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-04 18:42:04 +01:00
										 |  |  |  |     auto caption = DOM::create_element(document(), TagNames::caption, Namespace::HTML).release_value_but_fixme_should_propagate_errors(); | 
					
						
							| 
									
										
										
										
											2022-10-30 17:50:04 +00:00
										 |  |  |  |     MUST(pre_insert(caption, first_child())); | 
					
						
							| 
									
										
										
										
											2022-08-28 13:42:07 +02:00
										 |  |  |  |     return static_cast<HTMLTableCaptionElement&>(*caption); | 
					
						
							| 
									
										
										
										
											2021-05-02 17:43:42 +01:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-15 20:11:01 +01:00
										 |  |  |  | // https://html.spec.whatwg.org/multipage/tables.html#dom-table-deletecaption
 | 
					
						
							| 
									
										
										
										
											2021-05-02 17:43:42 +01:00
										 |  |  |  | void HTMLTableElement::delete_caption() | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     auto maybe_caption = caption(); | 
					
						
							|  |  |  |  |     if (maybe_caption) { | 
					
						
							|  |  |  |  |         maybe_caption->remove(false); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-15 20:04:44 +01:00
										 |  |  |  | // https://html.spec.whatwg.org/multipage/tables.html#dom-table-thead
 | 
					
						
							| 
									
										
										
										
											2024-11-15 04:01:23 +13:00
										 |  |  |  | GC::Ptr<HTMLTableSectionElement> HTMLTableElement::t_head() | 
					
						
							| 
									
										
										
										
											2021-05-02 18:03:47 +01:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2022-12-15 20:04:44 +01:00
										 |  |  |  |     // The tHead IDL attribute must return, on getting, the first thead element child of the table element,
 | 
					
						
							|  |  |  |  |     // if any, or null otherwise.
 | 
					
						
							| 
									
										
										
										
											2021-05-02 18:03:47 +01:00
										 |  |  |  |     for (auto* child = first_child(); child; child = child->next_sibling()) { | 
					
						
							|  |  |  |  |         if (is<HTMLTableSectionElement>(*child)) { | 
					
						
							| 
									
										
										
										
											2025-01-21 09:12:05 -05:00
										 |  |  |  |             auto table_section_element = &as<HTMLTableSectionElement>(*child); | 
					
						
							| 
									
										
										
										
											2021-07-10 20:47:32 +01:00
										 |  |  |  |             if (table_section_element->local_name() == TagNames::thead) | 
					
						
							| 
									
										
										
										
											2021-05-02 18:03:47 +01:00
										 |  |  |  |                 return table_section_element; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return nullptr; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-15 20:04:44 +01:00
										 |  |  |  | // https://html.spec.whatwg.org/multipage/tables.html#dom-table-thead
 | 
					
						
							| 
									
										
										
										
											2022-09-25 17:03:42 +01:00
										 |  |  |  | WebIDL::ExceptionOr<void> HTMLTableElement::set_t_head(HTMLTableSectionElement* thead) | 
					
						
							| 
									
										
										
										
											2021-05-02 18:03:47 +01:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2022-12-15 20:04:44 +01:00
										 |  |  |  |     // If the new value is neither null nor a thead element, then a "HierarchyRequestError" DOMException must be thrown instead.
 | 
					
						
							|  |  |  |  |     if (thead && thead->local_name() != TagNames::thead) | 
					
						
							| 
									
										
										
										
											2024-10-12 20:56:21 +02:00
										 |  |  |  |         return WebIDL::HierarchyRequestError::create(realm(), "Element is not thead"_string); | 
					
						
							| 
									
										
										
										
											2021-05-02 18:03:47 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-15 20:04:44 +01:00
										 |  |  |  |     // On setting, if the new value is null or a thead element, the first thead element child of the table element,
 | 
					
						
							|  |  |  |  |     // if any, must be removed,
 | 
					
						
							| 
									
										
										
										
											2021-05-02 18:03:47 +01:00
										 |  |  |  |     delete_t_head(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-15 20:04:44 +01:00
										 |  |  |  |     if (!thead) | 
					
						
							|  |  |  |  |         return {}; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // and the new value, if not null, must be inserted immediately before the first element in the table element
 | 
					
						
							|  |  |  |  |     // that is neither a caption element nor a colgroup element, if any,
 | 
					
						
							|  |  |  |  |     // or at the end of the table if there are no such elements.
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-02 18:03:47 +01:00
										 |  |  |  |     // We insert the new thead after any <caption> or <colgroup> elements
 | 
					
						
							| 
									
										
										
										
											2022-12-16 19:14:26 +01:00
										 |  |  |  |     DOM::Node* child_to_insert_before = nullptr; | 
					
						
							| 
									
										
										
										
											2021-05-02 18:03:47 +01:00
										 |  |  |  |     for (auto* child = first_child(); child; child = child->next_sibling()) { | 
					
						
							|  |  |  |  |         if (!is<HTMLElement>(*child)) | 
					
						
							|  |  |  |  |             continue; | 
					
						
							|  |  |  |  |         if (is<HTMLTableCaptionElement>(*child)) | 
					
						
							|  |  |  |  |             continue; | 
					
						
							|  |  |  |  |         if (is<HTMLTableColElement>(*child)) { | 
					
						
							| 
									
										
										
										
											2025-01-21 09:12:05 -05:00
										 |  |  |  |             auto table_col_element = &as<HTMLTableColElement>(*child); | 
					
						
							| 
									
										
										
										
											2021-07-10 20:47:32 +01:00
										 |  |  |  |             if (table_col_element->local_name() == TagNames::colgroup) | 
					
						
							| 
									
										
										
										
											2021-05-02 18:03:47 +01:00
										 |  |  |  |                 continue; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         // We have found an element which is not a <caption> or <colgroup>, we'll insert before this
 | 
					
						
							| 
									
										
										
										
											2022-12-16 19:14:26 +01:00
										 |  |  |  |         child_to_insert_before = child; | 
					
						
							| 
									
										
										
										
											2021-05-02 18:03:47 +01:00
										 |  |  |  |         break; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-16 19:14:26 +01:00
										 |  |  |  |     TRY(pre_insert(*thead, child_to_insert_before)); | 
					
						
							| 
									
										
										
										
											2021-05-02 18:03:47 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     return {}; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-15 20:11:01 +01:00
										 |  |  |  | // https://html.spec.whatwg.org/multipage/tables.html#dom-table-createthead
 | 
					
						
							| 
									
										
										
										
											2024-11-15 04:01:23 +13:00
										 |  |  |  | GC::Ref<HTMLTableSectionElement> HTMLTableElement::create_t_head() | 
					
						
							| 
									
										
										
										
											2021-05-02 18:03:47 +01:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     auto maybe_thead = t_head(); | 
					
						
							|  |  |  |  |     if (maybe_thead) | 
					
						
							|  |  |  |  |         return *maybe_thead; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-04 18:42:04 +01:00
										 |  |  |  |     auto thead = DOM::create_element(document(), TagNames::thead, Namespace::HTML).release_value_but_fixme_should_propagate_errors(); | 
					
						
							| 
									
										
										
										
											2021-05-02 18:03:47 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     // We insert the new thead after any <caption> or <colgroup> elements
 | 
					
						
							| 
									
										
										
										
											2022-12-16 19:14:26 +01:00
										 |  |  |  |     DOM::Node* child_to_insert_before = nullptr; | 
					
						
							| 
									
										
										
										
											2021-05-02 18:03:47 +01:00
										 |  |  |  |     for (auto* child = first_child(); child; child = child->next_sibling()) { | 
					
						
							|  |  |  |  |         if (!is<HTMLElement>(*child)) | 
					
						
							|  |  |  |  |             continue; | 
					
						
							|  |  |  |  |         if (is<HTMLTableCaptionElement>(*child)) | 
					
						
							|  |  |  |  |             continue; | 
					
						
							|  |  |  |  |         if (is<HTMLTableColElement>(*child)) { | 
					
						
							| 
									
										
										
										
											2025-01-21 09:12:05 -05:00
										 |  |  |  |             auto table_col_element = &as<HTMLTableColElement>(*child); | 
					
						
							| 
									
										
										
										
											2021-07-10 20:47:32 +01:00
										 |  |  |  |             if (table_col_element->local_name() == TagNames::colgroup) | 
					
						
							| 
									
										
										
										
											2021-05-02 18:03:47 +01:00
										 |  |  |  |                 continue; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         // We have found an element which is not a <caption> or <colgroup>, we'll insert before this
 | 
					
						
							| 
									
										
										
										
											2022-12-16 19:14:26 +01:00
										 |  |  |  |         child_to_insert_before = child; | 
					
						
							| 
									
										
										
										
											2021-05-02 18:03:47 +01:00
										 |  |  |  |         break; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-16 19:14:26 +01:00
										 |  |  |  |     MUST(pre_insert(thead, child_to_insert_before)); | 
					
						
							| 
									
										
										
										
											2021-05-02 18:03:47 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-28 13:42:07 +02:00
										 |  |  |  |     return static_cast<HTMLTableSectionElement&>(*thead); | 
					
						
							| 
									
										
										
										
											2021-05-02 18:03:47 +01:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-15 20:11:01 +01:00
										 |  |  |  | // https://html.spec.whatwg.org/multipage/tables.html#dom-table-deletethead
 | 
					
						
							| 
									
										
										
										
											2021-05-02 18:03:47 +01:00
										 |  |  |  | void HTMLTableElement::delete_t_head() | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     auto maybe_thead = t_head(); | 
					
						
							|  |  |  |  |     if (maybe_thead) { | 
					
						
							|  |  |  |  |         maybe_thead->remove(false); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-15 20:08:13 +01:00
										 |  |  |  | // https://html.spec.whatwg.org/multipage/tables.html#dom-table-tfoot
 | 
					
						
							| 
									
										
										
										
											2024-11-15 04:01:23 +13:00
										 |  |  |  | GC::Ptr<HTMLTableSectionElement> HTMLTableElement::t_foot() | 
					
						
							| 
									
										
										
										
											2021-05-02 18:07:45 +01:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2022-12-15 20:08:13 +01:00
										 |  |  |  |     // The tFoot IDL attribute must return, on getting, the first tfoot element child of the table element,
 | 
					
						
							|  |  |  |  |     // if any, or null otherwise.
 | 
					
						
							| 
									
										
										
										
											2021-05-02 18:07:45 +01:00
										 |  |  |  |     for (auto* child = first_child(); child; child = child->next_sibling()) { | 
					
						
							|  |  |  |  |         if (is<HTMLTableSectionElement>(*child)) { | 
					
						
							| 
									
										
										
										
											2025-01-21 09:12:05 -05:00
										 |  |  |  |             auto table_section_element = &as<HTMLTableSectionElement>(*child); | 
					
						
							| 
									
										
										
										
											2021-07-10 20:47:32 +01:00
										 |  |  |  |             if (table_section_element->local_name() == TagNames::tfoot) | 
					
						
							| 
									
										
										
										
											2021-05-02 18:07:45 +01:00
										 |  |  |  |                 return table_section_element; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return nullptr; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-15 20:08:13 +01:00
										 |  |  |  | // https://html.spec.whatwg.org/multipage/tables.html#dom-table-tfoot
 | 
					
						
							| 
									
										
										
										
											2022-09-25 17:03:42 +01:00
										 |  |  |  | WebIDL::ExceptionOr<void> HTMLTableElement::set_t_foot(HTMLTableSectionElement* tfoot) | 
					
						
							| 
									
										
										
										
											2021-05-02 18:07:45 +01:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2022-12-15 20:08:13 +01:00
										 |  |  |  |     // If the new value is neither null nor a tfoot element, then a "HierarchyRequestError" DOMException must be thrown instead.
 | 
					
						
							|  |  |  |  |     if (tfoot && tfoot->local_name() != TagNames::tfoot) | 
					
						
							| 
									
										
										
										
											2024-10-12 20:56:21 +02:00
										 |  |  |  |         return WebIDL::HierarchyRequestError::create(realm(), "Element is not tfoot"_string); | 
					
						
							| 
									
										
										
										
											2021-05-02 18:07:45 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-15 20:08:13 +01:00
										 |  |  |  |     // On setting, if the new value is null or a tfoot element, the first tfoot element child of the table element,
 | 
					
						
							|  |  |  |  |     // if any, must be removed,
 | 
					
						
							| 
									
										
										
										
											2021-05-02 18:07:45 +01:00
										 |  |  |  |     delete_t_foot(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-15 20:08:13 +01:00
										 |  |  |  |     // and the new value, if not null, must be inserted at the end of the table.
 | 
					
						
							|  |  |  |  |     if (tfoot) { | 
					
						
							|  |  |  |  |         TRY(append_child(*tfoot)); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-05-02 18:07:45 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     return {}; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-15 20:11:01 +01:00
										 |  |  |  | // https://html.spec.whatwg.org/multipage/tables.html#dom-table-createtfoot
 | 
					
						
							| 
									
										
										
										
											2024-11-15 04:01:23 +13:00
										 |  |  |  | GC::Ref<HTMLTableSectionElement> HTMLTableElement::create_t_foot() | 
					
						
							| 
									
										
										
										
											2021-05-02 18:07:45 +01:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     auto maybe_tfoot = t_foot(); | 
					
						
							|  |  |  |  |     if (maybe_tfoot) | 
					
						
							|  |  |  |  |         return *maybe_tfoot; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-04 18:42:04 +01:00
										 |  |  |  |     auto tfoot = DOM::create_element(document(), TagNames::tfoot, Namespace::HTML).release_value_but_fixme_should_propagate_errors(); | 
					
						
							| 
									
										
										
										
											2022-10-30 17:50:04 +00:00
										 |  |  |  |     MUST(append_child(tfoot)); | 
					
						
							| 
									
										
										
										
											2022-08-28 13:42:07 +02:00
										 |  |  |  |     return static_cast<HTMLTableSectionElement&>(*tfoot); | 
					
						
							| 
									
										
										
										
											2021-05-02 18:07:45 +01:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-15 20:11:01 +01:00
										 |  |  |  | // https://html.spec.whatwg.org/multipage/tables.html#dom-table-deletetfoot
 | 
					
						
							| 
									
										
										
										
											2021-05-02 18:07:45 +01:00
										 |  |  |  | void HTMLTableElement::delete_t_foot() | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     auto maybe_tfoot = t_foot(); | 
					
						
							|  |  |  |  |     if (maybe_tfoot) { | 
					
						
							|  |  |  |  |         maybe_tfoot->remove(false); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-24 19:04:47 +01:00
										 |  |  |  | // https://html.spec.whatwg.org/multipage/tables.html#dom-table-tbodies
 | 
					
						
							| 
									
										
										
										
											2024-11-15 04:01:23 +13:00
										 |  |  |  | GC::Ref<DOM::HTMLCollection> HTMLTableElement::t_bodies() | 
					
						
							| 
									
										
										
										
											2021-05-02 18:17:08 +01:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2022-11-24 19:04:47 +01:00
										 |  |  |  |     // The tBodies attribute must return an HTMLCollection rooted at the table node,
 | 
					
						
							|  |  |  |  |     // whose filter matches only tbody elements that are children of the table element.
 | 
					
						
							|  |  |  |  |     if (!m_t_bodies) { | 
					
						
							| 
									
										
										
										
											2023-05-23 11:25:07 +02:00
										 |  |  |  |         m_t_bodies = DOM::HTMLCollection::create(*this, DOM::HTMLCollection::Scope::Children, [](DOM::Element const& element) { | 
					
						
							| 
									
										
										
										
											2022-11-24 19:04:47 +01:00
										 |  |  |  |             return element.local_name() == TagNames::tbody; | 
					
						
							| 
									
										
										
										
											2023-08-13 13:05:26 +02:00
										 |  |  |  |         }); | 
					
						
							| 
									
										
										
										
											2022-11-24 19:04:47 +01:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |     return *m_t_bodies; | 
					
						
							| 
									
										
										
										
											2021-05-02 18:17:08 +01:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-15 20:11:01 +01:00
										 |  |  |  | // https://html.spec.whatwg.org/multipage/tables.html#dom-table-createtbody
 | 
					
						
							| 
									
										
										
										
											2024-11-15 04:01:23 +13:00
										 |  |  |  | GC::Ref<HTMLTableSectionElement> HTMLTableElement::create_t_body() | 
					
						
							| 
									
										
										
										
											2021-05-02 18:17:08 +01:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2023-11-04 18:42:04 +01:00
										 |  |  |  |     auto tbody = DOM::create_element(document(), TagNames::tbody, Namespace::HTML).release_value_but_fixme_should_propagate_errors(); | 
					
						
							| 
									
										
										
										
											2021-05-02 18:17:08 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     // We insert the new tbody after the last <tbody> element
 | 
					
						
							| 
									
										
										
										
											2022-12-16 19:14:26 +01:00
										 |  |  |  |     DOM::Node* child_to_insert_before = nullptr; | 
					
						
							| 
									
										
										
										
											2021-05-02 18:17:08 +01:00
										 |  |  |  |     for (auto* child = last_child(); child; child = child->previous_sibling()) { | 
					
						
							|  |  |  |  |         if (!is<HTMLElement>(*child)) | 
					
						
							|  |  |  |  |             continue; | 
					
						
							|  |  |  |  |         if (is<HTMLTableSectionElement>(*child)) { | 
					
						
							| 
									
										
										
										
											2025-01-21 09:12:05 -05:00
										 |  |  |  |             auto table_section_element = &as<HTMLTableSectionElement>(*child); | 
					
						
							| 
									
										
										
										
											2021-07-10 20:47:32 +01:00
										 |  |  |  |             if (table_section_element->local_name() == TagNames::tbody) { | 
					
						
							| 
									
										
										
										
											2021-05-02 18:17:08 +01:00
										 |  |  |  |                 // We have found an element which is a <tbody> we'll insert after this
 | 
					
						
							| 
									
										
										
										
											2022-12-16 19:14:26 +01:00
										 |  |  |  |                 child_to_insert_before = child->next_sibling(); | 
					
						
							| 
									
										
										
										
											2021-05-02 18:17:08 +01:00
										 |  |  |  |                 break; | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-16 19:14:26 +01:00
										 |  |  |  |     MUST(pre_insert(tbody, child_to_insert_before)); | 
					
						
							| 
									
										
										
										
											2021-05-02 18:17:08 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-28 13:42:07 +02:00
										 |  |  |  |     return static_cast<HTMLTableSectionElement&>(*tbody); | 
					
						
							| 
									
										
										
										
											2021-05-02 18:17:08 +01:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-24 19:04:47 +01:00
										 |  |  |  | // https://html.spec.whatwg.org/multipage/tables.html#dom-table-rows
 | 
					
						
							| 
									
										
										
										
											2024-11-15 04:01:23 +13:00
										 |  |  |  | GC::Ref<DOM::HTMLCollection> HTMLTableElement::rows() | 
					
						
							| 
									
										
										
										
											2021-04-26 16:36:40 +01:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     HTMLTableElement* table_node = this; | 
					
						
							|  |  |  |  |     // FIXME:  The elements in the collection must be ordered such that those elements whose parent is a thead are
 | 
					
						
							|  |  |  |  |     //         included first, in tree order, followed by those elements whose parent is either a table or tbody
 | 
					
						
							|  |  |  |  |     //         element, again in tree order, followed finally by those elements whose parent is a tfoot element,
 | 
					
						
							|  |  |  |  |     //         still in tree order.
 | 
					
						
							|  |  |  |  |     // How do you sort HTMLCollection?
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-24 19:04:47 +01:00
										 |  |  |  |     if (!m_rows) { | 
					
						
							| 
									
										
										
										
											2023-05-23 11:25:07 +02:00
										 |  |  |  |         m_rows = DOM::HTMLCollection::create(*this, DOM::HTMLCollection::Scope::Descendants, [table_node](DOM::Element const& element) { | 
					
						
							| 
									
										
										
										
											2022-11-24 19:04:47 +01:00
										 |  |  |  |             // Only match TR elements which are:
 | 
					
						
							|  |  |  |  |             // * children of the table element
 | 
					
						
							|  |  |  |  |             // * children of the thead, tbody, or tfoot elements that are themselves children of the table element
 | 
					
						
							|  |  |  |  |             if (!is<HTMLTableRowElement>(element)) { | 
					
						
							|  |  |  |  |                 return false; | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |             if (element.parent_element() == table_node) | 
					
						
							|  |  |  |  |                 return true; | 
					
						
							| 
									
										
										
										
											2021-04-26 16:36:40 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-24 19:04:47 +01:00
										 |  |  |  |             if (element.parent_element() && (element.parent_element()->local_name() == TagNames::thead || element.parent_element()->local_name() == TagNames::tbody || element.parent_element()->local_name() == TagNames::tfoot) | 
					
						
							|  |  |  |  |                 && element.parent()->parent() == table_node) { | 
					
						
							|  |  |  |  |                 return true; | 
					
						
							|  |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-04-26 16:36:40 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-24 19:04:47 +01:00
										 |  |  |  |             return false; | 
					
						
							| 
									
										
										
										
											2023-08-13 13:05:26 +02:00
										 |  |  |  |         }); | 
					
						
							| 
									
										
										
										
											2022-11-24 19:04:47 +01:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |     return *m_rows; | 
					
						
							| 
									
										
										
										
											2021-04-26 16:36:40 +01:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-15 20:11:01 +01:00
										 |  |  |  | // https://html.spec.whatwg.org/multipage/tables.html#dom-table-insertrow
 | 
					
						
							| 
									
										
										
										
											2024-11-15 04:01:23 +13:00
										 |  |  |  | WebIDL::ExceptionOr<GC::Ref<HTMLTableRowElement>> HTMLTableElement::insert_row(WebIDL::Long index) | 
					
						
							| 
									
										
										
										
											2021-04-26 16:36:40 +01:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     auto rows = this->rows(); | 
					
						
							|  |  |  |  |     auto rows_length = rows->length(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-12 23:48:05 +02:00
										 |  |  |  |     if (index < -1 || index > (long)rows_length) { | 
					
						
							| 
									
										
										
										
											2024-10-12 20:56:21 +02:00
										 |  |  |  |         return WebIDL::IndexSizeError::create(realm(), "Index is negative or greater than the number of rows"_string); | 
					
						
							| 
									
										
										
										
											2021-04-26 16:36:40 +01:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-11-04 18:42:04 +01:00
										 |  |  |  |     auto& tr = static_cast<HTMLTableRowElement&>(*TRY(DOM::create_element(document(), TagNames::tr, Namespace::HTML))); | 
					
						
							| 
									
										
										
										
											2021-04-26 16:36:40 +01:00
										 |  |  |  |     if (rows_length == 0 && !has_child_of_type<HTMLTableRowElement>()) { | 
					
						
							| 
									
										
										
										
											2023-11-04 18:42:04 +01:00
										 |  |  |  |         auto tbody = TRY(DOM::create_element(document(), TagNames::tbody, Namespace::HTML)); | 
					
						
							| 
									
										
										
										
											2022-10-30 17:50:04 +00:00
										 |  |  |  |         TRY(tbody->append_child(tr)); | 
					
						
							|  |  |  |  |         TRY(append_child(tbody)); | 
					
						
							| 
									
										
										
										
											2021-04-26 16:36:40 +01:00
										 |  |  |  |     } else if (rows_length == 0) { | 
					
						
							|  |  |  |  |         auto tbody = last_child_of_type<HTMLTableRowElement>(); | 
					
						
							| 
									
										
										
										
											2022-10-30 17:50:04 +00:00
										 |  |  |  |         TRY(tbody->append_child(tr)); | 
					
						
							| 
									
										
										
										
											2021-04-26 16:36:40 +01:00
										 |  |  |  |     } else if (index == -1 || index == (long)rows_length) { | 
					
						
							|  |  |  |  |         auto parent_of_last_tr = rows->item(rows_length - 1)->parent_element(); | 
					
						
							| 
									
										
										
										
											2022-10-30 17:50:04 +00:00
										 |  |  |  |         TRY(parent_of_last_tr->append_child(tr)); | 
					
						
							| 
									
										
										
										
											2021-04-26 16:36:40 +01:00
										 |  |  |  |     } else { | 
					
						
							|  |  |  |  |         rows->item(index)->parent_element()->insert_before(tr, rows->item(index)); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-11-15 04:01:23 +13:00
										 |  |  |  |     return GC::Ref(tr); | 
					
						
							| 
									
										
										
										
											2021-04-26 16:36:40 +01:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-15 09:08:55 +00:00
										 |  |  |  | // https://html.spec.whatwg.org/multipage/tables.html#dom-table-deleterow
 | 
					
						
							| 
									
										
										
										
											2024-02-26 18:54:36 +01:00
										 |  |  |  | WebIDL::ExceptionOr<void> HTMLTableElement::delete_row(WebIDL::Long index) | 
					
						
							| 
									
										
										
										
											2021-04-26 16:36:40 +01:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     auto rows = this->rows(); | 
					
						
							|  |  |  |  |     auto rows_length = rows->length(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-15 09:08:55 +00:00
										 |  |  |  |     // 1. If index is less than −1 or greater than or equal to the number of elements in the rows collection, then throw an "IndexSizeError" DOMException.
 | 
					
						
							|  |  |  |  |     if (index < -1 || index >= (long)rows_length) | 
					
						
							| 
									
										
										
										
											2024-10-12 20:56:21 +02:00
										 |  |  |  |         return WebIDL::IndexSizeError::create(realm(), "Index is negative or greater than or equal to the number of rows"_string); | 
					
						
							| 
									
										
										
										
											2021-12-15 09:08:55 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     // 2. If index is −1, then remove the last element in the rows collection from its parent, or do nothing if the rows collection is empty.
 | 
					
						
							|  |  |  |  |     if (index == -1) { | 
					
						
							|  |  |  |  |         if (rows_length == 0) | 
					
						
							|  |  |  |  |             return {}; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-26 16:36:40 +01:00
										 |  |  |  |         auto row_to_remove = rows->item(rows_length - 1); | 
					
						
							|  |  |  |  |         row_to_remove->remove(false); | 
					
						
							| 
									
										
										
										
											2021-12-15 09:08:55 +00:00
										 |  |  |  |         return {}; | 
					
						
							| 
									
										
										
										
											2021-04-26 16:36:40 +01:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-15 09:08:55 +00:00
										 |  |  |  |     // 3. Otherwise, remove the indexth element in the rows collection from its parent.
 | 
					
						
							|  |  |  |  |     auto row_to_remove = rows->item(index); | 
					
						
							|  |  |  |  |     row_to_remove->remove(false); | 
					
						
							| 
									
										
										
										
											2021-04-26 16:36:40 +01:00
										 |  |  |  |     return {}; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-21 23:44:07 +00:00
										 |  |  |  | unsigned int HTMLTableElement::border() const | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2024-01-16 19:04:45 +01:00
										 |  |  |  |     return parse_border(get_attribute_value(HTML::AttributeNames::border)); | 
					
						
							| 
									
										
										
										
											2023-08-21 23:44:07 +00:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-21 00:43:26 +01:00
										 |  |  |  | Optional<u32> HTMLTableElement::cellpadding() const | 
					
						
							| 
									
										
										
										
											2023-11-19 22:19:32 +04:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2025-02-21 00:43:26 +01:00
										 |  |  |  |     return m_cellpadding; | 
					
						
							| 
									
										
										
										
											2023-11-19 22:19:32 +04:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-07 23:10:45 +02:00
										 |  |  |  | } |