forked from mirrors/gecko-dev
		
	
		
			
				
	
	
		
			451 lines
		
	
	
	
		
			15 KiB
		
	
	
	
		
			Rust
		
	
	
		
			Executable file
		
	
	
	
	
			
		
		
	
	
			451 lines
		
	
	
	
		
			15 KiB
		
	
	
	
		
			Rust
		
	
	
		
			Executable file
		
	
	
	
	
| /* This Source Code Form is subject to the terms of the Mozilla Public
 | |
|  * License, v. 2.0. If a copy of the MPL was not distributed with this
 | |
|  * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
 | |
| 
 | |
| use dom::attr::Attr;
 | |
| use dom::bindings::codegen::Bindings::ElementBinding::ElementMethods;
 | |
| use dom::bindings::codegen::Bindings::HTMLCollectionBinding::HTMLCollectionMethods;
 | |
| use dom::bindings::codegen::Bindings::HTMLOptionElementBinding::HTMLOptionElementMethods;
 | |
| use dom::bindings::codegen::Bindings::HTMLOptionsCollectionBinding::HTMLOptionsCollectionMethods;
 | |
| use dom::bindings::codegen::Bindings::HTMLSelectElementBinding;
 | |
| use dom::bindings::codegen::Bindings::HTMLSelectElementBinding::HTMLSelectElementMethods;
 | |
| use dom::bindings::codegen::Bindings::NodeBinding::NodeMethods;
 | |
| use dom::bindings::codegen::UnionTypes::HTMLElementOrLong;
 | |
| use dom::bindings::codegen::UnionTypes::HTMLOptionElementOrHTMLOptGroupElement;
 | |
| //use dom::bindings::error::ErrorResult;
 | |
| use dom::bindings::inheritance::Castable;
 | |
| use dom::bindings::root::{DomRoot, MutNullableDom};
 | |
| use dom::bindings::str::DOMString;
 | |
| use dom::document::Document;
 | |
| use dom::element::{AttributeMutation, Element};
 | |
| use dom::htmlcollection::CollectionFilter;
 | |
| use dom::htmlelement::HTMLElement;
 | |
| use dom::htmlfieldsetelement::HTMLFieldSetElement;
 | |
| use dom::htmlformelement::{FormDatumValue, FormControl, FormDatum, HTMLFormElement};
 | |
| use dom::htmloptgroupelement::HTMLOptGroupElement;
 | |
| use dom::htmloptionelement::HTMLOptionElement;
 | |
| use dom::htmloptionscollection::HTMLOptionsCollection;
 | |
| use dom::node::{Node, UnbindContext, window_from_node};
 | |
| use dom::nodelist::NodeList;
 | |
| use dom::validation::Validatable;
 | |
| use dom::validitystate::{ValidityState, ValidationFlags};
 | |
| use dom::virtualmethods::VirtualMethods;
 | |
| use dom_struct::dom_struct;
 | |
| use html5ever::{LocalName, Prefix};
 | |
| use std::default::Default;
 | |
| use std::iter;
 | |
| use style::attr::AttrValue;
 | |
| use style::element_state::*;
 | |
| 
 | |
| #[derive(JSTraceable, MallocSizeOf)]
 | |
| struct OptionsFilter;
 | |
| impl CollectionFilter for OptionsFilter {
 | |
|     fn filter<'a>(&self, elem: &'a Element, root: &'a Node) -> bool {
 | |
|         if !elem.is::<HTMLOptionElement>() {
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         let node = elem.upcast::<Node>();
 | |
|         if root.is_parent_of(node) {
 | |
|             return true;
 | |
|         }
 | |
| 
 | |
|         match node.GetParentNode() {
 | |
|             Some(optgroup) =>
 | |
|                 optgroup.is::<HTMLOptGroupElement>() && root.is_parent_of(&optgroup),
 | |
|             None => false,
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| #[dom_struct]
 | |
| pub struct HTMLSelectElement {
 | |
|     htmlelement: HTMLElement,
 | |
|     options: MutNullableDom<HTMLOptionsCollection>,
 | |
|     form_owner: MutNullableDom<HTMLFormElement>,
 | |
| }
 | |
| 
 | |
| static DEFAULT_SELECT_SIZE: u32 = 0;
 | |
| 
 | |
| impl HTMLSelectElement {
 | |
|     fn new_inherited(local_name: LocalName,
 | |
|                      prefix: Option<Prefix>,
 | |
|                      document: &Document) -> HTMLSelectElement {
 | |
|         HTMLSelectElement {
 | |
|             htmlelement:
 | |
|                 HTMLElement::new_inherited_with_state(IN_ENABLED_STATE,
 | |
|                                                       local_name, prefix, document),
 | |
|                 options: Default::default(),
 | |
|                 form_owner: Default::default(),
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     #[allow(unrooted_must_root)]
 | |
|     pub fn new(local_name: LocalName,
 | |
|                prefix: Option<Prefix>,
 | |
|                document: &Document) -> DomRoot<HTMLSelectElement> {
 | |
|         Node::reflect_node(Box::new(HTMLSelectElement::new_inherited(local_name, prefix, document)),
 | |
|                            document,
 | |
|                            HTMLSelectElementBinding::Wrap)
 | |
|     }
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/#concept-select-option-list
 | |
|     fn list_of_options(&self) -> impl Iterator<Item=DomRoot<HTMLOptionElement>> {
 | |
|         self.upcast::<Node>()
 | |
|             .children()
 | |
|             .flat_map(|node| {
 | |
|                 if node.is::<HTMLOptionElement>() {
 | |
|                     let node = DomRoot::downcast::<HTMLOptionElement>(node).unwrap();
 | |
|                     Choice3::First(iter::once(node))
 | |
|                 } else if node.is::<HTMLOptGroupElement>() {
 | |
|                     Choice3::Second(node.children().filter_map(DomRoot::downcast))
 | |
|                 } else {
 | |
|                     Choice3::Third(iter::empty())
 | |
|                 }
 | |
|             })
 | |
|     }
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/#the-select-element:concept-form-reset-control
 | |
|     pub fn reset(&self) {
 | |
|         for opt in self.list_of_options() {
 | |
|             opt.set_selectedness(opt.DefaultSelected());
 | |
|             opt.set_dirtiness(false);
 | |
|         }
 | |
|         self.ask_for_reset();
 | |
|     }
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/#ask-for-a-reset
 | |
|     pub fn ask_for_reset(&self) {
 | |
|         if self.Multiple() {
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         let mut first_enabled: Option<DomRoot<HTMLOptionElement>> = None;
 | |
|         let mut last_selected: Option<DomRoot<HTMLOptionElement>> = None;
 | |
| 
 | |
|         for opt in self.list_of_options() {
 | |
|             if opt.Selected() {
 | |
|                 opt.set_selectedness(false);
 | |
|                 last_selected = Some(DomRoot::from_ref(&opt));
 | |
|             }
 | |
|             let element = opt.upcast::<Element>();
 | |
|             if first_enabled.is_none() && !element.disabled_state() {
 | |
|                 first_enabled = Some(DomRoot::from_ref(&opt));
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         if let Some(last_selected) = last_selected {
 | |
|             last_selected.set_selectedness(true);
 | |
|         } else {
 | |
|             if self.display_size() == 1 {
 | |
|                 if let Some(first_enabled) = first_enabled {
 | |
|                     first_enabled.set_selectedness(true);
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     pub fn push_form_data(&self, data_set: &mut Vec<FormDatum>) {
 | |
|         if self.Name().is_empty() {
 | |
|             return;
 | |
|         }
 | |
|         for opt in self.list_of_options() {
 | |
|             let element = opt.upcast::<Element>();
 | |
|             if opt.Selected() && element.enabled_state() {
 | |
|                 data_set.push(FormDatum {
 | |
|                     ty: self.Type(),
 | |
|                     name: self.Name(),
 | |
|                     value:  FormDatumValue::String(opt.Value())
 | |
|                 });
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/#concept-select-pick
 | |
|     pub fn pick_option(&self, picked: &HTMLOptionElement) {
 | |
|         if !self.Multiple() {
 | |
|             let picked = picked.upcast();
 | |
|             for opt in self.list_of_options() {
 | |
|                 if opt.upcast::<HTMLElement>() != picked {
 | |
|                     opt.set_selectedness(false);
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/#concept-select-size
 | |
|     fn display_size(&self) -> u32 {
 | |
|          if self.Size() == 0 {
 | |
|              if self.Multiple() {
 | |
|                  4
 | |
|              } else {
 | |
|                  1
 | |
|              }
 | |
|          } else {
 | |
|              self.Size()
 | |
|          }
 | |
|      }
 | |
| }
 | |
| 
 | |
| impl HTMLSelectElementMethods for HTMLSelectElement {
 | |
|     // https://html.spec.whatwg.org/multipage/#dom-cva-validity
 | |
|     fn Validity(&self) -> DomRoot<ValidityState> {
 | |
|         let window = window_from_node(self);
 | |
|         ValidityState::new(&window, self.upcast())
 | |
|     }
 | |
| 
 | |
|     // Note: this function currently only exists for union.html.
 | |
|     // https://html.spec.whatwg.org/multipage/#dom-select-add
 | |
|     fn Add(&self, _element: HTMLOptionElementOrHTMLOptGroupElement, _before: Option<HTMLElementOrLong>) {
 | |
|     }
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/#dom-fe-disabled
 | |
|     make_bool_getter!(Disabled, "disabled");
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/#dom-fe-disabled
 | |
|     make_bool_setter!(SetDisabled, "disabled");
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/#dom-fae-form
 | |
|     fn GetForm(&self) -> Option<DomRoot<HTMLFormElement>> {
 | |
|         self.form_owner()
 | |
|     }
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/#dom-select-multiple
 | |
|     make_bool_getter!(Multiple, "multiple");
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/#dom-select-multiple
 | |
|     make_bool_setter!(SetMultiple, "multiple");
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/#dom-fe-name
 | |
|     make_getter!(Name, "name");
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/#dom-fe-name
 | |
|     make_setter!(SetName, "name");
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/#dom-select-size
 | |
|     make_uint_getter!(Size, "size", DEFAULT_SELECT_SIZE);
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/#dom-select-size
 | |
|     make_uint_setter!(SetSize, "size", DEFAULT_SELECT_SIZE);
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/#dom-select-type
 | |
|     fn Type(&self) -> DOMString {
 | |
|         DOMString::from(if self.Multiple() {
 | |
|             "select-multiple"
 | |
|         } else {
 | |
|             "select-one"
 | |
|         })
 | |
|     }
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/#dom-lfe-labels
 | |
|     fn Labels(&self) -> DomRoot<NodeList> {
 | |
|         self.upcast::<HTMLElement>().labels()
 | |
|     }
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/#dom-select-options
 | |
|     fn Options(&self) -> DomRoot<HTMLOptionsCollection> {
 | |
|         self.options.or_init(|| {
 | |
|             let window = window_from_node(self);
 | |
|             HTMLOptionsCollection::new(
 | |
|                 &window, self, Box::new(OptionsFilter))
 | |
|         })
 | |
|     }
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/#dom-select-length
 | |
|     fn Length(&self) -> u32 {
 | |
|         self.Options().Length()
 | |
|     }
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/#dom-select-length
 | |
|     fn SetLength(&self, length: u32) {
 | |
|         self.Options().SetLength(length)
 | |
|     }
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/#dom-select-item
 | |
|     fn Item(&self, index: u32) -> Option<DomRoot<Element>> {
 | |
|         self.Options().upcast().Item(index)
 | |
|     }
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/#dom-select-item
 | |
|     fn IndexedGetter(&self, index: u32) -> Option<DomRoot<Element>> {
 | |
|         self.Options().IndexedGetter(index)
 | |
|     }
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/#dom-select-nameditem
 | |
|     fn NamedItem(&self, name: DOMString) -> Option<DomRoot<HTMLOptionElement>> {
 | |
|         self.Options().NamedGetter(name).map_or(None, |e| DomRoot::downcast::<HTMLOptionElement>(e))
 | |
|     }
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/#dom-select-remove
 | |
|     fn Remove_(&self, index: i32) {
 | |
|         self.Options().Remove(index)
 | |
|     }
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/#dom-select-remove
 | |
|     fn Remove(&self) {
 | |
|         self.upcast::<Element>().Remove()
 | |
|     }
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/#dom-select-value
 | |
|     fn Value(&self) -> DOMString {
 | |
|         self.list_of_options()
 | |
|             .filter(|opt_elem| opt_elem.Selected())
 | |
|             .map(|opt_elem| opt_elem.Value())
 | |
|             .next()
 | |
|             .unwrap_or_default()
 | |
|     }
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/#dom-select-value
 | |
|     fn SetValue(&self, value: DOMString) {
 | |
|         let mut opt_iter = self.list_of_options();
 | |
|         // Reset until we find an <option> with a matching value
 | |
|         for opt in opt_iter.by_ref() {
 | |
|             if opt.Value() == value {
 | |
|                 opt.set_selectedness(true);
 | |
|                 opt.set_dirtiness(true);
 | |
|                 break;
 | |
|             }
 | |
|             opt.set_selectedness(false);
 | |
|         }
 | |
|         // Reset remaining <option> elements
 | |
|         for opt in opt_iter {
 | |
|             opt.set_selectedness(false);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/#dom-select-selectedindex
 | |
|     fn SelectedIndex(&self) -> i32 {
 | |
|         self.list_of_options()
 | |
|             .enumerate()
 | |
|             .filter(|&(_, ref opt_elem)| opt_elem.Selected())
 | |
|             .map(|(i, _)| i as i32)
 | |
|             .next()
 | |
|             .unwrap_or(-1)
 | |
|     }
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/#dom-select-selectedindex
 | |
|     fn SetSelectedIndex(&self, index: i32) {
 | |
|         let mut opt_iter = self.list_of_options();
 | |
|         for opt in opt_iter.by_ref().take(index as usize) {
 | |
|             opt.set_selectedness(false);
 | |
|         }
 | |
|         if let Some(opt) = opt_iter.next() {
 | |
|             opt.set_selectedness(true);
 | |
|             opt.set_dirtiness(true);
 | |
|             // Reset remaining <option> elements
 | |
|             for opt in opt_iter {
 | |
|                 opt.set_selectedness(false);
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| impl VirtualMethods for HTMLSelectElement {
 | |
|     fn super_type(&self) -> Option<&VirtualMethods> {
 | |
|         Some(self.upcast::<HTMLElement>() as &VirtualMethods)
 | |
|     }
 | |
| 
 | |
|     fn attribute_mutated(&self, attr: &Attr, mutation: AttributeMutation) {
 | |
|         self.super_type().unwrap().attribute_mutated(attr, mutation);
 | |
|         match attr.local_name() {
 | |
|             &local_name!("disabled") => {
 | |
|                 let el = self.upcast::<Element>();
 | |
|                 match mutation {
 | |
|                     AttributeMutation::Set(_) => {
 | |
|                         el.set_disabled_state(true);
 | |
|                         el.set_enabled_state(false);
 | |
|                     },
 | |
|                     AttributeMutation::Removed => {
 | |
|                         el.set_disabled_state(false);
 | |
|                         el.set_enabled_state(true);
 | |
|                         el.check_ancestors_disabled_state_for_form_control();
 | |
|                     }
 | |
|                 }
 | |
|             },
 | |
|             &local_name!("form") => {
 | |
|                 self.form_attribute_mutated(mutation);
 | |
|             },
 | |
|             _ => {},
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     fn bind_to_tree(&self, tree_in_doc: bool) {
 | |
|         if let Some(ref s) = self.super_type() {
 | |
|             s.bind_to_tree(tree_in_doc);
 | |
|         }
 | |
| 
 | |
|         self.upcast::<Element>().check_ancestors_disabled_state_for_form_control();
 | |
|     }
 | |
| 
 | |
|     fn unbind_from_tree(&self, context: &UnbindContext) {
 | |
|         self.super_type().unwrap().unbind_from_tree(context);
 | |
| 
 | |
|         let node = self.upcast::<Node>();
 | |
|         let el = self.upcast::<Element>();
 | |
|         if node.ancestors().any(|ancestor| ancestor.is::<HTMLFieldSetElement>()) {
 | |
|             el.check_ancestors_disabled_state_for_form_control();
 | |
|         } else {
 | |
|             el.check_disabled_attribute();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     fn parse_plain_attribute(&self, local_name: &LocalName, value: DOMString) -> AttrValue {
 | |
|         match *local_name {
 | |
|             local_name!("size") => AttrValue::from_u32(value.into(), DEFAULT_SELECT_SIZE),
 | |
|             _ => self.super_type().unwrap().parse_plain_attribute(local_name, value),
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| impl FormControl for HTMLSelectElement {
 | |
|     fn form_owner(&self) -> Option<DomRoot<HTMLFormElement>> {
 | |
|         self.form_owner.get()
 | |
|     }
 | |
| 
 | |
|     fn set_form_owner(&self, form: Option<&HTMLFormElement>) {
 | |
|         self.form_owner.set(form);
 | |
|     }
 | |
| 
 | |
|     fn to_element<'a>(&'a self) -> &'a Element {
 | |
|         self.upcast::<Element>()
 | |
|     }
 | |
| }
 | |
| 
 | |
| impl Validatable for HTMLSelectElement {
 | |
|     fn is_instance_validatable(&self) -> bool {
 | |
|         true
 | |
|     }
 | |
|     fn validate(&self, validate_flags: ValidationFlags) -> bool {
 | |
|         if validate_flags.is_empty() {}
 | |
|         // Need more flag check for different validation types later
 | |
|         true
 | |
|     }
 | |
| }
 | |
| 
 | |
| enum Choice3<I, J, K> {
 | |
|     First(I),
 | |
|     Second(J),
 | |
|     Third(K),
 | |
| }
 | |
| 
 | |
| impl<I, J, K, T> Iterator for Choice3<I, J, K>
 | |
|     where I: Iterator<Item=T>, J: Iterator<Item=T>, K: Iterator<Item=T>
 | |
| {
 | |
|     type Item = T;
 | |
| 
 | |
|     fn next(&mut self) -> Option<T> {
 | |
|         match *self {
 | |
|             Choice3::First(ref mut i) => i.next(),
 | |
|             Choice3::Second(ref mut j) => j.next(),
 | |
|             Choice3::Third(ref mut k) => k.next(),
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     fn size_hint(&self) -> (usize, Option<usize>) {
 | |
|         match *self {
 | |
|             Choice3::First(ref i) => i.size_hint(),
 | |
|             Choice3::Second(ref j) => j.size_hint(),
 | |
|             Choice3::Third(ref k) => k.size_hint(),
 | |
|         }
 | |
|     }
 | |
| }
 | 
