use std::convert::TryFrom; use std::iter::{Iterator, Peekable}; use num_bigint::BigInt; use num_rational::BigRational; use num_traits::pow; use crate::args::eat_call_args; use crate::builtin::GLOBAL_FUNCTIONS; use crate::color::Color; use crate::common::{Keyword, ListSeparator, Op, QuoteKind, Scope, Symbol}; use crate::units::Unit; use crate::utils::{devour_whitespace_or_comment, parse_interpolation}; use crate::value::Value; use crate::{Token, TokenKind}; use super::number::Number; fn parse_hex(s: String) -> Value { match s.len() { 3 => { let v = u16::from_str_radix(&s, 16).unwrap(); let red = (((v & 0xf00) >> 8) * 0x11) as u8; let green = (((v & 0x0f0) >> 4) * 0x11) as u8; let blue = ((v & 0x00f) * 0x11) as u8; Value::Color(Color::new(red, green, blue, 1, format!("#{}", s))) } 4 => { let v = u16::from_str_radix(&s, 16).unwrap(); let red = (((v & 0xf000) >> 12) * 0x11) as u8; let green = (((v & 0x0f00) >> 8) * 0x11) as u8; let blue = (((v & 0x00f0) >> 4) * 0x11) as u8; let alpha = ((v & 0x000f) * 0x11) as u8; Value::Color(Color::new(red, green, blue, alpha, format!("#{}", s))) } 6 => { let v = u32::from_str_radix(&s, 16).unwrap(); let red = ((v & 0x00ff_0000) >> 16) as u8; let green = ((v & 0x0000_ff00) >> 8) as u8; let blue = (v & 0x0000_00ff) as u8; Value::Color(Color::new(red, green, blue, 1, format!("#{}", s))) } 8 => { let v = u32::from_str_radix(&s, 16).unwrap(); let red = ((v & 0xff00_0000) >> 24) as u8; let green = ((v & 0x00ff_0000) >> 16) as u8; let blue = ((v & 0x0000_ff00) >> 8) as u8; let alpha = (v & 0x0000_00ff) as u8; Value::Color(Color::new(red, green, blue, alpha, format!("#{}", s))) } _ => Value::Ident(s, QuoteKind::None), } } fn flatten_ident>(toks: &mut Peekable, scope: &Scope) -> String { let mut s = String::new(); while let Some(tok) = toks.peek() { match tok.kind.clone() { TokenKind::Interpolation => { toks.next(); s.push_str( &parse_interpolation(toks, scope) .iter() .map(|x| x.kind.to_string()) .collect::(), ) } TokenKind::Ident(ref i) => { toks.next(); s.push_str(i) } TokenKind::Number(ref n) => { toks.next(); s.push_str(n) } _ => break, } } s } impl Value { pub fn from_tokens>( toks: &mut Peekable, scope: &Scope, ) -> Option { let left = Self::_from_tokens(toks, scope)?; let whitespace = devour_whitespace_or_comment(toks); let next = match toks.peek() { Some(x) => x, None => return Some(left), }; match next.kind { TokenKind::Symbol(Symbol::SemiColon) | TokenKind::Symbol(Symbol::CloseParen) => { Some(left) } TokenKind::Symbol(Symbol::Comma) => { toks.next(); devour_whitespace_or_comment(toks); let right = match Self::from_tokens(toks, scope) { Some(x) => x, None => return Some(left), }; Some(Value::List(vec![left, right], ListSeparator::Comma)) } TokenKind::Symbol(Symbol::Plus) | TokenKind::Symbol(Symbol::Minus) | TokenKind::Op(_) | TokenKind::Symbol(Symbol::Mul) | TokenKind::Symbol(Symbol::Div) | TokenKind::Symbol(Symbol::Percent) => { let op = match next.kind { TokenKind::Symbol(Symbol::Plus) => Op::Plus, TokenKind::Symbol(Symbol::Minus) => Op::Minus, TokenKind::Symbol(Symbol::Mul) => Op::Mul, TokenKind::Symbol(Symbol::Div) => Op::Div, TokenKind::Symbol(Symbol::Percent) => Op::Rem, TokenKind::Op(op) => op, _ => unsafe { std::hint::unreachable_unchecked() }, }; toks.next(); devour_whitespace_or_comment(toks); let right = match Self::from_tokens(toks, scope) { Some(x) => x, None => return Some(left), }; Some(Value::BinaryOp(Box::new(left), op, Box::new(right))) } _ if whitespace => { devour_whitespace_or_comment(toks); let right = match Self::from_tokens(toks, scope) { Some(x) => x, None => return Some(left), }; Some(Value::List(vec![left, right], ListSeparator::Space)) } _ => { dbg!(&next.kind); todo!("unimplemented token in value") } } } fn _from_tokens>( toks: &mut Peekable, scope: &Scope, ) -> Option { let kind = if let Some(tok) = toks.next() { tok.kind } else { return None; }; match kind { TokenKind::Number(val) => { let unit = if let Some(tok) = toks.peek() { match tok.kind.clone() { TokenKind::Ident(i) => { toks.next(); Unit::from(&i) } TokenKind::Symbol(Symbol::Percent) => { toks.next(); Unit::Percent } _ => Unit::None, } } else { Unit::None }; let n = if let Ok(v) = val.parse::() { // the number is an integer! v // the number is floating point } else { let mut num = String::new(); let mut chars = val.chars(); let mut num_dec = 0; while let Some(c) = chars.next() { if c == '.' { break; } num.push(c); } for c in chars { num_dec += 1; num.push(c); } BigRational::new(num.parse().unwrap(), pow(BigInt::from(10), num_dec)) }; Some(Value::Dimension(Number::new(n), unit)) } TokenKind::Symbol(Symbol::OpenParen) => { devour_whitespace_or_comment(toks); let val = Self::from_tokens(toks, scope)?; assert_eq!( toks.next().unwrap().kind, TokenKind::Symbol(Symbol::CloseParen) ); Some(Value::Paren(Box::new(val))) } TokenKind::Symbol(Symbol::BitAnd) => { Some(Value::Ident(String::from("&"), QuoteKind::None)) } TokenKind::Symbol(Symbol::Hash) => Some(parse_hex(flatten_ident(toks, scope))), // TokenKind::Interpolation => { // Some(Value::Ident( // parse_interpolation(toks, scope) // .iter() // .map(|x| x.kind.to_string()) // .collect::(), // QuoteKind::None // )) // } TokenKind::Ident(mut s) => { s.push_str(&flatten_ident(toks, scope)); match toks.peek() { Some(Token { kind: TokenKind::Symbol(Symbol::OpenParen), .. }) => { toks.next(); let func = match scope.get_fn(&s) { Ok(f) => f, Err(_) => match GLOBAL_FUNCTIONS.get(&s) { Some(f) => return f(&eat_call_args(toks, scope), scope).ok(), None => { s.push('('); let mut unclosed_parens = 0; while let Some(t) = toks.next() { match &t.kind { TokenKind::Symbol(Symbol::OpenParen) => { unclosed_parens += 1; } TokenKind::Interpolation => s.push_str( &parse_interpolation(toks, scope) .iter() .map(|x| x.kind.to_string()) .collect::(), ), TokenKind::Variable(v) => s.push_str( &scope .get_var(v) .expect("expected variable to exist") .to_string(), ), TokenKind::Symbol(Symbol::CloseParen) => { if unclosed_parens <= 1 { s.push(')'); break; } else { unclosed_parens -= 1; } } _ => {} } s.push_str(&t.kind.to_string()); } return Some(Value::Ident(s, QuoteKind::None)); } }, }; Some(func.clone().args(&eat_call_args(toks, scope)).call()) } _ => { if let Ok(c) = crate::color::ColorName::try_from(s.as_ref()) { Some(Value::Color(c.into_color(s))) } else { Some(Value::Ident(s, QuoteKind::None)) } } } } TokenKind::Symbol(Symbol::DoubleQuote) => { let mut s = String::new(); let mut is_escaped = false; while let Some(tok) = toks.next() { match tok.kind { TokenKind::Symbol(Symbol::DoubleQuote) if !is_escaped => break, TokenKind::Symbol(Symbol::BackSlash) if !is_escaped => is_escaped = true, TokenKind::Symbol(Symbol::BackSlash) => s.push('\\'), _ => {} } if is_escaped && tok.kind != TokenKind::Symbol(Symbol::BackSlash) { is_escaped = false; } s.push_str(&tok.kind.to_string()); } Some(Value::Ident(s, QuoteKind::Double)) } TokenKind::Symbol(Symbol::SingleQuote) => { let mut s = String::new(); while let Some(tok) = toks.next() { if tok.kind == TokenKind::Symbol(Symbol::SingleQuote) { break; } s.push_str(&tok.kind.to_string()); } Some(Value::Ident(s, QuoteKind::Single)) } TokenKind::Variable(ref v) => { Some(scope.get_var(v).expect("expected variable").clone()) } TokenKind::Interpolation => { let mut s = parse_interpolation(toks, scope) .iter() .map(|x| x.kind.to_string()) .collect::(); while let Some(tok) = toks.peek() { match tok.kind.clone() { TokenKind::Interpolation => { toks.next(); s.push_str( &parse_interpolation(toks, scope) .iter() .map(|x| x.kind.to_string()) .collect::(), ) } TokenKind::Ident(ref i) => { toks.next(); s.push_str(i) } _ => break, } } Some(Value::Ident(s, QuoteKind::None)) } TokenKind::Keyword(Keyword::Important) => Some(Value::Important), TokenKind::Keyword(Keyword::True) => Some(Value::True), TokenKind::Keyword(Keyword::False) => Some(Value::False), TokenKind::Keyword(Keyword::Null) => Some(Value::Null), _ => None, } } }