mirror of
https://github.com/servo/servo.git
synced 2025-06-30 12:03:38 +01:00
144 lines
4.7 KiB
Rust
144 lines
4.7 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/. */
|
|
|
|
//! `list` specified values.
|
|
|
|
use cssparser::{Parser, Token};
|
|
use parser::{Parse, ParserContext};
|
|
use std::fmt::{self, Write};
|
|
use style_traits::{CssWriter, ParseError, StyleParseErrorKind, ToCss};
|
|
#[cfg(feature = "gecko")]
|
|
use values::CustomIdent;
|
|
#[cfg(feature = "gecko")]
|
|
use values::generics::CounterStyleOrNone;
|
|
|
|
/// Specified and computed `list-style-type` property.
|
|
#[cfg(feature = "gecko")]
|
|
#[derive(Clone, Debug, Eq, MallocSizeOf, PartialEq, SpecifiedValueInfo,
|
|
ToComputedValue, ToCss)]
|
|
pub enum ListStyleType {
|
|
/// <counter-style> | none
|
|
CounterStyle(CounterStyleOrNone),
|
|
/// <string>
|
|
String(String),
|
|
}
|
|
|
|
#[cfg(feature = "gecko")]
|
|
impl ListStyleType {
|
|
/// Initial specified value for `list-style-type`.
|
|
#[inline]
|
|
pub fn disc() -> Self {
|
|
ListStyleType::CounterStyle(CounterStyleOrNone::disc())
|
|
}
|
|
|
|
/// Convert from gecko keyword to list-style-type.
|
|
///
|
|
/// This should only be used for mapping type attribute to
|
|
/// list-style-type, and thus only values possible in that
|
|
/// attribute is considered here.
|
|
pub fn from_gecko_keyword(value: u32) -> Self {
|
|
use gecko_bindings::structs;
|
|
|
|
if value == structs::NS_STYLE_LIST_STYLE_NONE {
|
|
return ListStyleType::CounterStyle(CounterStyleOrNone::None);
|
|
}
|
|
|
|
ListStyleType::CounterStyle(CounterStyleOrNone::Name(CustomIdent(match value {
|
|
structs::NS_STYLE_LIST_STYLE_DISC => atom!("disc"),
|
|
structs::NS_STYLE_LIST_STYLE_CIRCLE => atom!("circle"),
|
|
structs::NS_STYLE_LIST_STYLE_SQUARE => atom!("square"),
|
|
structs::NS_STYLE_LIST_STYLE_DECIMAL => atom!("decimal"),
|
|
structs::NS_STYLE_LIST_STYLE_LOWER_ROMAN => atom!("lower-roman"),
|
|
structs::NS_STYLE_LIST_STYLE_UPPER_ROMAN => atom!("upper-roman"),
|
|
structs::NS_STYLE_LIST_STYLE_LOWER_ALPHA => atom!("lower-alpha"),
|
|
structs::NS_STYLE_LIST_STYLE_UPPER_ALPHA => atom!("upper-alpha"),
|
|
_ => unreachable!("Unknown counter style keyword value"),
|
|
})))
|
|
}
|
|
}
|
|
|
|
#[cfg(feature = "gecko")]
|
|
impl Parse for ListStyleType {
|
|
fn parse<'i, 't>(
|
|
context: &ParserContext,
|
|
input: &mut Parser<'i, 't>,
|
|
) -> Result<Self, ParseError<'i>> {
|
|
if let Ok(style) = input.try(|i| CounterStyleOrNone::parse(context, i)) {
|
|
return Ok(ListStyleType::CounterStyle(style));
|
|
}
|
|
|
|
Ok(ListStyleType::String(
|
|
input.expect_string()?.as_ref().to_owned(),
|
|
))
|
|
}
|
|
}
|
|
|
|
/// Specified and computed `quote` property.
|
|
///
|
|
/// FIXME(emilio): It's a shame that this allocates all the time it's computed,
|
|
/// probably should just be refcounted.
|
|
/// FIXME This can probably derive ToCss.
|
|
#[derive(Clone, Debug, MallocSizeOf, PartialEq, SpecifiedValueInfo,
|
|
ToComputedValue)]
|
|
pub struct Quotes(#[css(if_empty = "none")] pub Box<[(Box<str>, Box<str>)]>);
|
|
|
|
impl ToCss for Quotes {
|
|
fn to_css<W>(&self, dest: &mut CssWriter<W>) -> fmt::Result
|
|
where
|
|
W: Write,
|
|
{
|
|
let mut iter = self.0.iter();
|
|
|
|
match iter.next() {
|
|
Some(&(ref l, ref r)) => {
|
|
l.to_css(dest)?;
|
|
dest.write_char(' ')?;
|
|
r.to_css(dest)?;
|
|
},
|
|
None => return dest.write_str("none"),
|
|
}
|
|
|
|
for &(ref l, ref r) in iter {
|
|
dest.write_char(' ')?;
|
|
l.to_css(dest)?;
|
|
dest.write_char(' ')?;
|
|
r.to_css(dest)?;
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
impl Parse for Quotes {
|
|
fn parse<'i, 't>(
|
|
_: &ParserContext,
|
|
input: &mut Parser<'i, 't>,
|
|
) -> Result<Quotes, ParseError<'i>> {
|
|
if input
|
|
.try(|input| input.expect_ident_matching("none"))
|
|
.is_ok()
|
|
{
|
|
return Ok(Quotes(Vec::new().into_boxed_slice()));
|
|
}
|
|
|
|
let mut quotes = Vec::new();
|
|
loop {
|
|
let location = input.current_source_location();
|
|
let first = match input.next() {
|
|
Ok(&Token::QuotedString(ref value)) => value.as_ref().to_owned().into_boxed_str(),
|
|
Ok(t) => return Err(location.new_unexpected_token_error(t.clone())),
|
|
Err(_) => break,
|
|
};
|
|
|
|
let second = input.expect_string()?.as_ref().to_owned().into_boxed_str();
|
|
quotes.push((first, second))
|
|
}
|
|
|
|
if !quotes.is_empty() {
|
|
Ok(Quotes(quotes.into_boxed_slice()))
|
|
} else {
|
|
Err(input.new_custom_error(StyleParseErrorKind::UnspecifiedError))
|
|
}
|
|
}
|
|
}
|