1use std::cell::Cell;
6use std::default::Default;
7
8use dom_struct::dom_struct;
9use html5ever::{LocalName, Prefix, local_name};
10use js::context::JSContext;
11use js::rust::HandleObject;
12use script_bindings::codegen::GenericBindings::AttrBinding::AttrMethods;
13use script_bindings::codegen::GenericBindings::DocumentBinding::DocumentMethods;
14use script_bindings::codegen::GenericBindings::DocumentFragmentBinding::DocumentFragmentMethods;
15use script_bindings::codegen::GenericBindings::NodeBinding::NodeMethods;
16use servo_config::pref;
17use stylo_dom::ElementState;
18
19use crate::dom::activation::Activatable;
20use crate::dom::attr::Attr;
21use crate::dom::bindings::codegen::Bindings::HTMLButtonElementBinding::HTMLButtonElementMethods;
22use crate::dom::bindings::codegen::Bindings::NodeBinding::GetRootNodeOptions;
23use crate::dom::bindings::inheritance::Castable;
24use crate::dom::bindings::root::{DomRoot, MutNullableDom};
25use crate::dom::bindings::str::DOMString;
26use crate::dom::commandevent::CommandEvent;
27use crate::dom::document::Document;
28use crate::dom::documentfragment::DocumentFragment;
29use crate::dom::element::{AttributeMutation, Element};
30use crate::dom::event::{Event, EventBubbles, EventCancelable};
31use crate::dom::eventtarget::EventTarget;
32use crate::dom::html::htmlelement::HTMLElement;
33use crate::dom::html::htmlfieldsetelement::HTMLFieldSetElement;
34use crate::dom::html::htmlformelement::{
35 FormControl, FormDatum, FormDatumValue, FormSubmitterElement, HTMLFormElement, ResetFrom,
36 SubmittedFrom,
37};
38use crate::dom::node::{BindContext, Node, NodeTraits, UnbindContext};
39use crate::dom::nodelist::NodeList;
40use crate::dom::validation::{Validatable, is_barred_by_datalist_ancestor};
41use crate::dom::validitystate::{ValidationFlags, ValidityState};
42use crate::dom::virtualmethods::{VirtualMethods, vtable_for};
43use crate::script_runtime::CanGc;
44
45#[derive(Clone, Copy, JSTraceable, MallocSizeOf, PartialEq)]
46enum ButtonType {
47 Submit,
48 Reset,
49 Button,
50}
51
52#[dom_struct]
53pub(crate) struct HTMLButtonElement {
54 htmlelement: HTMLElement,
55 button_type: Cell<ButtonType>,
56 form_owner: MutNullableDom<HTMLFormElement>,
57 labels_node_list: MutNullableDom<NodeList>,
58 validity_state: MutNullableDom<ValidityState>,
59}
60
61impl HTMLButtonElement {
62 fn new_inherited(
63 local_name: LocalName,
64 prefix: Option<Prefix>,
65 document: &Document,
66 ) -> HTMLButtonElement {
67 HTMLButtonElement {
68 htmlelement: HTMLElement::new_inherited_with_state(
69 ElementState::ENABLED,
70 local_name,
71 prefix,
72 document,
73 ),
74 button_type: Cell::new(ButtonType::Submit),
75 form_owner: Default::default(),
76 labels_node_list: Default::default(),
77 validity_state: Default::default(),
78 }
79 }
80
81 pub(crate) fn new(
82 local_name: LocalName,
83 prefix: Option<Prefix>,
84 document: &Document,
85 proto: Option<HandleObject>,
86 can_gc: CanGc,
87 ) -> DomRoot<HTMLButtonElement> {
88 Node::reflect_node_with_proto(
89 Box::new(HTMLButtonElement::new_inherited(
90 local_name, prefix, document,
91 )),
92 document,
93 proto,
94 can_gc,
95 )
96 }
97
98 #[inline]
99 pub(crate) fn is_submit_button(&self) -> bool {
100 self.button_type.get() == ButtonType::Submit
101 }
102}
103
104impl HTMLButtonElementMethods<crate::DomTypeHolder> for HTMLButtonElement {
105 fn Command(&self) -> DOMString {
107 match self.command_state() {
109 CommandState::Custom => self
111 .upcast::<Element>()
112 .get_string_attribute(&local_name!("command")),
113 CommandState::Unknown => DOMString::default(),
115 CommandState::Close => DOMString::from("close"),
117 CommandState::ShowModal => DOMString::from("show-modal"),
118 }
119 }
120
121 make_setter!(SetCommand, "command");
123
124 make_bool_getter!(Disabled, "disabled");
126
127 make_bool_setter!(SetDisabled, "disabled");
129
130 fn GetForm(&self) -> Option<DomRoot<HTMLFormElement>> {
132 self.form_owner()
133 }
134
135 fn Type(&self) -> DOMString {
137 match self.button_type.get() {
138 ButtonType::Submit => DOMString::from("submit"),
139 ButtonType::Button => DOMString::from("button"),
140 ButtonType::Reset => DOMString::from("reset"),
141 }
142 }
143
144 make_setter!(SetType, "type");
146
147 make_form_action_getter!(FormAction, "formaction");
149
150 make_setter!(SetFormAction, "formaction");
152
153 make_enumerated_getter!(
155 FormEnctype,
156 "formenctype",
157 "application/x-www-form-urlencoded" | "multipart/form-data" | "text/plain",
158 invalid => "application/x-www-form-urlencoded"
159 );
160
161 make_setter!(SetFormEnctype, "formenctype");
163
164 make_enumerated_getter!(
166 FormMethod,
167 "formmethod",
168 "get" | "post" | "dialog",
169 invalid => "get"
170 );
171
172 make_setter!(SetFormMethod, "formmethod");
174
175 make_getter!(FormTarget, "formtarget");
177
178 make_setter!(SetFormTarget, "formtarget");
180
181 make_bool_getter!(FormNoValidate, "formnovalidate");
183
184 make_bool_setter!(SetFormNoValidate, "formnovalidate");
186
187 make_getter!(Name, "name");
189
190 make_atomic_setter!(SetName, "name");
192
193 make_getter!(Value, "value");
195
196 make_setter!(SetValue, "value");
198
199 make_labels_getter!(Labels, labels_node_list);
201
202 fn WillValidate(&self) -> bool {
204 self.is_instance_validatable()
205 }
206
207 fn Validity(&self, can_gc: CanGc) -> DomRoot<ValidityState> {
209 self.validity_state(can_gc)
210 }
211
212 fn CheckValidity(&self, cx: &mut JSContext) -> bool {
214 self.check_validity(cx)
215 }
216
217 fn ReportValidity(&self, cx: &mut JSContext) -> bool {
219 self.report_validity(cx)
220 }
221
222 fn ValidationMessage(&self) -> DOMString {
224 self.validation_message()
225 }
226
227 fn SetCustomValidity(&self, error: DOMString, can_gc: CanGc) {
229 self.validity_state(can_gc).set_custom_error_message(error);
230 }
231}
232
233impl HTMLButtonElement {
234 pub(crate) fn form_datum(&self, submitter: Option<FormSubmitterElement>) -> Option<FormDatum> {
237 if let Some(FormSubmitterElement::Button(submitter)) = submitter {
241 if submitter != self {
242 return None;
243 }
244 } else {
245 return None;
246 }
247 let ty = self.Type();
249 let name = self.Name();
251
252 if name.is_empty() {
253 return None;
255 }
256
257 Some(FormDatum {
259 ty,
260 name,
261 value: FormDatumValue::String(self.Value()),
262 })
263 }
264
265 fn set_type(&self, value: DOMString, can_gc: CanGc) {
266 let value = match value.to_ascii_lowercase().as_str() {
267 "reset" => ButtonType::Reset,
268 "button" => ButtonType::Button,
269 "submit" => ButtonType::Submit,
270 _ => {
271 if pref!(dom_command_invokers_enabled) {
272 let element = self.upcast::<Element>();
273 if element.has_attribute(&local_name!("command")) ||
274 element.has_attribute(&local_name!("commandfor"))
275 {
276 ButtonType::Button
277 } else {
278 ButtonType::Submit
279 }
280 } else {
281 ButtonType::Submit
282 }
283 },
284 };
285 self.button_type.set(value);
286 self.validity_state(can_gc)
287 .perform_validation_and_update(ValidationFlags::all(), can_gc);
288 }
289
290 fn command_for_element(&self) -> Option<DomRoot<Element>> {
291 let command_for_value = self
292 .upcast::<Element>()
293 .get_attribute(&local_name!("commandfor"))?
294 .Value();
295
296 let root_node = self
297 .upcast::<Node>()
298 .GetRootNode(&GetRootNodeOptions::empty());
299
300 if let Some(document) = root_node.downcast::<Document>() {
301 return document.GetElementById(command_for_value);
302 } else if let Some(document_fragment) = root_node.downcast::<DocumentFragment>() {
303 return document_fragment.GetElementById(command_for_value);
304 }
305 unreachable!("Button element must be in a document or document fragment");
306 }
307
308 fn command_state(&self) -> CommandState {
309 let command = self
310 .upcast::<Element>()
311 .get_string_attribute(&local_name!("command"));
312 if command.starts_with_str("--") {
313 return CommandState::Custom;
314 }
315 let value = command.to_ascii_lowercase();
316 if value == "close" {
317 return CommandState::Close;
318 }
319 if value == "show-modal" {
320 return CommandState::ShowModal;
321 }
322
323 CommandState::Unknown
324 }
325
326 fn determine_if_command_is_valid_for_target(
328 command: CommandState,
329 target: DomRoot<Element>,
330 ) -> bool {
331 if command == CommandState::Unknown {
333 return false;
334 }
335 if command == CommandState::Custom {
337 return true;
338 }
339 if !target.is_html_element() {
341 return false;
342 }
343 vtable_for(target.upcast::<Node>()).is_valid_command_steps(command)
351 }
352
353 pub(crate) fn optional_value(&self) -> Option<DOMString> {
355 self.upcast::<Element>()
358 .get_attribute(&local_name!("value"))
359 .map(|attribute| attribute.Value())
360 }
361}
362
363impl VirtualMethods for HTMLButtonElement {
364 fn super_type(&self) -> Option<&dyn VirtualMethods> {
365 Some(self.upcast::<HTMLElement>() as &dyn VirtualMethods)
366 }
367
368 fn attribute_mutated(&self, attr: &Attr, mutation: AttributeMutation, can_gc: CanGc) {
369 self.super_type()
370 .unwrap()
371 .attribute_mutated(attr, mutation, can_gc);
372 match *attr.local_name() {
373 local_name!("disabled") => {
374 let el = self.upcast::<Element>();
375 match mutation {
376 AttributeMutation::Set(Some(_), _) => {},
377 AttributeMutation::Set(None, _) => {
378 el.set_disabled_state(true);
379 el.set_enabled_state(false);
380 },
381 AttributeMutation::Removed => {
382 el.set_disabled_state(false);
383 el.set_enabled_state(true);
384 el.check_ancestors_disabled_state_for_form_control();
385 },
386 }
387 self.validity_state(can_gc)
388 .perform_validation_and_update(ValidationFlags::all(), can_gc);
389 },
390 local_name!("type") => self.set_type(attr.Value(), can_gc),
391 local_name!("command") => self.set_type(
392 self.upcast::<Element>()
393 .get_string_attribute(&local_name!("type")),
394 can_gc,
395 ),
396 local_name!("commandfor") => self.set_type(
397 self.upcast::<Element>()
398 .get_string_attribute(&local_name!("type")),
399 can_gc,
400 ),
401 local_name!("form") => {
402 self.form_attribute_mutated(mutation, can_gc);
403 self.validity_state(can_gc)
404 .perform_validation_and_update(ValidationFlags::empty(), can_gc);
405 },
406 _ => {},
407 }
408 }
409
410 fn bind_to_tree(&self, context: &BindContext, can_gc: CanGc) {
411 if let Some(s) = self.super_type() {
412 s.bind_to_tree(context, can_gc);
413 }
414
415 self.upcast::<Element>()
416 .check_ancestors_disabled_state_for_form_control();
417 }
418
419 fn unbind_from_tree(&self, context: &UnbindContext, can_gc: CanGc) {
420 self.super_type().unwrap().unbind_from_tree(context, can_gc);
421
422 let node = self.upcast::<Node>();
423 let el = self.upcast::<Element>();
424 if node
425 .ancestors()
426 .any(|ancestor| ancestor.is::<HTMLFieldSetElement>())
427 {
428 el.check_ancestors_disabled_state_for_form_control();
429 } else {
430 el.check_disabled_attribute();
431 }
432 }
433}
434
435impl FormControl for HTMLButtonElement {
436 fn form_owner(&self) -> Option<DomRoot<HTMLFormElement>> {
437 self.form_owner.get()
438 }
439
440 fn set_form_owner(&self, form: Option<&HTMLFormElement>) {
441 self.form_owner.set(form);
442 }
443
444 fn to_element(&self) -> &Element {
445 self.upcast::<Element>()
446 }
447}
448
449impl Validatable for HTMLButtonElement {
450 fn as_element(&self) -> &Element {
451 self.upcast()
452 }
453
454 fn validity_state(&self, can_gc: CanGc) -> DomRoot<ValidityState> {
455 self.validity_state
456 .or_init(|| ValidityState::new(&self.owner_window(), self.upcast(), can_gc))
457 }
458
459 fn is_instance_validatable(&self) -> bool {
460 self.button_type.get() == ButtonType::Submit &&
464 !self.upcast::<Element>().disabled_state() &&
465 !is_barred_by_datalist_ancestor(self.upcast())
466 }
467}
468
469impl Activatable for HTMLButtonElement {
470 fn as_element(&self) -> &Element {
471 self.upcast()
472 }
473
474 fn is_instance_activatable(&self) -> bool {
475 !self.upcast::<Element>().disabled_state()
477 }
478
479 fn activation_behavior(&self, _event: &Event, target: &EventTarget, can_gc: CanGc) {
481 if !target
483 .downcast::<Node>()
484 .is_none_or(|node| node.owner_document().is_fully_active())
485 {
486 return;
487 }
488
489 let button_type = self.button_type.get();
490 if let Some(owner) = self.form_owner() {
492 if button_type == ButtonType::Submit {
495 owner.submit(
496 SubmittedFrom::NotFromForm,
497 FormSubmitterElement::Button(self),
498 can_gc,
499 );
500 return;
501 }
502 if button_type == ButtonType::Reset {
505 owner.reset(ResetFrom::NotFromForm, can_gc);
506 return;
507 }
508 if button_type == ButtonType::Button &&
510 self.upcast::<Element>()
511 .get_string_attribute(&local_name!("type"))
512 .to_ascii_lowercase() !=
513 "button"
514 {
515 return;
516 }
517 }
518 if let Some(target) = self.command_for_element() {
522 let command = self.command_state();
524 if !Self::determine_if_command_is_valid_for_target(command, target.clone()) {
526 return;
527 }
528 let event = CommandEvent::new(
534 &self.owner_window(),
535 atom!("command"),
536 EventBubbles::DoesNotBubble,
537 EventCancelable::Cancelable,
538 Some(DomRoot::from_ref(self.upcast())),
539 self.upcast::<Element>()
540 .get_string_attribute(&local_name!("command")),
541 can_gc,
542 );
543 let event = event.upcast::<Event>();
544 if !event.fire(target.upcast::<EventTarget>(), can_gc) {
545 return;
546 }
547 let target_node = target.upcast::<Node>();
549 if !target_node.is_connected() {
550 return;
551 }
552 if command == CommandState::Custom {
554 return;
555 }
556 let _ = vtable_for(target_node).command_steps(DomRoot::from_ref(self), command, can_gc);
560 }
561 }
564}
565
566#[derive(Copy, Clone, Eq, PartialEq, Debug)]
567pub(crate) enum CommandState {
568 Unknown,
569 Custom,
570 ShowModal,
571 Close,
572}