forked from mirrors/gecko-dev
		
	See #18809 Still haven't had time to test it but it should fix the tests failures that appeared in m-c Source-Repo: https://github.com/servo/servo Source-Revision: fe4139b779b3af749ec1426ddf4e1393c7b85442 --HG-- extra : subtree_source : https%3A//hg.mozilla.org/projects/converted-servo-linear extra : subtree_revision : 9a0549d7dd8a99315ba3f8eec5d0f9fdf9eb4c0f
		
			
				
	
	
		
			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::ElementState;
 | 
						|
 | 
						|
#[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(ElementState::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(),
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 |