mirror of
https://github.com/servo/servo.git
synced 2025-08-05 21:50:18 +01:00
Hoist AttrValue into style.
This commit is contained in:
parent
4ec96963f8
commit
488c1291d4
4 changed files with 175 additions and 163 deletions
|
@ -2,7 +2,6 @@
|
||||||
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
* 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/. */
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
||||||
|
|
||||||
use cssparser::RGBA;
|
|
||||||
use devtools_traits::AttrInfo;
|
use devtools_traits::AttrInfo;
|
||||||
use dom::bindings::cell::DOMRefCell;
|
use dom::bindings::cell::DOMRefCell;
|
||||||
use dom::bindings::codegen::Bindings::AttrBinding::{self, AttrMethods};
|
use dom::bindings::codegen::Bindings::AttrBinding::{self, AttrMethods};
|
||||||
|
@ -12,174 +11,14 @@ use dom::bindings::js::{JS, MutNullableHeap};
|
||||||
use dom::bindings::js::{LayoutJS, Root, RootedReference};
|
use dom::bindings::js::{LayoutJS, Root, RootedReference};
|
||||||
use dom::bindings::reflector::{Reflector, reflect_dom_object};
|
use dom::bindings::reflector::{Reflector, reflect_dom_object};
|
||||||
use dom::element::{AttributeMutation, Element};
|
use dom::element::{AttributeMutation, Element};
|
||||||
use dom::values::UNSIGNED_LONG_MAX;
|
|
||||||
use dom::virtualmethods::vtable_for;
|
use dom::virtualmethods::vtable_for;
|
||||||
use dom::window::Window;
|
use dom::window::Window;
|
||||||
use std::borrow::ToOwned;
|
use std::borrow::ToOwned;
|
||||||
use std::cell::Ref;
|
use std::cell::Ref;
|
||||||
use std::mem;
|
use std::mem;
|
||||||
use std::ops::Deref;
|
|
||||||
use string_cache::{Atom, Namespace};
|
use string_cache::{Atom, Namespace};
|
||||||
use style::values::specified::Length;
|
pub use style::attr::AttrValue;
|
||||||
use util::str::{DOMString, LengthOrPercentageOrAuto, parse_unsigned_integer, parse_legacy_color, parse_length};
|
use util::str::DOMString;
|
||||||
use util::str::{split_html_space_chars, str_join};
|
|
||||||
|
|
||||||
#[derive(JSTraceable, PartialEq, Clone, HeapSizeOf)]
|
|
||||||
pub enum AttrValue {
|
|
||||||
String(DOMString),
|
|
||||||
TokenList(DOMString, Vec<Atom>),
|
|
||||||
UInt(DOMString, u32),
|
|
||||||
Atom(Atom),
|
|
||||||
Length(DOMString, Option<Length>),
|
|
||||||
Color(DOMString, Option<RGBA>),
|
|
||||||
Dimension(DOMString, LengthOrPercentageOrAuto),
|
|
||||||
}
|
|
||||||
|
|
||||||
impl AttrValue {
|
|
||||||
pub fn from_serialized_tokenlist(tokens: DOMString) -> AttrValue {
|
|
||||||
let atoms =
|
|
||||||
split_html_space_chars(&tokens)
|
|
||||||
.map(Atom::from_slice)
|
|
||||||
.fold(vec![], |mut acc, atom| {
|
|
||||||
if !acc.contains(&atom) { acc.push(atom) }
|
|
||||||
acc
|
|
||||||
});
|
|
||||||
AttrValue::TokenList(tokens, atoms)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn from_atomic_tokens(atoms: Vec<Atom>) -> AttrValue {
|
|
||||||
let tokens = DOMString(str_join(&atoms, "\x20"));
|
|
||||||
AttrValue::TokenList(tokens, atoms)
|
|
||||||
}
|
|
||||||
|
|
||||||
// https://html.spec.whatwg.org/multipage/#reflecting-content-attributes-in-idl-attributes:idl-unsigned-long
|
|
||||||
pub fn from_u32(string: DOMString, default: u32) -> AttrValue {
|
|
||||||
let result = parse_unsigned_integer(string.chars()).unwrap_or(default);
|
|
||||||
let result = if result > UNSIGNED_LONG_MAX {
|
|
||||||
default
|
|
||||||
} else {
|
|
||||||
result
|
|
||||||
};
|
|
||||||
AttrValue::UInt(string, result)
|
|
||||||
}
|
|
||||||
|
|
||||||
// https://html.spec.whatwg.org/multipage/#limited-to-only-non-negative-numbers-greater-than-zero
|
|
||||||
pub fn from_limited_u32(string: DOMString, default: u32) -> AttrValue {
|
|
||||||
let result = parse_unsigned_integer(string.chars()).unwrap_or(default);
|
|
||||||
let result = if result == 0 || result > UNSIGNED_LONG_MAX {
|
|
||||||
default
|
|
||||||
} else {
|
|
||||||
result
|
|
||||||
};
|
|
||||||
AttrValue::UInt(string, result)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn from_atomic(string: DOMString) -> AttrValue {
|
|
||||||
let value = Atom::from_slice(&string);
|
|
||||||
AttrValue::Atom(value)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn from_legacy_color(string: DOMString) -> AttrValue {
|
|
||||||
let parsed = parse_legacy_color(&string).ok();
|
|
||||||
AttrValue::Color(string, parsed)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn from_dimension(string: DOMString) -> AttrValue {
|
|
||||||
let parsed = parse_length(&string);
|
|
||||||
AttrValue::Dimension(string, parsed)
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Assumes the `AttrValue` is a `TokenList` and returns its tokens
|
|
||||||
///
|
|
||||||
/// ## Panics
|
|
||||||
///
|
|
||||||
/// Panics if the `AttrValue` is not a `TokenList`
|
|
||||||
pub fn as_tokens(&self) -> &[Atom] {
|
|
||||||
match *self {
|
|
||||||
AttrValue::TokenList(_, ref tokens) => tokens,
|
|
||||||
_ => panic!("Tokens not found"),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Assumes the `AttrValue` is an `Atom` and returns its value
|
|
||||||
///
|
|
||||||
/// ## Panics
|
|
||||||
///
|
|
||||||
/// Panics if the `AttrValue` is not an `Atom`
|
|
||||||
pub fn as_atom(&self) -> &Atom {
|
|
||||||
match *self {
|
|
||||||
AttrValue::Atom(ref value) => value,
|
|
||||||
_ => panic!("Atom not found"),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Assumes the `AttrValue` is a `Color` and returns its value
|
|
||||||
///
|
|
||||||
/// ## Panics
|
|
||||||
///
|
|
||||||
/// Panics if the `AttrValue` is not a `Color`
|
|
||||||
pub fn as_color(&self) -> Option<&RGBA> {
|
|
||||||
match *self {
|
|
||||||
AttrValue::Color(_, ref color) => color.as_ref(),
|
|
||||||
_ => panic!("Color not found"),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Assumes the `AttrValue` is a `Length` and returns its value
|
|
||||||
///
|
|
||||||
/// ## Panics
|
|
||||||
///
|
|
||||||
/// Panics if the `AttrValue` is not a `Length`
|
|
||||||
pub fn as_length(&self) -> Option<&Length> {
|
|
||||||
match *self {
|
|
||||||
AttrValue::Length(_, ref length) => length.as_ref(),
|
|
||||||
_ => panic!("Length not found"),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Assumes the `AttrValue` is a `Dimension` and returns its value
|
|
||||||
///
|
|
||||||
/// ## Panics
|
|
||||||
///
|
|
||||||
/// Panics if the `AttrValue` is not a `Dimension`
|
|
||||||
pub fn as_dimension(&self) -> &LengthOrPercentageOrAuto {
|
|
||||||
match *self {
|
|
||||||
AttrValue::Dimension(_, ref l) => l,
|
|
||||||
_ => panic!("Dimension not found"),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Return the AttrValue as its integer representation, if any.
|
|
||||||
/// This corresponds to attribute values returned as `AttrValue::UInt(_)`
|
|
||||||
/// by `VirtualMethods::parse_plain_attribute()`.
|
|
||||||
///
|
|
||||||
/// ## Panics
|
|
||||||
///
|
|
||||||
/// Panics if the `AttrValue` is not a `UInt`
|
|
||||||
pub fn as_uint(&self) -> u32 {
|
|
||||||
if let AttrValue::UInt(_, value) = *self {
|
|
||||||
value
|
|
||||||
} else {
|
|
||||||
panic!("Uint not found");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Deref for AttrValue {
|
|
||||||
type Target = str;
|
|
||||||
|
|
||||||
fn deref(&self) -> &str {
|
|
||||||
match *self {
|
|
||||||
AttrValue::String(ref value) |
|
|
||||||
AttrValue::TokenList(ref value, _) |
|
|
||||||
AttrValue::UInt(ref value, _) |
|
|
||||||
AttrValue::Length(ref value, _) |
|
|
||||||
AttrValue::Color(ref value, _) |
|
|
||||||
AttrValue::Dimension(ref value, _) => &value,
|
|
||||||
AttrValue::Atom(ref value) => &value,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// https://dom.spec.whatwg.org/#interface-attr
|
// https://dom.spec.whatwg.org/#interface-attr
|
||||||
#[dom_struct]
|
#[dom_struct]
|
||||||
|
|
|
@ -82,6 +82,7 @@ use std::sync::Arc;
|
||||||
use std::sync::atomic::AtomicBool;
|
use std::sync::atomic::AtomicBool;
|
||||||
use std::sync::mpsc::{Receiver, Sender};
|
use std::sync::mpsc::{Receiver, Sender};
|
||||||
use string_cache::{Atom, Namespace, QualName};
|
use string_cache::{Atom, Namespace, QualName};
|
||||||
|
use style::attr::AttrValue;
|
||||||
use style::properties::PropertyDeclarationBlock;
|
use style::properties::PropertyDeclarationBlock;
|
||||||
use style::values::specified::Length;
|
use style::values::specified::Length;
|
||||||
use url::Url;
|
use url::Url;
|
||||||
|
@ -289,6 +290,7 @@ no_jsmanaged_fields!(Length);
|
||||||
no_jsmanaged_fields!(ElementState);
|
no_jsmanaged_fields!(ElementState);
|
||||||
no_jsmanaged_fields!(DOMString);
|
no_jsmanaged_fields!(DOMString);
|
||||||
no_jsmanaged_fields!(Mime);
|
no_jsmanaged_fields!(Mime);
|
||||||
|
no_jsmanaged_fields!(AttrValue);
|
||||||
|
|
||||||
impl JSTraceable for Box<ScriptChan + Send> {
|
impl JSTraceable for Box<ScriptChan + Send> {
|
||||||
#[inline]
|
#[inline]
|
||||||
|
|
170
components/style/attr.rs
Normal file
170
components/style/attr.rs
Normal file
|
@ -0,0 +1,170 @@
|
||||||
|
/* 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 cssparser::RGBA;
|
||||||
|
use std::ops::Deref;
|
||||||
|
use string_cache::{Atom, Namespace};
|
||||||
|
use util::str::{DOMString, LengthOrPercentageOrAuto, parse_unsigned_integer, parse_legacy_color, parse_length};
|
||||||
|
use util::str::{split_html_space_chars, str_join};
|
||||||
|
use values::specified::{Length};
|
||||||
|
|
||||||
|
// Duplicated from script::dom::values.
|
||||||
|
const UNSIGNED_LONG_MAX: u32 = 2147483647;
|
||||||
|
|
||||||
|
#[derive(PartialEq, Clone, HeapSizeOf)]
|
||||||
|
pub enum AttrValue {
|
||||||
|
String(DOMString),
|
||||||
|
TokenList(DOMString, Vec<Atom>),
|
||||||
|
UInt(DOMString, u32),
|
||||||
|
Atom(Atom),
|
||||||
|
Length(DOMString, Option<Length>),
|
||||||
|
Color(DOMString, Option<RGBA>),
|
||||||
|
Dimension(DOMString, LengthOrPercentageOrAuto),
|
||||||
|
}
|
||||||
|
|
||||||
|
impl AttrValue {
|
||||||
|
pub fn from_serialized_tokenlist(tokens: DOMString) -> AttrValue {
|
||||||
|
let atoms =
|
||||||
|
split_html_space_chars(&tokens)
|
||||||
|
.map(Atom::from_slice)
|
||||||
|
.fold(vec![], |mut acc, atom| {
|
||||||
|
if !acc.contains(&atom) { acc.push(atom) }
|
||||||
|
acc
|
||||||
|
});
|
||||||
|
AttrValue::TokenList(tokens, atoms)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn from_atomic_tokens(atoms: Vec<Atom>) -> AttrValue {
|
||||||
|
let tokens = DOMString(str_join(&atoms, "\x20"));
|
||||||
|
AttrValue::TokenList(tokens, atoms)
|
||||||
|
}
|
||||||
|
|
||||||
|
// https://html.spec.whatwg.org/multipage/#reflecting-content-attributes-in-idl-attributes:idl-unsigned-long
|
||||||
|
pub fn from_u32(string: DOMString, default: u32) -> AttrValue {
|
||||||
|
let result = parse_unsigned_integer(string.chars()).unwrap_or(default);
|
||||||
|
let result = if result > UNSIGNED_LONG_MAX {
|
||||||
|
default
|
||||||
|
} else {
|
||||||
|
result
|
||||||
|
};
|
||||||
|
AttrValue::UInt(string, result)
|
||||||
|
}
|
||||||
|
|
||||||
|
// https://html.spec.whatwg.org/multipage/#limited-to-only-non-negative-numbers-greater-than-zero
|
||||||
|
pub fn from_limited_u32(string: DOMString, default: u32) -> AttrValue {
|
||||||
|
let result = parse_unsigned_integer(string.chars()).unwrap_or(default);
|
||||||
|
let result = if result == 0 || result > UNSIGNED_LONG_MAX {
|
||||||
|
default
|
||||||
|
} else {
|
||||||
|
result
|
||||||
|
};
|
||||||
|
AttrValue::UInt(string, result)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn from_atomic(string: DOMString) -> AttrValue {
|
||||||
|
let value = Atom::from_slice(&string);
|
||||||
|
AttrValue::Atom(value)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn from_legacy_color(string: DOMString) -> AttrValue {
|
||||||
|
let parsed = parse_legacy_color(&string).ok();
|
||||||
|
AttrValue::Color(string, parsed)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn from_dimension(string: DOMString) -> AttrValue {
|
||||||
|
let parsed = parse_length(&string);
|
||||||
|
AttrValue::Dimension(string, parsed)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Assumes the `AttrValue` is a `TokenList` and returns its tokens
|
||||||
|
///
|
||||||
|
/// ## Panics
|
||||||
|
///
|
||||||
|
/// Panics if the `AttrValue` is not a `TokenList`
|
||||||
|
pub fn as_tokens(&self) -> &[Atom] {
|
||||||
|
match *self {
|
||||||
|
AttrValue::TokenList(_, ref tokens) => tokens,
|
||||||
|
_ => panic!("Tokens not found"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Assumes the `AttrValue` is an `Atom` and returns its value
|
||||||
|
///
|
||||||
|
/// ## Panics
|
||||||
|
///
|
||||||
|
/// Panics if the `AttrValue` is not an `Atom`
|
||||||
|
pub fn as_atom(&self) -> &Atom {
|
||||||
|
match *self {
|
||||||
|
AttrValue::Atom(ref value) => value,
|
||||||
|
_ => panic!("Atom not found"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Assumes the `AttrValue` is a `Color` and returns its value
|
||||||
|
///
|
||||||
|
/// ## Panics
|
||||||
|
///
|
||||||
|
/// Panics if the `AttrValue` is not a `Color`
|
||||||
|
pub fn as_color(&self) -> Option<&RGBA> {
|
||||||
|
match *self {
|
||||||
|
AttrValue::Color(_, ref color) => color.as_ref(),
|
||||||
|
_ => panic!("Color not found"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Assumes the `AttrValue` is a `Length` and returns its value
|
||||||
|
///
|
||||||
|
/// ## Panics
|
||||||
|
///
|
||||||
|
/// Panics if the `AttrValue` is not a `Length`
|
||||||
|
pub fn as_length(&self) -> Option<&Length> {
|
||||||
|
match *self {
|
||||||
|
AttrValue::Length(_, ref length) => length.as_ref(),
|
||||||
|
_ => panic!("Length not found"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Assumes the `AttrValue` is a `Dimension` and returns its value
|
||||||
|
///
|
||||||
|
/// ## Panics
|
||||||
|
///
|
||||||
|
/// Panics if the `AttrValue` is not a `Dimension`
|
||||||
|
pub fn as_dimension(&self) -> &LengthOrPercentageOrAuto {
|
||||||
|
match *self {
|
||||||
|
AttrValue::Dimension(_, ref l) => l,
|
||||||
|
_ => panic!("Dimension not found"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Return the AttrValue as its integer representation, if any.
|
||||||
|
/// This corresponds to attribute values returned as `AttrValue::UInt(_)`
|
||||||
|
/// by `VirtualMethods::parse_plain_attribute()`.
|
||||||
|
///
|
||||||
|
/// ## Panics
|
||||||
|
///
|
||||||
|
/// Panics if the `AttrValue` is not a `UInt`
|
||||||
|
pub fn as_uint(&self) -> u32 {
|
||||||
|
if let AttrValue::UInt(_, value) = *self {
|
||||||
|
value
|
||||||
|
} else {
|
||||||
|
panic!("Uint not found");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Deref for AttrValue {
|
||||||
|
type Target = str;
|
||||||
|
|
||||||
|
fn deref(&self) -> &str {
|
||||||
|
match *self {
|
||||||
|
AttrValue::String(ref value) |
|
||||||
|
AttrValue::TokenList(ref value, _) |
|
||||||
|
AttrValue::UInt(ref value, _) |
|
||||||
|
AttrValue::Length(ref value, _) |
|
||||||
|
AttrValue::Color(ref value, _) |
|
||||||
|
AttrValue::Dimension(ref value, _) => &value,
|
||||||
|
AttrValue::Atom(ref value) => &value,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -42,6 +42,7 @@ extern crate url;
|
||||||
extern crate util;
|
extern crate util;
|
||||||
|
|
||||||
pub mod animation;
|
pub mod animation;
|
||||||
|
pub mod attr;
|
||||||
mod custom_properties;
|
mod custom_properties;
|
||||||
pub mod font_face;
|
pub mod font_face;
|
||||||
pub mod legacy;
|
pub mod legacy;
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue