mirror of
https://github.com/servo/servo.git
synced 2025-06-24 00:54:32 +01:00
This patch puts in the initial framework for incremental reflow. Nodes' styles are no longer recalculated unless the node has changed. I've been hacking on the general problem of incremental reflow for the past couple weeks, and I've yet to get a full implementation that actually passes all the reftests + wikipedia + cnn. Therefore, I'm going to try to land the different parts of it one by one. This patch only does incremental style recalc, without incremental flow construction, inline-size bubbling, reflow, or display lists. Those will be coming in that order as I finish them. At least with this strategy, I can land a working version of incremental reflow, even if not yet complete. r? @pcwalton
1055 lines
37 KiB
Rust
1055 lines
37 KiB
Rust
/* 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/. */
|
|
|
|
//! Element nodes.
|
|
|
|
use dom::attr::{Attr, ReplacedAttr, FirstSetAttr, AttrHelpers, AttrHelpersForLayout};
|
|
use dom::attr::{AttrValue, StringAttrValue, UIntAttrValue, AtomAttrValue};
|
|
use dom::namednodemap::NamedNodeMap;
|
|
use dom::bindings::codegen::Bindings::AttrBinding::AttrMethods;
|
|
use dom::bindings::codegen::Bindings::ElementBinding;
|
|
use dom::bindings::codegen::Bindings::ElementBinding::ElementMethods;
|
|
use dom::bindings::codegen::Bindings::NamedNodeMapBinding::NamedNodeMapMethods;
|
|
use dom::bindings::codegen::InheritTypes::{ElementDerived, NodeCast};
|
|
use dom::bindings::js::{MutNullableJS, JS, JSRef, Temporary, TemporaryPushable};
|
|
use dom::bindings::js::{OptionalSettable, OptionalRootable, Root};
|
|
use dom::bindings::utils::{Reflectable, Reflector};
|
|
use dom::bindings::error::{ErrorResult, Fallible, NamespaceError, InvalidCharacter, Syntax};
|
|
use dom::bindings::utils::{QName, Name, InvalidXMLName, xml_name_type};
|
|
use dom::domrect::DOMRect;
|
|
use dom::domrectlist::DOMRectList;
|
|
use dom::document::{Document, DocumentHelpers};
|
|
use dom::domtokenlist::DOMTokenList;
|
|
use dom::eventtarget::{EventTarget, NodeTargetTypeId};
|
|
use dom::htmlcollection::HTMLCollection;
|
|
use dom::htmlserializer::serialize;
|
|
use dom::node::{ElementNodeTypeId, Node, NodeHelpers, NodeIterator, document_from_node};
|
|
use dom::node::{window_from_node, LayoutNodeHelpers};
|
|
use dom::nodelist::NodeList;
|
|
use dom::virtualmethods::{VirtualMethods, vtable_for};
|
|
use devtools_traits::AttrInfo;
|
|
use style::{matches, parse_selector_list_from_str};
|
|
use style;
|
|
use servo_util::namespace;
|
|
use servo_util::str::DOMString;
|
|
|
|
use std::ascii::StrAsciiExt;
|
|
use std::cell::RefCell;
|
|
use std::default::Default;
|
|
use std::mem;
|
|
use string_cache::{Atom, Namespace};
|
|
use url::UrlParser;
|
|
|
|
#[jstraceable]
|
|
#[must_root]
|
|
pub struct Element {
|
|
pub node: Node,
|
|
pub local_name: Atom,
|
|
pub namespace: Namespace,
|
|
pub prefix: Option<DOMString>,
|
|
pub attrs: RefCell<Vec<JS<Attr>>>,
|
|
pub style_attribute: RefCell<Option<style::PropertyDeclarationBlock>>,
|
|
pub attr_list: MutNullableJS<NamedNodeMap>,
|
|
class_list: MutNullableJS<DOMTokenList>,
|
|
}
|
|
|
|
impl ElementDerived for EventTarget {
|
|
fn is_element(&self) -> bool {
|
|
match self.type_id {
|
|
NodeTargetTypeId(ElementNodeTypeId(_)) => true,
|
|
_ => false
|
|
}
|
|
}
|
|
}
|
|
|
|
impl Reflectable for Element {
|
|
fn reflector<'a>(&'a self) -> &'a Reflector {
|
|
self.node.reflector()
|
|
}
|
|
}
|
|
|
|
#[deriving(PartialEq)]
|
|
#[jstraceable]
|
|
pub enum ElementTypeId {
|
|
HTMLElementTypeId,
|
|
HTMLAnchorElementTypeId,
|
|
HTMLAppletElementTypeId,
|
|
HTMLAreaElementTypeId,
|
|
HTMLAudioElementTypeId,
|
|
HTMLBaseElementTypeId,
|
|
HTMLBRElementTypeId,
|
|
HTMLBodyElementTypeId,
|
|
HTMLButtonElementTypeId,
|
|
HTMLCanvasElementTypeId,
|
|
HTMLDataElementTypeId,
|
|
HTMLDataListElementTypeId,
|
|
HTMLDirectoryElementTypeId,
|
|
HTMLDListElementTypeId,
|
|
HTMLDivElementTypeId,
|
|
HTMLEmbedElementTypeId,
|
|
HTMLFieldSetElementTypeId,
|
|
HTMLFontElementTypeId,
|
|
HTMLFormElementTypeId,
|
|
HTMLFrameElementTypeId,
|
|
HTMLFrameSetElementTypeId,
|
|
HTMLHRElementTypeId,
|
|
HTMLHeadElementTypeId,
|
|
HTMLHeadingElementTypeId,
|
|
HTMLHtmlElementTypeId,
|
|
HTMLIFrameElementTypeId,
|
|
HTMLImageElementTypeId,
|
|
HTMLInputElementTypeId,
|
|
HTMLLabelElementTypeId,
|
|
HTMLLegendElementTypeId,
|
|
HTMLLinkElementTypeId,
|
|
HTMLLIElementTypeId,
|
|
HTMLMapElementTypeId,
|
|
HTMLMediaElementTypeId,
|
|
HTMLMetaElementTypeId,
|
|
HTMLMeterElementTypeId,
|
|
HTMLModElementTypeId,
|
|
HTMLObjectElementTypeId,
|
|
HTMLOListElementTypeId,
|
|
HTMLOptGroupElementTypeId,
|
|
HTMLOptionElementTypeId,
|
|
HTMLOutputElementTypeId,
|
|
HTMLParagraphElementTypeId,
|
|
HTMLParamElementTypeId,
|
|
HTMLPreElementTypeId,
|
|
HTMLProgressElementTypeId,
|
|
HTMLQuoteElementTypeId,
|
|
HTMLScriptElementTypeId,
|
|
HTMLSelectElementTypeId,
|
|
HTMLSourceElementTypeId,
|
|
HTMLSpanElementTypeId,
|
|
HTMLStyleElementTypeId,
|
|
HTMLTableElementTypeId,
|
|
HTMLTableCaptionElementTypeId,
|
|
HTMLTableDataCellElementTypeId,
|
|
HTMLTableHeaderCellElementTypeId,
|
|
HTMLTableColElementTypeId,
|
|
HTMLTableRowElementTypeId,
|
|
HTMLTableSectionElementTypeId,
|
|
HTMLTemplateElementTypeId,
|
|
HTMLTextAreaElementTypeId,
|
|
HTMLTimeElementTypeId,
|
|
HTMLTitleElementTypeId,
|
|
HTMLTrackElementTypeId,
|
|
HTMLUListElementTypeId,
|
|
HTMLVideoElementTypeId,
|
|
HTMLUnknownElementTypeId,
|
|
|
|
ElementTypeId_,
|
|
}
|
|
|
|
//
|
|
// Element methods
|
|
//
|
|
|
|
impl Element {
|
|
pub fn new_inherited(type_id: ElementTypeId, local_name: DOMString, namespace: Namespace, prefix: Option<DOMString>, document: JSRef<Document>) -> Element {
|
|
Element {
|
|
node: Node::new_inherited(ElementNodeTypeId(type_id), document),
|
|
local_name: Atom::from_slice(local_name.as_slice()),
|
|
namespace: namespace,
|
|
prefix: prefix,
|
|
attrs: RefCell::new(vec!()),
|
|
attr_list: Default::default(),
|
|
class_list: Default::default(),
|
|
style_attribute: RefCell::new(None),
|
|
}
|
|
}
|
|
|
|
pub fn new(local_name: DOMString, namespace: Namespace, prefix: Option<DOMString>, document: JSRef<Document>) -> Temporary<Element> {
|
|
Node::reflect_node(box Element::new_inherited(ElementTypeId_, local_name, namespace, prefix, document),
|
|
document, ElementBinding::Wrap)
|
|
}
|
|
}
|
|
|
|
pub trait RawLayoutElementHelpers {
|
|
unsafe fn get_attr_val_for_layout<'a>(&'a self, namespace: &Namespace, name: &str) -> Option<&'a str>;
|
|
unsafe fn get_attr_vals_for_layout<'a>(&'a self, name: &str) -> Vec<&'a str>;
|
|
unsafe fn get_attr_atom_for_layout(&self, namespace: &Namespace, name: &str) -> Option<Atom>;
|
|
unsafe fn has_class_for_layout(&self, name: &str) -> bool;
|
|
}
|
|
|
|
impl RawLayoutElementHelpers for Element {
|
|
#[inline]
|
|
#[allow(unrooted_must_root)]
|
|
unsafe fn get_attr_val_for_layout<'a>(&'a self, namespace: &Namespace, name: &str)
|
|
-> Option<&'a str> {
|
|
// cast to point to T in RefCell<T> directly
|
|
let attrs: *const Vec<JS<Attr>> = mem::transmute(&self.attrs);
|
|
(*attrs).iter().find(|attr: & &JS<Attr>| {
|
|
let attr = attr.unsafe_get();
|
|
name == (*attr).local_name_atom_forever().as_slice() &&
|
|
(*attr).namespace == *namespace
|
|
}).map(|attr| {
|
|
let attr = attr.unsafe_get();
|
|
(*attr).value_ref_forever()
|
|
})
|
|
}
|
|
|
|
#[inline]
|
|
#[allow(unrooted_must_root)]
|
|
unsafe fn get_attr_vals_for_layout<'a>(&'a self, name: &str) -> Vec<&'a str> {
|
|
// cast to point to T in RefCell<T> directly
|
|
let attrs: *const Vec<JS<Attr>> = mem::transmute(&self.attrs);
|
|
(*attrs).iter().filter_map(|attr: &JS<Attr>| {
|
|
let attr = attr.unsafe_get();
|
|
if name == (*attr).local_name_atom_forever().as_slice() {
|
|
Some((*attr).value_ref_forever())
|
|
} else {
|
|
None
|
|
}
|
|
}).collect()
|
|
}
|
|
|
|
#[inline]
|
|
#[allow(unrooted_must_root)]
|
|
unsafe fn get_attr_atom_for_layout(&self, namespace: &Namespace, name: &str)
|
|
-> Option<Atom> {
|
|
// cast to point to T in RefCell<T> directly
|
|
let attrs: *const Vec<JS<Attr>> = mem::transmute(&self.attrs);
|
|
(*attrs).iter().find(|attr: & &JS<Attr>| {
|
|
let attr = attr.unsafe_get();
|
|
name == (*attr).local_name_atom_forever().as_slice() &&
|
|
(*attr).namespace == *namespace
|
|
}).and_then(|attr| {
|
|
let attr = attr.unsafe_get();
|
|
(*attr).value_atom_forever()
|
|
})
|
|
}
|
|
|
|
#[inline]
|
|
#[allow(unrooted_must_root)]
|
|
unsafe fn has_class_for_layout(&self, name: &str) -> bool {
|
|
let attrs: *const Vec<JS<Attr>> = mem::transmute(&self.attrs);
|
|
(*attrs).iter().find(|attr: & &JS<Attr>| {
|
|
let attr = attr.unsafe_get();
|
|
(*attr).local_name_atom_forever().as_slice() == "class"
|
|
}).map_or(false, |attr| {
|
|
let attr = attr.unsafe_get();
|
|
(*attr).value_tokens_forever().map(|mut tokens| { tokens.any(|atom| atom.as_slice() == name) })
|
|
}.take().unwrap())
|
|
}
|
|
}
|
|
|
|
pub trait LayoutElementHelpers {
|
|
unsafe fn html_element_in_html_document_for_layout(&self) -> bool;
|
|
}
|
|
|
|
impl LayoutElementHelpers for JS<Element> {
|
|
#[allow(unrooted_must_root)]
|
|
unsafe fn html_element_in_html_document_for_layout(&self) -> bool {
|
|
if (*self.unsafe_get()).namespace != ns!(HTML) {
|
|
return false
|
|
}
|
|
let node: JS<Node> = self.transmute_copy();
|
|
let owner_doc = node.owner_doc_for_layout().unsafe_get();
|
|
(*owner_doc).is_html_document
|
|
}
|
|
}
|
|
|
|
pub trait ElementHelpers<'a> {
|
|
fn html_element_in_html_document(self) -> bool;
|
|
fn get_local_name(self) -> &'a Atom;
|
|
fn get_namespace(self) -> &'a Namespace;
|
|
fn summarize(self) -> Vec<AttrInfo>;
|
|
fn is_void(self) -> bool;
|
|
}
|
|
|
|
impl<'a> ElementHelpers<'a> for JSRef<'a, Element> {
|
|
fn html_element_in_html_document(self) -> bool {
|
|
let node: JSRef<Node> = NodeCast::from_ref(self);
|
|
self.namespace == ns!(HTML) && node.is_in_html_doc()
|
|
}
|
|
|
|
fn get_local_name(self) -> &'a Atom {
|
|
&self.extended_deref().local_name
|
|
}
|
|
|
|
fn get_namespace(self) -> &'a Namespace {
|
|
&self.extended_deref().namespace
|
|
}
|
|
|
|
fn summarize(self) -> Vec<AttrInfo> {
|
|
let attrs = self.Attributes().root();
|
|
let mut i = 0;
|
|
let mut summarized = vec!();
|
|
while i < attrs.Length() {
|
|
let attr = attrs.Item(i).unwrap().root();
|
|
summarized.push(attr.summarize());
|
|
i += 1;
|
|
}
|
|
summarized
|
|
}
|
|
|
|
fn is_void(self) -> bool {
|
|
if self.namespace != ns!(HTML) {
|
|
return false
|
|
}
|
|
match self.local_name.as_slice() {
|
|
/* List of void elements from
|
|
http://www.whatwg.org/specs/web-apps/current-work/multipage/the-end.html#html-fragment-serialization-algorithm */
|
|
"area" | "base" | "basefont" | "bgsound" | "br" | "col" | "embed" |
|
|
"frame" | "hr" | "img" | "input" | "keygen" | "link" | "menuitem" |
|
|
"meta" | "param" | "source" | "track" | "wbr" => true,
|
|
_ => false
|
|
}
|
|
}
|
|
}
|
|
|
|
pub trait AttributeHandlers {
|
|
/// Returns the attribute with given namespace and case-sensitive local
|
|
/// name, if any.
|
|
fn get_attribute(self, namespace: Namespace, local_name: &str)
|
|
-> Option<Temporary<Attr>>;
|
|
fn get_attributes(self, local_name: &str)
|
|
-> Vec<Temporary<Attr>>;
|
|
fn set_attribute_from_parser(self, local_name: Atom,
|
|
value: DOMString, namespace: Namespace,
|
|
prefix: Option<DOMString>);
|
|
fn set_attribute(self, name: &str, value: AttrValue);
|
|
fn do_set_attribute(self, local_name: Atom, value: AttrValue,
|
|
name: Atom, namespace: Namespace,
|
|
prefix: Option<DOMString>, cb: |JSRef<Attr>| -> bool);
|
|
fn parse_attribute(self, namespace: &Namespace, local_name: &Atom,
|
|
value: DOMString) -> AttrValue;
|
|
|
|
fn remove_attribute(self, namespace: Namespace, name: &str);
|
|
fn notify_attribute_changed(self, local_name: &Atom);
|
|
fn has_class(&self, name: &str) -> bool;
|
|
fn notify_attribute_removed(self);
|
|
|
|
fn set_atomic_attribute(self, name: &str, value: DOMString);
|
|
|
|
// http://www.whatwg.org/html/#reflecting-content-attributes-in-idl-attributes
|
|
fn has_attribute(self, name: &str) -> bool;
|
|
fn set_bool_attribute(self, name: &str, value: bool);
|
|
fn get_url_attribute(self, name: &str) -> DOMString;
|
|
fn set_url_attribute(self, name: &str, value: DOMString);
|
|
fn get_string_attribute(self, name: &str) -> DOMString;
|
|
fn set_string_attribute(self, name: &str, value: DOMString);
|
|
fn set_tokenlist_attribute(self, name: &str, value: DOMString);
|
|
fn get_uint_attribute(self, name: &str) -> u32;
|
|
fn set_uint_attribute(self, name: &str, value: u32);
|
|
}
|
|
|
|
impl<'a> AttributeHandlers for JSRef<'a, Element> {
|
|
fn get_attribute(self, namespace: Namespace, local_name: &str) -> Option<Temporary<Attr>> {
|
|
self.get_attributes(local_name).iter().map(|attr| attr.root())
|
|
.find(|attr| attr.namespace == namespace)
|
|
.map(|x| Temporary::from_rooted(*x))
|
|
}
|
|
|
|
fn get_attributes(self, local_name: &str) -> Vec<Temporary<Attr>> {
|
|
let local_name = Atom::from_slice(local_name);
|
|
self.attrs.borrow().iter().map(|attr| attr.root()).filter_map(|attr| {
|
|
if *attr.local_name() == local_name {
|
|
Some(Temporary::from_rooted(*attr))
|
|
} else {
|
|
None
|
|
}
|
|
}).collect()
|
|
}
|
|
|
|
fn set_attribute_from_parser(self, local_name: Atom,
|
|
value: DOMString, namespace: Namespace,
|
|
prefix: Option<DOMString>) {
|
|
let name = match prefix {
|
|
None => local_name.clone(),
|
|
Some(ref prefix) => {
|
|
let name = format!("{:s}:{:s}", *prefix, local_name.as_slice());
|
|
Atom::from_slice(name.as_slice())
|
|
},
|
|
};
|
|
let value = self.parse_attribute(&namespace, &local_name, value);
|
|
self.do_set_attribute(local_name, value, name, namespace, prefix, |_| false)
|
|
}
|
|
|
|
fn set_attribute(self, name: &str, value: AttrValue) {
|
|
assert!(name == name.to_ascii_lower().as_slice());
|
|
assert!(!name.contains(":"));
|
|
|
|
let node: JSRef<Node> = NodeCast::from_ref(self);
|
|
node.wait_until_safe_to_modify_dom();
|
|
|
|
let name = Atom::from_slice(name);
|
|
self.do_set_attribute(name.clone(), value, name.clone(),
|
|
ns!(""), None, |attr| *attr.local_name() == name);
|
|
}
|
|
|
|
fn do_set_attribute(self, local_name: Atom, value: AttrValue,
|
|
name: Atom, namespace: Namespace,
|
|
prefix: Option<DOMString>, cb: |JSRef<Attr>| -> bool) {
|
|
let idx = self.attrs.borrow().iter()
|
|
.map(|attr| attr.root())
|
|
.position(|attr| cb(*attr));
|
|
let (idx, set_type) = match idx {
|
|
Some(idx) => (idx, ReplacedAttr),
|
|
None => {
|
|
let window = window_from_node(self).root();
|
|
let attr = Attr::new(*window, local_name, value.clone(),
|
|
name, namespace.clone(), prefix, self);
|
|
self.attrs.borrow_mut().push_unrooted(&attr);
|
|
(self.attrs.borrow().len() - 1, FirstSetAttr)
|
|
}
|
|
};
|
|
|
|
(*self.attrs.borrow())[idx].root().set_value(set_type, value);
|
|
}
|
|
|
|
fn parse_attribute(self, namespace: &Namespace, local_name: &Atom,
|
|
value: DOMString) -> AttrValue {
|
|
if *namespace == ns!("") {
|
|
vtable_for(&NodeCast::from_ref(self))
|
|
.parse_plain_attribute(local_name.as_slice(), value)
|
|
} else {
|
|
StringAttrValue(value)
|
|
}
|
|
}
|
|
|
|
fn remove_attribute(self, namespace: Namespace, name: &str) {
|
|
let (_, local_name) = get_attribute_parts(name);
|
|
let local_name = Atom::from_slice(local_name);
|
|
|
|
let idx = self.attrs.borrow().iter().map(|attr| attr.root()).position(|attr| {
|
|
*attr.local_name() == local_name
|
|
});
|
|
|
|
match idx {
|
|
None => (),
|
|
Some(idx) => {
|
|
{
|
|
let node: JSRef<Node> = NodeCast::from_ref(self);
|
|
node.wait_until_safe_to_modify_dom();
|
|
}
|
|
|
|
if namespace == ns!("") {
|
|
let removed_raw_value = (*self.attrs.borrow())[idx].root().Value();
|
|
vtable_for(&NodeCast::from_ref(self))
|
|
.before_remove_attr(&local_name,
|
|
removed_raw_value);
|
|
}
|
|
|
|
self.attrs.borrow_mut().remove(idx);
|
|
self.notify_attribute_removed();
|
|
}
|
|
};
|
|
}
|
|
|
|
fn notify_attribute_changed(self, _local_name: &Atom) {
|
|
let node: JSRef<Node> = NodeCast::from_ref(self);
|
|
if node.is_in_doc() {
|
|
let document = node.owner_doc().root();
|
|
document.content_changed(node);
|
|
}
|
|
}
|
|
|
|
fn notify_attribute_removed(self) {
|
|
let node: JSRef<Node> = NodeCast::from_ref(self);
|
|
if node.is_in_doc() {
|
|
let document = node.owner_doc().root();
|
|
document.content_changed(node);
|
|
}
|
|
}
|
|
|
|
fn has_class(&self, name: &str) -> bool {
|
|
self.get_attribute(ns!(""), "class").root().map(|attr| {
|
|
attr.value().tokens().map(|mut tokens| {
|
|
tokens.any(|atom| atom.as_slice() == name)
|
|
}).unwrap_or(false)
|
|
}).unwrap_or(false)
|
|
}
|
|
|
|
fn set_atomic_attribute(self, name: &str, value: DOMString) {
|
|
assert!(name == name.to_ascii_lower().as_slice());
|
|
let value = AttrValue::from_atomic(value);
|
|
self.set_attribute(name, value);
|
|
}
|
|
|
|
fn has_attribute(self, name: &str) -> bool {
|
|
let name = match self.html_element_in_html_document() {
|
|
true => Atom::from_slice(name.to_ascii_lower().as_slice()),
|
|
false => Atom::from_slice(name)
|
|
};
|
|
self.attrs.borrow().iter().map(|attr| attr.root()).any(|attr| {
|
|
*attr.local_name() == name && attr.namespace == ns!("")
|
|
})
|
|
}
|
|
|
|
fn set_bool_attribute(self, name: &str, value: bool) {
|
|
if self.has_attribute(name) == value { return; }
|
|
if value {
|
|
self.set_string_attribute(name, String::new());
|
|
} else {
|
|
self.remove_attribute(ns!(""), name);
|
|
}
|
|
}
|
|
|
|
fn get_url_attribute(self, name: &str) -> DOMString {
|
|
assert!(name == name.to_ascii_lower().as_slice());
|
|
if !self.has_attribute(name) {
|
|
return "".to_string();
|
|
}
|
|
let url = self.get_string_attribute(name);
|
|
let doc = document_from_node(self).root();
|
|
let base = doc.url();
|
|
// https://html.spec.whatwg.org/multipage/infrastructure.html#reflect
|
|
// XXXManishearth this doesn't handle `javascript:` urls properly
|
|
match UrlParser::new().base_url(base).parse(url.as_slice()) {
|
|
Ok(parsed) => parsed.serialize(),
|
|
Err(_) => "".to_string()
|
|
}
|
|
}
|
|
fn set_url_attribute(self, name: &str, value: DOMString) {
|
|
self.set_string_attribute(name, value);
|
|
}
|
|
|
|
fn get_string_attribute(self, name: &str) -> DOMString {
|
|
assert!(name == name.to_ascii_lower().as_slice());
|
|
match self.get_attribute(ns!(""), name) {
|
|
Some(x) => x.root().Value(),
|
|
None => "".to_string()
|
|
}
|
|
}
|
|
fn set_string_attribute(self, name: &str, value: DOMString) {
|
|
assert!(name == name.to_ascii_lower().as_slice());
|
|
self.set_attribute(name, StringAttrValue(value));
|
|
}
|
|
|
|
fn set_tokenlist_attribute(self, name: &str, value: DOMString) {
|
|
assert!(name == name.to_ascii_lower().as_slice());
|
|
self.set_attribute(name, AttrValue::from_tokenlist(value));
|
|
}
|
|
|
|
fn get_uint_attribute(self, name: &str) -> u32 {
|
|
assert!(name == name.to_ascii_lower().as_slice());
|
|
let attribute = self.get_attribute(ns!(""), name).root();
|
|
match attribute {
|
|
Some(attribute) => {
|
|
match *attribute.value() {
|
|
UIntAttrValue(_, value) => value,
|
|
_ => fail!("Expected a UIntAttrValue"),
|
|
}
|
|
}
|
|
None => 0,
|
|
}
|
|
}
|
|
fn set_uint_attribute(self, name: &str, value: u32) {
|
|
assert!(name == name.to_ascii_lower().as_slice());
|
|
self.set_attribute(name, UIntAttrValue(value.to_string(), value));
|
|
}
|
|
}
|
|
|
|
impl<'a> ElementMethods for JSRef<'a, Element> {
|
|
// http://dom.spec.whatwg.org/#dom-element-namespaceuri
|
|
fn GetNamespaceURI(self) -> Option<DOMString> {
|
|
match self.namespace {
|
|
ns!("") => None,
|
|
Namespace(ref ns) => Some(ns.as_slice().to_string())
|
|
}
|
|
}
|
|
|
|
fn LocalName(self) -> DOMString {
|
|
self.local_name.as_slice().to_string()
|
|
}
|
|
|
|
// http://dom.spec.whatwg.org/#dom-element-prefix
|
|
fn GetPrefix(self) -> Option<DOMString> {
|
|
self.prefix.clone()
|
|
}
|
|
|
|
// http://dom.spec.whatwg.org/#dom-element-tagname
|
|
fn TagName(self) -> DOMString {
|
|
let qualified_name = match self.prefix {
|
|
Some(ref prefix) => format!("{}:{}", prefix, self.local_name).into_maybe_owned(),
|
|
None => self.local_name.as_slice().into_maybe_owned()
|
|
};
|
|
if self.html_element_in_html_document() {
|
|
qualified_name.as_slice().to_ascii_upper()
|
|
} else {
|
|
qualified_name.into_string()
|
|
}
|
|
}
|
|
|
|
// http://dom.spec.whatwg.org/#dom-element-id
|
|
fn Id(self) -> DOMString {
|
|
self.get_string_attribute("id")
|
|
}
|
|
|
|
// http://dom.spec.whatwg.org/#dom-element-id
|
|
fn SetId(self, id: DOMString) {
|
|
self.set_atomic_attribute("id", id);
|
|
}
|
|
|
|
// http://dom.spec.whatwg.org/#dom-element-classname
|
|
fn ClassName(self) -> DOMString {
|
|
self.get_string_attribute("class")
|
|
}
|
|
|
|
// http://dom.spec.whatwg.org/#dom-element-classname
|
|
fn SetClassName(self, class: DOMString) {
|
|
self.set_tokenlist_attribute("class", class);
|
|
}
|
|
|
|
// http://dom.spec.whatwg.org/#dom-element-classlist
|
|
fn ClassList(self) -> Temporary<DOMTokenList> {
|
|
if self.class_list.get().is_none() {
|
|
let class_list = DOMTokenList::new(self, "class");
|
|
self.class_list.assign(Some(class_list));
|
|
}
|
|
self.class_list.get().unwrap()
|
|
}
|
|
|
|
// http://dom.spec.whatwg.org/#dom-element-attributes
|
|
fn Attributes(self) -> Temporary<NamedNodeMap> {
|
|
if self.attr_list.get().is_none() {
|
|
let doc = {
|
|
let node: JSRef<Node> = NodeCast::from_ref(self);
|
|
node.owner_doc().root()
|
|
};
|
|
let window = doc.window.root();
|
|
let list = NamedNodeMap::new(*window, self);
|
|
self.attr_list.assign(Some(list));
|
|
}
|
|
self.attr_list.get().unwrap()
|
|
}
|
|
|
|
// http://dom.spec.whatwg.org/#dom-element-getattribute
|
|
fn GetAttribute(self, name: DOMString) -> Option<DOMString> {
|
|
let name = if self.html_element_in_html_document() {
|
|
name.as_slice().to_ascii_lower()
|
|
} else {
|
|
name
|
|
};
|
|
self.get_attribute(ns!(""), name.as_slice()).root()
|
|
.map(|s| s.Value())
|
|
}
|
|
|
|
// http://dom.spec.whatwg.org/#dom-element-getattributens
|
|
fn GetAttributeNS(self,
|
|
namespace: Option<DOMString>,
|
|
local_name: DOMString) -> Option<DOMString> {
|
|
let namespace = namespace::from_domstring(namespace);
|
|
self.get_attribute(namespace, local_name.as_slice()).root()
|
|
.map(|attr| attr.Value())
|
|
}
|
|
|
|
// http://dom.spec.whatwg.org/#dom-element-setattribute
|
|
fn SetAttribute(self,
|
|
name: DOMString,
|
|
value: DOMString) -> ErrorResult {
|
|
{
|
|
let node: JSRef<Node> = NodeCast::from_ref(self);
|
|
node.wait_until_safe_to_modify_dom();
|
|
}
|
|
|
|
// Step 1.
|
|
match xml_name_type(name.as_slice()) {
|
|
InvalidXMLName => return Err(InvalidCharacter),
|
|
_ => {}
|
|
}
|
|
|
|
// Step 2.
|
|
let name = if self.html_element_in_html_document() {
|
|
name.as_slice().to_ascii_lower()
|
|
} else {
|
|
name
|
|
};
|
|
|
|
// Step 3-5.
|
|
let name = Atom::from_slice(name.as_slice());
|
|
let value = self.parse_attribute(&ns!(""), &name, value);
|
|
self.do_set_attribute(name.clone(), value, name.clone(), ns!(""), None, |attr| {
|
|
attr.name.as_slice() == name.as_slice()
|
|
});
|
|
Ok(())
|
|
}
|
|
|
|
// http://dom.spec.whatwg.org/#dom-element-setattributens
|
|
fn SetAttributeNS(self,
|
|
namespace_url: Option<DOMString>,
|
|
name: DOMString,
|
|
value: DOMString) -> ErrorResult {
|
|
{
|
|
let node: JSRef<Node> = NodeCast::from_ref(self);
|
|
node.wait_until_safe_to_modify_dom();
|
|
}
|
|
|
|
// Step 1.
|
|
let namespace = namespace::from_domstring(namespace_url);
|
|
|
|
let name_type = xml_name_type(name.as_slice());
|
|
match name_type {
|
|
// Step 2.
|
|
InvalidXMLName => return Err(InvalidCharacter),
|
|
// Step 3.
|
|
Name => return Err(NamespaceError),
|
|
QName => {}
|
|
}
|
|
|
|
// Step 4.
|
|
let (prefix, local_name) = get_attribute_parts(name.as_slice());
|
|
match prefix {
|
|
Some(ref prefix_str) => {
|
|
// Step 5.
|
|
if namespace == ns!("") {
|
|
return Err(NamespaceError);
|
|
}
|
|
|
|
// Step 6.
|
|
if "xml" == prefix_str.as_slice() && namespace != ns!(XML) {
|
|
return Err(NamespaceError);
|
|
}
|
|
|
|
// Step 7b.
|
|
if "xmlns" == prefix_str.as_slice() && namespace != ns!(XMLNS) {
|
|
return Err(NamespaceError);
|
|
}
|
|
},
|
|
None => {}
|
|
}
|
|
|
|
let name = Atom::from_slice(name.as_slice());
|
|
let local_name = Atom::from_slice(local_name);
|
|
let xmlns = Atom::from_slice("xmlns"); // TODO: Make this a static atom type
|
|
|
|
// Step 7a.
|
|
if xmlns == name && namespace != ns!(XMLNS) {
|
|
return Err(NamespaceError);
|
|
}
|
|
|
|
// Step 8.
|
|
if namespace == ns!(XMLNS) && xmlns != name && Some("xmlns") != prefix {
|
|
return Err(NamespaceError);
|
|
}
|
|
|
|
// Step 9.
|
|
let value = self.parse_attribute(&namespace, &local_name, value);
|
|
self.do_set_attribute(local_name.clone(), value, name,
|
|
namespace.clone(), prefix.map(|s| s.to_string()),
|
|
|attr| {
|
|
*attr.local_name() == local_name &&
|
|
attr.namespace == namespace
|
|
});
|
|
Ok(())
|
|
}
|
|
|
|
// http://dom.spec.whatwg.org/#dom-element-removeattribute
|
|
fn RemoveAttribute(self, name: DOMString) {
|
|
let name = if self.html_element_in_html_document() {
|
|
name.as_slice().to_ascii_lower()
|
|
} else {
|
|
name
|
|
};
|
|
self.remove_attribute(ns!(""), name.as_slice())
|
|
}
|
|
|
|
// http://dom.spec.whatwg.org/#dom-element-removeattributens
|
|
fn RemoveAttributeNS(self,
|
|
namespace: Option<DOMString>,
|
|
localname: DOMString) {
|
|
let namespace = namespace::from_domstring(namespace);
|
|
self.remove_attribute(namespace, localname.as_slice())
|
|
}
|
|
|
|
// http://dom.spec.whatwg.org/#dom-element-hasattribute
|
|
fn HasAttribute(self,
|
|
name: DOMString) -> bool {
|
|
self.has_attribute(name.as_slice())
|
|
}
|
|
|
|
// http://dom.spec.whatwg.org/#dom-element-hasattributens
|
|
fn HasAttributeNS(self,
|
|
namespace: Option<DOMString>,
|
|
local_name: DOMString) -> bool {
|
|
self.GetAttributeNS(namespace, local_name).is_some()
|
|
}
|
|
|
|
fn GetElementsByTagName(self, localname: DOMString) -> Temporary<HTMLCollection> {
|
|
let window = window_from_node(self).root();
|
|
HTMLCollection::by_tag_name(*window, NodeCast::from_ref(self), localname)
|
|
}
|
|
|
|
fn GetElementsByTagNameNS(self, maybe_ns: Option<DOMString>,
|
|
localname: DOMString) -> Temporary<HTMLCollection> {
|
|
let window = window_from_node(self).root();
|
|
HTMLCollection::by_tag_name_ns(*window, NodeCast::from_ref(self), localname, maybe_ns)
|
|
}
|
|
|
|
fn GetElementsByClassName(self, classes: DOMString) -> Temporary<HTMLCollection> {
|
|
let window = window_from_node(self).root();
|
|
HTMLCollection::by_class_name(*window, NodeCast::from_ref(self), classes)
|
|
}
|
|
|
|
// http://dev.w3.org/csswg/cssom-view/#dom-element-getclientrects
|
|
fn GetClientRects(self) -> Temporary<DOMRectList> {
|
|
let win = window_from_node(self).root();
|
|
let node: JSRef<Node> = NodeCast::from_ref(self);
|
|
let rects = node.get_content_boxes();
|
|
let rects: Vec<Root<DOMRect>> = rects.iter().map(|r| {
|
|
DOMRect::new(
|
|
*win,
|
|
r.origin.y,
|
|
r.origin.y + r.size.height,
|
|
r.origin.x,
|
|
r.origin.x + r.size.width).root()
|
|
}).collect();
|
|
|
|
DOMRectList::new(*win, rects.iter().map(|rect| rect.deref().clone()).collect())
|
|
}
|
|
|
|
// http://dev.w3.org/csswg/cssom-view/#dom-element-getboundingclientrect
|
|
fn GetBoundingClientRect(self) -> Temporary<DOMRect> {
|
|
let win = window_from_node(self).root();
|
|
let node: JSRef<Node> = NodeCast::from_ref(self);
|
|
let rect = node.get_bounding_content_box();
|
|
DOMRect::new(
|
|
*win,
|
|
rect.origin.y,
|
|
rect.origin.y + rect.size.height,
|
|
rect.origin.x,
|
|
rect.origin.x + rect.size.width)
|
|
}
|
|
|
|
fn GetInnerHTML(self) -> Fallible<DOMString> {
|
|
//XXX TODO: XML case
|
|
Ok(serialize(&mut NodeIterator::new(NodeCast::from_ref(self), false, false)))
|
|
}
|
|
|
|
fn GetOuterHTML(self) -> Fallible<DOMString> {
|
|
Ok(serialize(&mut NodeIterator::new(NodeCast::from_ref(self), true, false)))
|
|
}
|
|
|
|
// http://dom.spec.whatwg.org/#dom-parentnode-children
|
|
fn Children(self) -> Temporary<HTMLCollection> {
|
|
let window = window_from_node(self).root();
|
|
HTMLCollection::children(*window, NodeCast::from_ref(self))
|
|
}
|
|
|
|
// http://dom.spec.whatwg.org/#dom-parentnode-queryselector
|
|
fn QuerySelector(self, selectors: DOMString) -> Fallible<Option<Temporary<Element>>> {
|
|
let root: JSRef<Node> = NodeCast::from_ref(self);
|
|
root.query_selector(selectors)
|
|
}
|
|
|
|
// http://dom.spec.whatwg.org/#dom-parentnode-queryselectorall
|
|
fn QuerySelectorAll(self, selectors: DOMString) -> Fallible<Temporary<NodeList>> {
|
|
let root: JSRef<Node> = NodeCast::from_ref(self);
|
|
root.query_selector_all(selectors)
|
|
}
|
|
|
|
// http://dom.spec.whatwg.org/#dom-childnode-remove
|
|
fn Remove(self) {
|
|
let node: JSRef<Node> = NodeCast::from_ref(self);
|
|
node.remove_self();
|
|
}
|
|
|
|
// http://dom.spec.whatwg.org/#dom-element-matches
|
|
fn Matches(self, selectors: DOMString) -> Fallible<bool> {
|
|
match parse_selector_list_from_str(selectors.as_slice()) {
|
|
Err(()) => Err(Syntax),
|
|
Ok(ref selectors) => {
|
|
let root: JSRef<Node> = NodeCast::from_ref(self);
|
|
Ok(matches(selectors, &root, &mut None))
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn get_attribute_parts<'a>(name: &'a str) -> (Option<&'a str>, &'a str) {
|
|
//FIXME: Throw for XML-invalid names
|
|
//FIXME: Throw for XMLNS-invalid names
|
|
let (prefix, local_name) = if name.contains(":") {
|
|
let mut parts = name.splitn(1, ':');
|
|
(Some(parts.next().unwrap()), parts.next().unwrap())
|
|
} else {
|
|
(None, name)
|
|
};
|
|
|
|
(prefix, local_name)
|
|
}
|
|
|
|
impl<'a> VirtualMethods for JSRef<'a, Element> {
|
|
fn super_type<'a>(&'a self) -> Option<&'a VirtualMethods> {
|
|
let node: &JSRef<Node> = NodeCast::from_borrowed_ref(self);
|
|
Some(node as &VirtualMethods)
|
|
}
|
|
|
|
fn after_set_attr(&self, name: &Atom, value: DOMString) {
|
|
match self.super_type() {
|
|
Some(ref s) => s.after_set_attr(name, value.clone()),
|
|
_ => (),
|
|
}
|
|
|
|
match name.as_slice() {
|
|
"style" => {
|
|
let doc = document_from_node(*self).root();
|
|
let base_url = doc.url().clone();
|
|
let style = Some(style::parse_style_attribute(value.as_slice(), &base_url));
|
|
*self.style_attribute.borrow_mut() = style;
|
|
}
|
|
"id" => {
|
|
let node: JSRef<Node> = NodeCast::from_ref(*self);
|
|
if node.is_in_doc() && !value.is_empty() {
|
|
let doc = document_from_node(*self).root();
|
|
let value = Atom::from_slice(value.as_slice());
|
|
doc.register_named_element(*self, value);
|
|
}
|
|
}
|
|
_ => ()
|
|
}
|
|
|
|
self.notify_attribute_changed(name);
|
|
}
|
|
|
|
fn before_remove_attr(&self, name: &Atom, value: DOMString) {
|
|
match self.super_type() {
|
|
Some(ref s) => s.before_remove_attr(name, value.clone()),
|
|
_ => (),
|
|
}
|
|
|
|
match name.as_slice() {
|
|
"style" => {
|
|
*self.style_attribute.borrow_mut() = None;
|
|
}
|
|
"id" => {
|
|
let node: JSRef<Node> = NodeCast::from_ref(*self);
|
|
if node.is_in_doc() && !value.is_empty() {
|
|
let doc = document_from_node(*self).root();
|
|
let value = Atom::from_slice(value.as_slice());
|
|
doc.unregister_named_element(*self, value);
|
|
}
|
|
}
|
|
_ => ()
|
|
}
|
|
|
|
self.notify_attribute_changed(name);
|
|
}
|
|
|
|
fn parse_plain_attribute(&self, name: &str, value: DOMString) -> AttrValue {
|
|
match name {
|
|
"id" => AttrValue::from_atomic(value),
|
|
"class" => AttrValue::from_tokenlist(value),
|
|
_ => self.super_type().unwrap().parse_plain_attribute(name, value),
|
|
}
|
|
}
|
|
|
|
fn bind_to_tree(&self, tree_in_doc: bool) {
|
|
match self.super_type() {
|
|
Some(ref s) => s.bind_to_tree(tree_in_doc),
|
|
_ => (),
|
|
}
|
|
|
|
if !tree_in_doc { return; }
|
|
|
|
match self.get_attribute(ns!(""), "id").root() {
|
|
Some(attr) => {
|
|
let doc = document_from_node(*self).root();
|
|
let value = attr.Value();
|
|
if !value.is_empty() {
|
|
let value = Atom::from_slice(value.as_slice());
|
|
doc.register_named_element(*self, value);
|
|
}
|
|
}
|
|
_ => ()
|
|
}
|
|
}
|
|
|
|
fn unbind_from_tree(&self, tree_in_doc: bool) {
|
|
match self.super_type() {
|
|
Some(ref s) => s.unbind_from_tree(tree_in_doc),
|
|
_ => (),
|
|
}
|
|
|
|
if !tree_in_doc { return; }
|
|
|
|
match self.get_attribute(ns!(""), "id").root() {
|
|
Some(attr) => {
|
|
let doc = document_from_node(*self).root();
|
|
let value = attr.Value();
|
|
if !value.is_empty() {
|
|
let value = Atom::from_slice(value.as_slice());
|
|
doc.unregister_named_element(*self, value);
|
|
}
|
|
}
|
|
_ => ()
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<'a> style::TElement<'a> for JSRef<'a, Element> {
|
|
fn get_attr(self, namespace: &Namespace, attr: &str) -> Option<&'a str> {
|
|
self.get_attribute(namespace.clone(), attr).root().map(|attr| {
|
|
unsafe { mem::transmute(attr.value().as_slice()) }
|
|
})
|
|
}
|
|
fn get_attrs(self, attr: &str) -> Vec<&'a str> {
|
|
self.get_attributes(attr).iter().map(|attr| attr.root()).map(|attr| {
|
|
unsafe { mem::transmute(attr.value().as_slice()) }
|
|
}).collect()
|
|
}
|
|
fn get_link(self) -> Option<&'a str> {
|
|
// FIXME: This is HTML only.
|
|
let node: JSRef<Node> = NodeCast::from_ref(self);
|
|
match node.type_id() {
|
|
// http://www.whatwg.org/specs/web-apps/current-work/multipage/selectors.html#
|
|
// selector-link
|
|
ElementNodeTypeId(HTMLAnchorElementTypeId) |
|
|
ElementNodeTypeId(HTMLAreaElementTypeId) |
|
|
ElementNodeTypeId(HTMLLinkElementTypeId) => self.get_attr(&ns!(""), "href"),
|
|
_ => None,
|
|
}
|
|
}
|
|
fn get_local_name(self) -> &'a Atom {
|
|
// FIXME(zwarich): Remove this when UFCS lands and there is a better way
|
|
// of disambiguating methods.
|
|
fn get_local_name<'a, T: ElementHelpers<'a>>(this: T) -> &'a Atom {
|
|
this.get_local_name()
|
|
}
|
|
|
|
get_local_name(self)
|
|
}
|
|
fn get_namespace(self) -> &'a Namespace {
|
|
// FIXME(zwarich): Remove this when UFCS lands and there is a better way
|
|
// of disambiguating methods.
|
|
fn get_namespace<'a, T: ElementHelpers<'a>>(this: T) -> &'a Namespace {
|
|
this.get_namespace()
|
|
}
|
|
|
|
get_namespace(self)
|
|
}
|
|
fn get_hover_state(self) -> bool {
|
|
let node: JSRef<Node> = NodeCast::from_ref(self);
|
|
node.get_hover_state()
|
|
}
|
|
fn get_id(self) -> Option<Atom> {
|
|
self.get_attribute(ns!(""), "id").map(|attr| {
|
|
let attr = attr.root();
|
|
match *attr.value() {
|
|
AtomAttrValue(ref val) => val.clone(),
|
|
_ => fail!("`id` attribute should be AtomAttrValue"),
|
|
}
|
|
})
|
|
}
|
|
fn get_disabled_state(self) -> bool {
|
|
let node: JSRef<Node> = NodeCast::from_ref(self);
|
|
node.get_disabled_state()
|
|
}
|
|
fn get_enabled_state(self) -> bool {
|
|
let node: JSRef<Node> = NodeCast::from_ref(self);
|
|
node.get_enabled_state()
|
|
}
|
|
fn has_class(self, name: &str) -> bool {
|
|
// FIXME(zwarich): Remove this when UFCS lands and there is a better way
|
|
// of disambiguating methods.
|
|
fn has_class<T: AttributeHandlers>(this: T, name: &str) -> bool {
|
|
this.has_class(name)
|
|
}
|
|
|
|
has_class(self, name)
|
|
}
|
|
}
|