use peekmore::PeekMore; use codemap::{Span, Spanned}; use crate::{ color::Color, common::{Brackets, ListSeparator, Op, QuoteKind}, error::SassResult, parse::Parser, selector::Selector, unit::Unit, utils::hex_char_for, {Cow, Token}, }; use css_function::is_special_function; pub(crate) use map::SassMap; pub(crate) use number::Number; pub(crate) use sass_function::SassFunction; pub(crate) mod css_function; mod map; mod number; mod ops; mod sass_function; #[derive(Debug, Clone, PartialEq, Eq)] pub(crate) enum Value { Important, True, False, Null, Dimension(Number, Unit), List(Vec, ListSeparator, Brackets), Color(Box), UnaryOp(Op, Box), BinaryOp(Box, Op, Box), Paren(Box), String(String, QuoteKind), Map(SassMap), ArgList(Vec>), /// Returned by `get-function()` Function(SassFunction), } fn visit_quoted_string(buf: &mut String, force_double_quote: bool, string: &str) -> SassResult<()> { let mut has_single_quote = false; let mut has_double_quote = false; let mut buffer = String::new(); if force_double_quote { buffer.push('"'); } let mut iter = string.chars().peekable(); while let Some(c) = iter.next() { match c { '\'' => { if force_double_quote { buffer.push('\''); } else if has_double_quote { return visit_quoted_string(buf, true, string); } else { has_single_quote = true; buffer.push('\''); } } '"' => { if force_double_quote { buffer.push('\\'); buffer.push('"'); } else if has_single_quote { return visit_quoted_string(buf, true, string); } else { has_double_quote = true; buffer.push('"'); } } '\x00'..='\x08' | '\x0A'..='\x1F' => { buffer.push('\\'); if c as u32 > 0xF { buffer.push(hex_char_for(c as u32 >> 4)) } buffer.push(hex_char_for(c as u32 & 0xF)); let next = match iter.peek() { Some(v) => v, None => break, }; if next.is_ascii_hexdigit() || next == &' ' || next == &'\t' { buffer.push(' '); } } '\\' => { buffer.push('\\'); buffer.push('\\'); } _ => buffer.push(c), } } if force_double_quote { buffer.push('"'); } else { let quote = if has_double_quote { '\'' } else { '"' }; buffer = format!("{}{}{}", quote, buffer, quote); } buf.push_str(&buffer); Ok(()) } impl Value { pub fn is_null(&self, span: Span) -> SassResult { Ok(match self { Value::Null => true, Value::String(i, QuoteKind::None) if i.is_empty() => true, Self::BinaryOp(..) | Self::Paren(..) | Self::UnaryOp(..) => { self.clone().eval(span)?.is_null(span)? } Self::List(v, _, Brackets::Bracketed) if v.is_empty() => false, Self::List(v, ..) => v .iter() .map(|f| Ok(f.is_null(span)?)) .collect::>>()? .into_iter() .all(|f| f), _ => false, }) } pub fn to_css_string(&self, span: Span) -> SassResult> { Ok(match self { Self::Important => Cow::const_str("!important"), Self::Dimension(num, unit) => match unit { Unit::Mul(..) => { return Err((format!("{}{} isn't a valid CSS value.", num, unit), span).into()); } _ => Cow::owned(format!("{}{}", num, unit)), }, Self::Map(..) | Self::Function(..) => { return Err(( format!("{} isn't a valid CSS value.", self.inspect(span)?), span, ) .into()) } Self::List(vals, sep, brackets) => match brackets { Brackets::None => Cow::owned( vals.iter() .filter(|x| !x.is_null(span).unwrap_or(false)) .map(|x| x.to_css_string(span)) .collect::>>>()? .join(sep.as_str()), ), Brackets::Bracketed => Cow::owned(format!( "[{}]", vals.iter() .filter(|x| !x.is_null(span).unwrap_or(false)) .map(|x| x.to_css_string(span)) .collect::>>>()? .join(sep.as_str()), )), }, Self::Color(c) => Cow::owned(c.to_string()), Self::UnaryOp(..) | Self::BinaryOp(..) => { self.clone().eval(span)?.to_css_string(span)? } Self::Paren(val) => val.to_css_string(span)?, Self::String(string, QuoteKind::None) => { let mut after_newline = false; let mut buf = String::with_capacity(string.len()); for c in string.chars() { match c { '\n' => { buf.push(' '); after_newline = true; } ' ' => { if !after_newline { buf.push(' '); } } _ => { buf.push(c); after_newline = false; } } } Cow::owned(buf) } Self::String(string, QuoteKind::Quoted) => { let mut buf = String::with_capacity(string.len()); visit_quoted_string(&mut buf, false, string)?; Cow::owned(buf) } Self::True => Cow::const_str("true"), Self::False => Cow::const_str("false"), Self::Null => Cow::const_str(""), Self::ArgList(args) => Cow::owned( args.iter() .filter(|x| !x.is_null(span).unwrap_or(false)) .map(|a| Ok(a.node.to_css_string(span)?)) .collect::>>>()? .join(", "), ), }) } pub fn is_true(&self, span: Span) -> SassResult { match self { Value::Null | Value::False => Ok(false), Self::BinaryOp(..) | Self::Paren(..) | Self::UnaryOp(..) => { self.clone().eval(span)?.is_true(span) } _ => Ok(true), } } pub fn unquote(self) -> Self { match self { Self::String(s1, _) => Self::String(s1, QuoteKind::None), Self::List(v, sep, bracket) => { Self::List(v.into_iter().map(Value::unquote).collect(), sep, bracket) } v => v, } } pub const fn span(self, span: Span) -> Spanned { Spanned { node: self, span } } pub fn kind(&self, span: Span) -> SassResult<&'static str> { match self { Self::Color(..) => Ok("color"), Self::String(..) | Self::Important => Ok("string"), Self::Dimension(..) => Ok("number"), Self::List(..) => Ok("list"), Self::Function(..) => Ok("function"), Self::ArgList(..) => Ok("arglist"), Self::True | Self::False => Ok("bool"), Self::Null => Ok("null"), Self::Map(..) => Ok("map"), Self::BinaryOp(..) | Self::Paren(..) | Self::UnaryOp(..) => { self.clone().eval(span)?.kind(span) } } } pub fn is_special_function(&self) -> bool { match self { Self::String(s, QuoteKind::None) => is_special_function(s), _ => false, } } pub fn bool(b: bool) -> Self { if b { Value::True } else { Value::False } } // TODO: // https://github.com/sass/dart-sass/blob/d4adea7569832f10e3a26d0e420ae51640740cfb/lib/src/ast/sass/expression/list.dart#L39 pub fn inspect(&self, span: Span) -> SassResult> { Ok(match self { Value::List(v, _, brackets) if v.is_empty() => match brackets { Brackets::None => Cow::const_str("()"), Brackets::Bracketed => Cow::const_str("[]"), }, Value::List(v, sep, brackets) if v.len() == 1 => match brackets { Brackets::None => match sep { ListSeparator::Space => v[0].inspect(span)?, ListSeparator::Comma => Cow::owned(format!("({},)", v[0].inspect(span)?)), }, Brackets::Bracketed => match sep { ListSeparator::Space => Cow::owned(format!("[{}]", v[0].inspect(span)?)), ListSeparator::Comma => Cow::owned(format!("[{},]", v[0].inspect(span)?)), }, }, Self::List(vals, sep, brackets) => Cow::owned(match brackets { Brackets::None => vals .iter() .map(|x| x.inspect(span)) .collect::>>>()? .join(sep.as_str()), Brackets::Bracketed => format!( "[{}]", vals.iter() .map(|x| x.inspect(span)) .collect::>>>()? .join(sep.as_str()), ), }), Value::Function(f) => Cow::owned(format!("get-function(\"{}\")", f.name())), Value::Null => Cow::const_str("null"), Value::Map(map) => Cow::owned(format!( "({})", map.iter() .map(|(k, v)| Ok(format!( "{}: {}", k.to_css_string(span)?, v.to_css_string(span)? ))) .collect::>>()? .join(", ") )), Value::Paren(v) => v.inspect(span)?, v => v.to_css_string(span)?, }) } pub fn as_list(self) -> Vec { match self { Value::List(v, ..) => v, Value::Map(m) => m.entries(), Value::ArgList(v) => v.into_iter().map(|val| val.node).collect(), v => vec![v], } } /// Parses `self` as a selector list, in the same manner as the /// `selector-parse()` function. /// /// Returns a `SassError` if `self` isn't a type that can be parsed as a /// selector, or if parsing fails. If `allow_parent` is `true`, this allows /// parent selectors. Otherwise, they're considered parse errors. /// /// `name` is the argument name. It's used for error reporting. pub fn to_selector( self, parser: &mut Parser<'_>, name: &str, allows_parent: bool, ) -> SassResult { let string = match self.clone().selector_string(parser.span_before)? { Some(v) => v, None => return Err((format!("${}: {} is not a valid selector: it must be a string, a list of strings, or a list of lists of strings.", name, self.inspect(parser.span_before)?), parser.span_before).into()), }; Parser { toks: &mut string .chars() .map(|c| Token::new(parser.span_before, c)) .collect::>() .into_iter() .peekmore(), map: parser.map, path: parser.path, scopes: parser.scopes, global_scope: parser.global_scope, super_selectors: parser.super_selectors, span_before: parser.span_before, content: parser.content, in_mixin: parser.in_mixin, in_function: parser.in_function, in_control_flow: parser.in_control_flow, at_root: parser.at_root, at_root_has_selector: parser.at_root_has_selector, extender: parser.extender, } .parse_selector(allows_parent, true, String::new()) } fn selector_string(self, span: Span) -> SassResult> { Ok(Some(match self.eval(span)?.node { Self::String(text, ..) => text, Self::List(list, sep, ..) if !list.is_empty() => { let mut result = Vec::new(); match sep { ListSeparator::Comma => { for complex in list { if let Value::String(text, ..) = complex { result.push(text); } else if let Value::List(_, ListSeparator::Space, ..) = complex { result.push(match complex.selector_string(span)? { Some(v) => v, None => return Ok(None), }); } else { return Ok(None); } } } ListSeparator::Space => { for compound in list { if let Value::String(text, ..) = compound { result.push(text); } else { return Ok(None); } } } } result.join(sep.as_str()) } _ => return Ok(None), })) } }