use crate::common::{Pos, Symbol, Whitespace}; use crate::{Scope, Token, TokenKind}; use std::iter::{Iterator, Peekable}; pub(crate) trait IsWhitespace { fn is_whitespace(&self) -> bool; } pub(crate) fn devour_whitespace, W: IsWhitespace>( s: &mut Peekable, ) -> bool { let mut found_whitespace = false; while let Some(w) = s.peek() { if !w.is_whitespace() { break; } found_whitespace = true; s.next(); } found_whitespace } pub(crate) trait IsComment { fn is_comment(&self) -> bool; } pub(crate) fn devour_whitespace_or_comment, W: IsWhitespace + IsComment>( s: &mut Peekable, ) -> bool { let mut found_whitespace = false; while let Some(w) = s.peek() { if !w.is_whitespace() && !w.is_comment() { break; } found_whitespace = true; s.next(); } found_whitespace } #[cfg_attr(feature = "nightly", track_caller)] pub(crate) fn deref_variable(name: &str, scope: &Scope) -> Vec { let mut toks = scope .vars .get(name) .expect("todo! expected variable to exist") .iter() .peekable(); let mut val = Vec::with_capacity(toks.len()); while let Some(tok) = toks.next() { match tok.kind { TokenKind::Variable(ref v) => val.extend(deref_variable(v, scope)), TokenKind::Whitespace(_) => { devour_whitespace(&mut toks); if toks.peek().is_some() { val.push(Token { kind: TokenKind::Whitespace(Whitespace::Space), pos: tok.pos, }); } } _ => val.push(tok.clone()), } } val } pub(crate) fn eat_interpolation>( tokens: &mut I, scope: &Scope, ) -> Vec { let mut val = Vec::new(); while let Some(tok) = tokens.next() { match tok.kind { TokenKind::Symbol(Symbol::CloseCurlyBrace) => break, TokenKind::Symbol(Symbol::OpenCurlyBrace) => { todo!("invalid character in interpolation") } TokenKind::Variable(ref v) => val.extend(deref_variable(v, scope)), TokenKind::Interpolation => val.extend(eat_interpolation(tokens, scope)), _ => val.push(tok), } } val } pub(crate) fn eat_variable_value>( toks: &mut Peekable, scope: &Scope, ) -> Result, (Pos, String)> { devour_whitespace(toks); // todo!(line might not end with semicolon) let iter1 = toks.take_while(|x| x.kind != TokenKind::Symbol(Symbol::SemiColon)); let mut iter2 = Vec::new(); for tok in iter1 { match tok.kind { TokenKind::Variable(ref name) => iter2.extend(deref_variable(name, scope)), _ => iter2.push(tok), }; } devour_whitespace(toks); Ok(iter2) }