/* 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/. */ //! Generic types that share their serialization implementations //! for both specified and computed values. use counter_style::{Symbols, parse_counter_style_name}; use cssparser::Parser; use euclid::size::Size2D; use parser::{Parse, ParserContext}; use std::fmt; use style_traits::{HasViewportPercentage, OneOrMoreCommaSeparated, ToCss}; use super::CustomIdent; pub use self::basic_shape::serialize_radius_values; pub mod background; pub mod basic_shape; pub mod border; pub mod grid; pub mod image; pub mod position; pub mod rect; #[derive(Clone, Debug, PartialEq, ToComputedValue)] #[cfg_attr(feature = "servo", derive(HeapSizeOf))] /// A type for representing CSS `width` and `height` values. pub struct BorderRadiusSize(pub Size2D); impl HasViewportPercentage for BorderRadiusSize { #[inline] fn has_viewport_percentage(&self) -> bool { false } } impl From for BorderRadiusSize { fn from(other: L) -> Self { Self::new(other.clone(), other) } } impl BorderRadiusSize { #[inline] /// Create a new `BorderRadiusSize` for an area of given width and height. pub fn new(width: L, height: L) -> BorderRadiusSize { BorderRadiusSize(Size2D::new(width, height)) } } impl BorderRadiusSize { #[inline] /// Create a new `BorderRadiusSize` for a circle of given radius. pub fn circle(radius: L) -> BorderRadiusSize { BorderRadiusSize(Size2D::new(radius.clone(), radius)) } } impl ToCss for BorderRadiusSize { #[inline] fn to_css(&self, dest: &mut W) -> fmt::Result where W: fmt::Write { self.0.width.to_css(dest)?; dest.write_str(" ")?; self.0.height.to_css(dest) } } // https://drafts.csswg.org/css-counter-styles/#typedef-symbols-type define_css_keyword_enum! { SymbolsType: "cyclic" => Cyclic, "numeric" => Numeric, "alphabetic" => Alphabetic, "symbolic" => Symbolic, "fixed" => Fixed, } add_impls_for_keyword_enum!(SymbolsType); #[cfg(feature = "gecko")] impl SymbolsType { /// Convert symbols type to their corresponding Gecko values. pub fn to_gecko_keyword(self) -> u8 { use gecko_bindings::structs; match self { SymbolsType::Cyclic => structs::NS_STYLE_COUNTER_SYSTEM_CYCLIC as u8, SymbolsType::Numeric => structs::NS_STYLE_COUNTER_SYSTEM_NUMERIC as u8, SymbolsType::Alphabetic => structs::NS_STYLE_COUNTER_SYSTEM_ALPHABETIC as u8, SymbolsType::Symbolic => structs::NS_STYLE_COUNTER_SYSTEM_SYMBOLIC as u8, SymbolsType::Fixed => structs::NS_STYLE_COUNTER_SYSTEM_FIXED as u8, } } } /// https://drafts.csswg.org/css-counter-styles/#typedef-counter-style /// /// Since wherever is used, 'none' is a valid value as /// well, we combine them into one type to make code simpler. #[derive(Clone, Debug, PartialEq, Eq)] pub enum CounterStyleOrNone { /// none None_, /// Name(CustomIdent), /// symbols() Symbols(SymbolsType, Symbols), } impl CounterStyleOrNone { /// disc value pub fn disc() -> Self { CounterStyleOrNone::Name(CustomIdent(atom!("disc"))) } /// decimal value pub fn decimal() -> Self { CounterStyleOrNone::Name(CustomIdent(atom!("decimal"))) } } no_viewport_percentage!(CounterStyleOrNone); impl Parse for CounterStyleOrNone { fn parse(context: &ParserContext, input: &mut Parser) -> Result { if let Ok(name) = input.try(|i| parse_counter_style_name(i)) { return Ok(CounterStyleOrNone::Name(name)); } if input.try(|i| i.expect_ident_matching("none")).is_ok() { return Ok(CounterStyleOrNone::None_); } if input.try(|i| i.expect_function_matching("symbols")).is_ok() { return input.parse_nested_block(|input| { let symbols_type = input.try(|i| SymbolsType::parse(i)) .unwrap_or(SymbolsType::Symbolic); let symbols = Symbols::parse(context, input)?; // There must be at least two symbols for alphabetic or // numeric system. if (symbols_type == SymbolsType::Alphabetic || symbols_type == SymbolsType::Numeric) && symbols.0.len() < 2 { return Err(()); } // Identifier is not allowed in symbols() function. if symbols.0.iter().any(|sym| !sym.is_allowed_in_symbols()) { return Err(()); } Ok(CounterStyleOrNone::Symbols(symbols_type, symbols)) }); } Err(()) } } impl ToCss for CounterStyleOrNone { #[inline] fn to_css(&self, dest: &mut W) -> fmt::Result where W: fmt::Write { match self { &CounterStyleOrNone::None_ => dest.write_str("none"), &CounterStyleOrNone::Name(ref name) => name.to_css(dest), &CounterStyleOrNone::Symbols(ref symbols_type, ref symbols) => { dest.write_str("symbols(")?; symbols_type.to_css(dest)?; dest.write_str(" ")?; symbols.to_css(dest)?; dest.write_str(")") } } } } /// A settings tag, defined by a four-character tag and a setting value /// /// For font-feature-settings, this is a tag and an integer, /// for font-variation-settings this is a tag and a float #[derive(Clone, Debug, Eq, PartialEq)] #[cfg_attr(feature = "servo", derive(HeapSizeOf))] pub struct FontSettingTag { /// A four-character tag, packed into a u32 (one byte per character) pub tag: u32, /// The value pub value: T, } impl OneOrMoreCommaSeparated for FontSettingTag {} impl ToCss for FontSettingTag { fn to_css(&self, dest: &mut W) -> fmt::Result where W: fmt::Write { use byteorder::{WriteBytesExt, BigEndian}; use cssparser::serialize_string; use std::str; let mut raw: Vec = vec!(); raw.write_u32::(self.tag).unwrap(); serialize_string(str::from_utf8(&raw).unwrap_or_default(), dest)?; self.value.to_css(dest) } } impl Parse for FontSettingTag { /// https://www.w3.org/TR/css-fonts-3/#propdef-font-feature-settings /// https://drafts.csswg.org/css-fonts-4/#low-level-font-variation- /// settings-control-the-font-variation-settings-property /// [ on | off | ] /// fn parse(context: &ParserContext, input: &mut Parser) -> Result { use byteorder::{ReadBytesExt, BigEndian}; use std::io::Cursor; let tag = try!(input.expect_string()); // allowed strings of length 4 containing chars: if tag.len() != 4 || tag.chars().any(|c| c < ' ' || c > '~') { return Err(()) } let mut raw = Cursor::new(tag.as_bytes()); let u_tag = raw.read_u32::().unwrap(); Ok(FontSettingTag { tag: u_tag, value: T::parse(context, input)? }) } } /// A font settings value for font-variation-settings or font-feature-settings #[derive(Clone, Debug, Eq, PartialEq)] #[cfg_attr(feature = "servo", derive(HeapSizeOf))] pub enum FontSettings { /// No settings (default) Normal, /// Set of settings Tag(Vec>) } impl ToCss for FontSettings { fn to_css(&self, dest: &mut W) -> fmt::Result where W: fmt::Write { match *self { FontSettings::Normal => dest.write_str("normal"), FontSettings::Tag(ref ftvs) => ftvs.to_css(dest) } } } impl Parse for FontSettings { /// https://www.w3.org/TR/css-fonts-3/#propdef-font-feature-settings fn parse(context: &ParserContext, input: &mut Parser) -> Result { if input.try(|i| i.expect_ident_matching("normal")).is_ok() { return Ok(FontSettings::Normal); } Vec::parse(context, input).map(FontSettings::Tag) } } /// An integer that can also parse "on" and "off", /// for font-feature-settings /// /// Do not use this type anywhere except within FontSettings /// because it serializes with the preceding space #[derive(Clone, Copy, Debug, Eq, PartialEq)] #[cfg_attr(feature = "servo", derive(HeapSizeOf))] pub struct FontSettingTagInt(pub u32); /// A number value to be used for font-variation-settings /// /// Do not use this type anywhere except within FontSettings /// because it serializes with the preceding space #[derive(Clone, Debug, PartialEq)] #[cfg_attr(feature = "servo", derive(HeapSizeOf))] pub struct FontSettingTagFloat(pub f32); impl ToCss for FontSettingTagInt { fn to_css(&self, dest: &mut W) -> fmt::Result where W: fmt::Write { match self.0 { 1 => Ok(()), 0 => dest.write_str(" off"), x => write!(dest, " {}", x) } } } impl Parse for FontSettingTagInt { fn parse(_context: &ParserContext, input: &mut Parser) -> Result { if let Ok(value) = input.try(|input| input.expect_integer()) { // handle integer, throw if it is negative if value >= 0 { Ok(FontSettingTagInt(value as u32)) } else { Err(()) } } else if let Ok(_) = input.try(|input| input.expect_ident_matching("on")) { // on is an alias for '1' Ok(FontSettingTagInt(1)) } else if let Ok(_) = input.try(|input| input.expect_ident_matching("off")) { // off is an alias for '0' Ok(FontSettingTagInt(0)) } else { // empty value is an alias for '1' Ok(FontSettingTagInt(1)) } } } impl Parse for FontSettingTagFloat { fn parse(_: &ParserContext, input: &mut Parser) -> Result { input.expect_number().map(FontSettingTagFloat) } } impl ToCss for FontSettingTagFloat { fn to_css(&self, dest: &mut W) -> fmt::Result where W: fmt::Write { dest.write_str(" ")?; self.0.to_css(dest) } }