Simplify counter CSS properties

This commit is contained in:
Anthony Ramine 2018-02-04 03:15:15 +01:00
parent 2f8cd0fe22
commit df11a7fec7
7 changed files with 120 additions and 186 deletions

View file

@ -4,99 +4,64 @@
//! Specified types for counter properties.
use cssparser::Parser;
use cssparser::{Token, Parser};
use parser::{Parse, ParserContext};
use style_traits::ParseError;
use style_traits::{ParseError, StyleParseErrorKind};
use values::CustomIdent;
use values::generics::counters::CounterIntegerList;
use values::generics::counters::CounterIncrement as GenericCounterIncrement;
use values::generics::counters::CounterReset as GenericCounterReset;
use values::specified::Integer;
/// A specified value for the `counter-increment` and `counter-reset` property.
type SpecifiedIntegerList = CounterIntegerList<Integer>;
impl SpecifiedIntegerList {
fn parse_with_default<'i, 't>(
context: &ParserContext,
input: &mut Parser<'i, 't>,
default_value: i32
) -> Result<SpecifiedIntegerList, ParseError<'i>> {
use cssparser::Token;
use style_traits::StyleParseErrorKind;
if input.try(|input| input.expect_ident_matching("none")).is_ok() {
return Ok(CounterIntegerList::new(Vec::new()))
}
let mut counters: Vec<(CustomIdent, Integer)> = Vec::new();
loop {
let location = input.current_source_location();
let counter_name = match input.next() {
Ok(&Token::Ident(ref ident)) => CustomIdent::from_ident(location, ident, &["none"])?,
Ok(t) => return Err(location.new_unexpected_token_error(t.clone())),
Err(_) => break,
};
let counter_delta = input.try(|input| Integer::parse(context, input))
.unwrap_or(Integer::new(default_value));
counters.push((counter_name, counter_delta))
}
if !counters.is_empty() {
Ok(CounterIntegerList::new(counters))
} else {
Err(input.new_custom_error(StyleParseErrorKind::UnspecifiedError))
}
}
}
/// A specified value for the `counter-increment` property.
#[cfg_attr(feature = "gecko", derive(MallocSizeOf))]
#[derive(Clone, Debug, PartialEq, ToCss)]
pub struct CounterIncrement(pub SpecifiedIntegerList);
impl CounterIncrement {
/// Returns a new specified `counter-increment` object with the given values.
pub fn new(vec: Vec<(CustomIdent, Integer)>) -> CounterIncrement {
CounterIncrement(SpecifiedIntegerList::new(vec))
}
/// Returns the values of the specified `counter-increment` object.
pub fn get_values(&self) -> &[(CustomIdent, Integer)] {
self.0.get_values()
}
}
pub type CounterIncrement = GenericCounterIncrement<Integer>;
impl Parse for CounterIncrement {
fn parse<'i, 't>(
context: &ParserContext,
input: &mut Parser<'i, 't>
) -> Result<CounterIncrement, ParseError<'i>> {
Ok(CounterIncrement(SpecifiedIntegerList::parse_with_default(context, input, 1)?))
) -> Result<Self, ParseError<'i>> {
Ok(Self::new(parse_counters(context, input, 1)?))
}
}
/// A specified value for the `counter-reset` property.
#[cfg_attr(feature = "gecko", derive(MallocSizeOf))]
#[derive(Clone, Debug, PartialEq, ToCss)]
pub struct CounterReset(pub SpecifiedIntegerList);
impl CounterReset {
/// Returns a new specified `counter-reset` object with the given values.
pub fn new(vec: Vec<(CustomIdent, Integer)>) -> CounterReset {
CounterReset(SpecifiedIntegerList::new(vec))
}
/// Returns the values of the specified `counter-reset` object.
pub fn get_values(&self) -> &[(CustomIdent, Integer)] {
self.0.get_values()
}
}
/// A specified value for the `counter-increment` property.
pub type CounterReset = GenericCounterReset<Integer>;
impl Parse for CounterReset {
fn parse<'i, 't>(
context: &ParserContext,
input: &mut Parser<'i, 't>
) -> Result<CounterReset, ParseError<'i>> {
Ok(CounterReset(SpecifiedIntegerList::parse_with_default(context, input, 0)?))
) -> Result<Self, ParseError<'i>> {
Ok(Self::new(parse_counters(context, input, 0)?))
}
}
fn parse_counters<'i, 't>(
context: &ParserContext,
input: &mut Parser<'i, 't>,
default_value: i32,
) -> Result<Vec<(CustomIdent, Integer)>, ParseError<'i>> {
if input.try(|input| input.expect_ident_matching("none")).is_ok() {
return Ok(vec![]);
}
let mut counters = Vec::new();
loop {
let location = input.current_source_location();
let counter_name = match input.next() {
Ok(&Token::Ident(ref ident)) => CustomIdent::from_ident(location, ident, &["none"])?,
Ok(t) => return Err(location.new_unexpected_token_error(t.clone())),
Err(_) => break,
};
let counter_delta = input.try(|input| Integer::parse(context, input))
.unwrap_or(Integer::new(default_value));
counters.push((counter_name, counter_delta))
}
if !counters.is_empty() {
Ok(counters)
} else {
Err(input.new_custom_error(StyleParseErrorKind::UnspecifiedError))
}
}