mirror of
https://github.com/servo/servo.git
synced 2025-06-06 16:45:39 +00:00
http://www.robohornet.org gives a score of 101.36 on master, and 102.68 with this PR. The latter is slightly better, but probably within noise level. So it looks like this PR does not affect DOM performance. This is expected since `Box::new` is defined as: ```rust impl<T> Box<T> { #[inline(always)] pub fn new(x: T) -> Box<T> { box x } } ``` With inlining, it should compile to the same as box syntax.
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(HeapSizeOf, JSTraceable)]
|
|
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(),
|
|
}
|
|
}
|
|
}
|