use crate::common::{Pos, Scope, Symbol}; use crate::selector::Selector; use crate::utils::{devour_whitespace, parse_interpolation}; use crate::value::Value; use crate::{Expr, Token, TokenKind}; use std::fmt::{self, Display}; use std::iter::Peekable; /// A style: `color: red` #[derive(Clone, Debug, Eq, PartialEq)] pub(crate) struct Style { pub property: String, pub value: Value, } impl Display for Style { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { write!(f, "{}: {};", self.property, self.value) } } impl Style { pub fn parse_property>( toks: &mut Peekable, scope: &Scope, super_selector: &Selector, super_property: String, ) -> String { StyleParser::new(scope, super_selector).parse_property(toks, super_property) } pub fn parse_value>( toks: &mut Peekable, scope: &Scope, super_selector: &Selector, ) -> Value { StyleParser::new(scope, super_selector).parse_style_value(toks) } pub fn from_tokens>( toks: &mut Peekable, scope: &Scope, super_selector: &Selector, super_property: String, ) -> Result { StyleParser::new(scope, super_selector).eat_style_group(toks, super_property) } } struct StyleParser<'a> { scope: &'a Scope, super_selector: &'a Selector, } impl<'a> StyleParser<'a> { const fn new(scope: &'a Scope, super_selector: &'a Selector) -> Self { StyleParser { scope, super_selector, } } pub(crate) fn parse_style_value>( &self, toks: &mut Peekable, ) -> Value { let mut style = Vec::new(); let mut n = 0; devour_whitespace(toks); while let Some(tok) = toks.peek() { match tok.kind { TokenKind::MultilineComment(_) => { toks.next(); continue; } TokenKind::Symbol(Symbol::OpenCurlyBrace) | TokenKind::Interpolation => n += 1, TokenKind::Symbol(Symbol::CloseCurlyBrace) => { if n == 0 { break; } else { // todo: toks.next() and push n -= 1; } } TokenKind::Symbol(Symbol::SemiColon) => { toks.next(); break; } TokenKind::Symbol(Symbol::BitAnd) => { style.push(Token { kind: TokenKind::Ident(self.super_selector.to_string()), pos: Pos::new(), }); toks.next(); continue; } _ => {} }; style.push(toks.next().unwrap()); } devour_whitespace(toks); Value::from_tokens(&mut style.into_iter().peekable(), self.scope).unwrap() } pub(crate) fn eat_style_group>( &self, toks: &mut Peekable, super_property: String, ) -> Result { let mut styles = Vec::new(); devour_whitespace(toks); while let Some(tok) = toks.peek() { match tok.kind { TokenKind::Symbol(Symbol::OpenCurlyBrace) => { toks.next(); devour_whitespace(toks); loop { let property = self.parse_property(toks, super_property.clone()); if let Some(tok) = toks.peek() { match tok.kind { TokenKind::Symbol(Symbol::OpenCurlyBrace) => { if let Expr::Styles(s) = self.eat_style_group(toks, property)? { styles.extend(s); } devour_whitespace(toks); if let Some(tok) = toks.peek() { match tok.kind { TokenKind::Symbol(Symbol::CloseCurlyBrace) => { toks.next(); devour_whitespace(toks); return Ok(Expr::Styles(styles)); } _ => continue, } } continue; } _ => {} } } let value = self.parse_style_value(toks); styles.push(Style { property, value }); if let Some(tok) = toks.peek() { match tok.kind { TokenKind::Symbol(Symbol::CloseCurlyBrace) => { toks.next(); devour_whitespace(toks); return Ok(Expr::Styles(styles)); } _ => continue, } } } } _ => { let val = self.parse_style_value(toks); return Ok(Expr::Style(Style { property: super_property, value: val, })); } } } Ok(Expr::Styles(styles)) } pub(crate) fn parse_property>( &self, toks: &mut Peekable, mut super_property: String, ) -> String { let mut property = String::new(); while let Some(Token { kind, .. }) = toks.next() { match kind { TokenKind::Whitespace(_) | TokenKind::MultilineComment(_) => continue, TokenKind::Ident(ref s) => property.push_str(s), TokenKind::Interpolation => property.push_str( &parse_interpolation(toks, self.scope) .iter() .map(|x| x.kind.to_string()) .collect::(), ), TokenKind::Symbol(Symbol::Colon) => break, TokenKind::Symbol(Symbol::BitAnd) => { property.push_str(&self.super_selector.to_string()) } _ => property.push_str(&kind.to_string()), }; } devour_whitespace(toks); if super_property.is_empty() { property } else { super_property.reserve(1 + property.len()); super_property.push('-'); super_property.push_str(&property); super_property } } }