script/dom/html/
htmlfieldsetelement.rs1use std::default::Default;
6
7use dom_struct::dom_struct;
8use html5ever::{LocalName, Prefix, local_name};
9use js::context::JSContext;
10use js::rust::HandleObject;
11use stylo_dom::ElementState;
12
13use crate::dom::attr::Attr;
14use crate::dom::bindings::codegen::Bindings::HTMLFieldSetElementBinding::HTMLFieldSetElementMethods;
15use crate::dom::bindings::inheritance::{Castable, ElementTypeId, HTMLElementTypeId, NodeTypeId};
16use crate::dom::bindings::root::{DomRoot, MutNullableDom};
17use crate::dom::bindings::str::DOMString;
18use crate::dom::customelementregistry::CallbackReaction;
19use crate::dom::document::Document;
20use crate::dom::element::{AttributeMutation, Element};
21use crate::dom::html::htmlcollection::HTMLCollection;
22use crate::dom::html::htmlelement::HTMLElement;
23use crate::dom::html::htmlformelement::{FormControl, HTMLFormElement};
24use crate::dom::html::htmllegendelement::HTMLLegendElement;
25use crate::dom::node::{Node, NodeTraits, ShadowIncluding};
26use crate::dom::validation::Validatable;
27use crate::dom::validitystate::ValidityState;
28use crate::dom::virtualmethods::VirtualMethods;
29use crate::script_runtime::CanGc;
30use crate::script_thread::ScriptThread;
31
32#[dom_struct]
33pub(crate) struct HTMLFieldSetElement {
34 htmlelement: HTMLElement,
35 form_owner: MutNullableDom<HTMLFormElement>,
36 validity_state: MutNullableDom<ValidityState>,
37}
38
39impl HTMLFieldSetElement {
40 fn new_inherited(
41 local_name: LocalName,
42 prefix: Option<Prefix>,
43 document: &Document,
44 ) -> HTMLFieldSetElement {
45 HTMLFieldSetElement {
46 htmlelement: HTMLElement::new_inherited_with_state(
47 ElementState::ENABLED | ElementState::VALID,
48 local_name,
49 prefix,
50 document,
51 ),
52 form_owner: Default::default(),
53 validity_state: Default::default(),
54 }
55 }
56
57 pub(crate) fn new(
58 local_name: LocalName,
59 prefix: Option<Prefix>,
60 document: &Document,
61 proto: Option<HandleObject>,
62 can_gc: CanGc,
63 ) -> DomRoot<HTMLFieldSetElement> {
64 Node::reflect_node_with_proto(
65 Box::new(HTMLFieldSetElement::new_inherited(
66 local_name, prefix, document,
67 )),
68 document,
69 proto,
70 can_gc,
71 )
72 }
73
74 pub(crate) fn update_validity(&self, can_gc: CanGc) {
75 let has_invalid_child = self
76 .upcast::<Node>()
77 .traverse_preorder(ShadowIncluding::No)
78 .flat_map(DomRoot::downcast::<Element>)
79 .any(|element| element.is_invalid(false, can_gc));
80
81 self.upcast::<Element>()
82 .set_state(ElementState::VALID, !has_invalid_child);
83 self.upcast::<Element>()
84 .set_state(ElementState::INVALID, has_invalid_child);
85 }
86}
87
88impl HTMLFieldSetElementMethods<crate::DomTypeHolder> for HTMLFieldSetElement {
89 fn Elements(&self, can_gc: CanGc) -> DomRoot<HTMLCollection> {
91 HTMLCollection::new_with_filter_fn(
92 &self.owner_window(),
93 self.upcast(),
94 |element, _| {
95 element
96 .downcast::<HTMLElement>()
97 .is_some_and(HTMLElement::is_listed_element)
98 },
99 can_gc,
100 )
101 }
102
103 make_bool_getter!(Disabled, "disabled");
105
106 make_bool_setter!(SetDisabled, "disabled");
108
109 make_atomic_setter!(SetName, "name");
111
112 make_getter!(Name, "name");
114
115 fn GetForm(&self) -> Option<DomRoot<HTMLFormElement>> {
117 self.form_owner()
118 }
119
120 fn WillValidate(&self) -> bool {
122 self.is_instance_validatable()
123 }
124
125 fn Validity(&self, can_gc: CanGc) -> DomRoot<ValidityState> {
127 self.validity_state(can_gc)
128 }
129
130 fn CheckValidity(&self, cx: &mut JSContext) -> bool {
132 self.check_validity(cx)
133 }
134
135 fn ReportValidity(&self, cx: &mut JSContext) -> bool {
137 self.report_validity(cx)
138 }
139
140 fn ValidationMessage(&self) -> DOMString {
142 self.validation_message()
143 }
144
145 fn SetCustomValidity(&self, error: DOMString, can_gc: CanGc) {
147 self.validity_state(can_gc).set_custom_error_message(error);
148 }
149
150 fn Type(&self) -> DOMString {
152 "fieldset".into()
153 }
154}
155
156impl VirtualMethods for HTMLFieldSetElement {
157 fn super_type(&self) -> Option<&dyn VirtualMethods> {
158 Some(self.upcast::<HTMLElement>() as &dyn VirtualMethods)
159 }
160
161 fn attribute_mutated(&self, attr: &Attr, mutation: AttributeMutation, can_gc: CanGc) {
162 self.super_type()
163 .unwrap()
164 .attribute_mutated(attr, mutation, can_gc);
165 match *attr.local_name() {
166 local_name!("disabled") => {
167 let disabled_state = match mutation {
168 AttributeMutation::Set(None, _) => true,
169 AttributeMutation::Set(Some(_), _) => {
170 return;
172 },
173 AttributeMutation::Removed => false,
174 };
175 let node = self.upcast::<Node>();
176 let element = self.upcast::<Element>();
177 element.set_disabled_state(disabled_state);
178 element.set_enabled_state(!disabled_state);
179 let mut found_legend = false;
180 let children = node.children().filter(|node| {
181 if found_legend {
182 true
183 } else if node.is::<HTMLLegendElement>() {
184 found_legend = true;
185 false
186 } else {
187 true
188 }
189 });
190 let fields = children.flat_map(|child| {
191 child
192 .traverse_preorder(ShadowIncluding::No)
193 .filter(|descendant| match descendant.type_id() {
194 NodeTypeId::Element(ElementTypeId::HTMLElement(
195 HTMLElementTypeId::HTMLButtonElement |
196 HTMLElementTypeId::HTMLInputElement |
197 HTMLElementTypeId::HTMLSelectElement |
198 HTMLElementTypeId::HTMLTextAreaElement,
199 )) => true,
200 NodeTypeId::Element(ElementTypeId::HTMLElement(
201 HTMLElementTypeId::HTMLElement,
202 )) => descendant
203 .downcast::<HTMLElement>()
204 .unwrap()
205 .is_form_associated_custom_element(),
206 _ => false,
207 })
208 });
209 if disabled_state {
210 for field in fields {
211 let element = field.downcast::<Element>().unwrap();
212 if element.enabled_state() {
213 element.set_disabled_state(true);
214 element.set_enabled_state(false);
215 if element
216 .downcast::<HTMLElement>()
217 .is_some_and(|h| h.is_form_associated_custom_element())
218 {
219 ScriptThread::enqueue_callback_reaction(
220 element,
221 CallbackReaction::FormDisabled(true),
222 None,
223 );
224 }
225 }
226 }
227 } else {
228 for field in fields {
229 let element = field.downcast::<Element>().unwrap();
230 if element.disabled_state() {
231 element.check_disabled_attribute();
232 element.check_ancestors_disabled_state_for_form_control();
233 if element.enabled_state() &&
235 element
236 .downcast::<HTMLElement>()
237 .is_some_and(|h| h.is_form_associated_custom_element())
238 {
239 ScriptThread::enqueue_callback_reaction(
240 element,
241 CallbackReaction::FormDisabled(false),
242 None,
243 );
244 }
245 }
246 }
247 }
248 },
249 local_name!("form") => {
250 self.form_attribute_mutated(mutation, can_gc);
251 },
252 _ => {},
253 }
254 }
255}
256
257impl FormControl for HTMLFieldSetElement {
258 fn form_owner(&self) -> Option<DomRoot<HTMLFormElement>> {
259 self.form_owner.get()
260 }
261
262 fn set_form_owner(&self, form: Option<&HTMLFormElement>) {
263 self.form_owner.set(form);
264 }
265
266 fn to_element(&self) -> &Element {
267 self.upcast::<Element>()
268 }
269}
270
271impl Validatable for HTMLFieldSetElement {
272 fn as_element(&self) -> &Element {
273 self.upcast()
274 }
275
276 fn validity_state(&self, can_gc: CanGc) -> DomRoot<ValidityState> {
277 self.validity_state
278 .or_init(|| ValidityState::new(&self.owner_window(), self.upcast(), can_gc))
279 }
280
281 fn is_instance_validatable(&self) -> bool {
282 false
284 }
285}