Refactor function parsing into separate module

This commit is contained in:
ConnorSkees 2020-01-17 16:23:21 -05:00
parent 364480dc6f
commit cee40858ee
3 changed files with 154 additions and 136 deletions

140
src/function.rs Normal file
View File

@ -0,0 +1,140 @@
use std::iter::Peekable;
use crate::common::Symbol;
use crate::utils::devour_whitespace;
use crate::{Token, TokenKind};
#[derive(Debug, Clone, Eq, PartialEq)]
pub struct FuncArgs(pub Vec<FuncArg>);
#[derive(Debug, Clone, Eq, PartialEq)]
pub struct FuncArg {
pub name: String,
pub default: Option<Vec<Token>>,
}
impl FuncArgs {
pub const fn new() -> Self {
FuncArgs(Vec::new())
}
}
#[derive(Debug, Clone)]
pub struct CallArgs(pub Vec<CallArg>);
#[derive(Debug, Clone)]
pub struct CallArg {
pub name: Option<String>,
pub val: Vec<Token>,
}
impl CallArg {
pub fn is_named(&self) -> bool {
self.name.is_some()
}
}
impl CallArgs {
pub const fn new() -> Self {
CallArgs(Vec::new())
}
pub fn is_empty(&self) -> bool {
self.0.is_empty()
}
}
pub fn eat_func_args<I: Iterator<Item = Token>>(toks: &mut Peekable<I>) -> FuncArgs {
let mut args: Vec<FuncArg> = Vec::new();
devour_whitespace(toks);
while let Some(Token { kind, .. }) = toks.next() {
let name = match kind {
TokenKind::Variable(v) => v,
TokenKind::Symbol(Symbol::CloseParen) => break,
_ => todo!(),
};
devour_whitespace(toks);
let kind = if let Some(Token { kind, .. }) = toks.next() {
kind
} else {
todo!()
};
match kind {
TokenKind::Symbol(Symbol::Colon) => {
todo!("handle default values")
// let mut val: Vec<Token> = Vec::new();
// while let Some(tok) = toks.next() {
// match &kind {
// _ => val.push(tok),
// }
// }
}
TokenKind::Symbol(Symbol::Period) => todo!("handle varargs"),
TokenKind::Symbol(Symbol::CloseParen) => {
args.push(FuncArg {
name,
default: None,
});
break;
}
TokenKind::Symbol(Symbol::Comma) => args.push(FuncArg {
name,
default: None,
}),
_ => {}
}
devour_whitespace(toks);
}
devour_whitespace(toks);
if let Some(Token {
kind: TokenKind::Symbol(Symbol::OpenCurlyBrace),
..
}) = toks.next()
{
} else {
todo!("expected `{{` after mixin args")
}
FuncArgs(args)
}
pub fn eat_call_args<I: Iterator<Item = Token>>(toks: &mut Peekable<I>) -> CallArgs {
let mut args: Vec<CallArg> = Vec::new();
devour_whitespace(toks);
let mut name: Option<String> = None;
let mut val = Vec::new();
while let Some(Token { kind, pos }) = toks.next() {
match kind {
TokenKind::Variable(v) => name = Some(v),
TokenKind::Symbol(Symbol::Colon) => {
todo!("handle default values")
// let mut val: Vec<Token> = Vec::new();
// while let Some(Token { kind, .. }) = toks.next() {
// match &kind {
// _ => {}
// }
// }
}
TokenKind::Symbol(Symbol::CloseParen) => {
args.push(CallArg {
name: name.clone(),
val: val.clone(),
});
break;
}
TokenKind::Symbol(Symbol::Comma) => {
args.push(CallArg {
name: name.clone(),
val: val.clone(),
});
if let Some(ref mut s) = name {
s.clear();
}
val.clear();
}
_ => val.push(Token { kind, pos }),
}
devour_whitespace(toks);
}
CallArgs(args)
}

View File

@ -35,8 +35,9 @@ use crate::common::{AtRule, Keyword, Op, Pos, Printer, Scope, Symbol, Whitespace
use crate::css::Css; use crate::css::Css;
use crate::error::SassError; use crate::error::SassError;
use crate::format::PrettyPrinter; use crate::format::PrettyPrinter;
use crate::function::{eat_call_args, eat_func_args, CallArgs, FuncArgs};
use crate::lexer::Lexer; use crate::lexer::Lexer;
use crate::mixin::{CallArg, CallArgs, FuncArg, FuncArgs, Mixin}; use crate::mixin::Mixin;
use crate::selector::{Attribute, Selector}; use crate::selector::{Attribute, Selector};
use crate::style::Style; use crate::style::Style;
use crate::units::Unit; use crate::units::Unit;
@ -47,6 +48,7 @@ mod common;
mod css; mod css;
mod error; mod error;
mod format; mod format;
mod function;
mod imports; mod imports;
mod lexer; mod lexer;
mod mixin; mod mixin;
@ -379,100 +381,12 @@ fn eat_include<I: Iterator<Item = Token>>(
} else { } else {
return Err((pos, "expected identifier")); return Err((pos, "expected identifier"));
}; };
let rules = mixin.call_with_args(&args).eval(super_selector, &mut scope.clone()); let rules = mixin
.call_with_args(&args)
.eval(super_selector, &mut scope.clone());
Ok(rules) Ok(rules)
} }
fn eat_func_args<I: Iterator<Item = Token>>(toks: &mut Peekable<I>) -> FuncArgs {
let mut args: Vec<FuncArg> = Vec::new();
devour_whitespace(toks);
while let Some(Token { kind, .. }) = toks.next() {
let name = match kind {
TokenKind::Variable(v) => v,
TokenKind::Symbol(Symbol::CloseParen) => break,
_ => todo!(),
};
devour_whitespace(toks);
let kind = if let Some(Token { kind, .. }) = toks.next() {
kind
} else {
todo!()
};
match kind {
TokenKind::Symbol(Symbol::Colon) => {
todo!("handle default values")
// let mut val: Vec<Token> = Vec::new();
// while let Some(tok) = toks.next() {
// match &kind {
// _ => val.push(tok),
// }
// }
}
TokenKind::Symbol(Symbol::Period) => todo!("handle varargs"),
TokenKind::Symbol(Symbol::CloseParen) => {
args.push(FuncArg {
name,
default: None,
});
break;
}
TokenKind::Symbol(Symbol::Comma) => args.push(FuncArg {
name,
default: None,
}),
_ => {}
}
devour_whitespace(toks);
}
devour_whitespace(toks);
if let Some(Token { kind: TokenKind::Symbol(Symbol::OpenCurlyBrace), .. }) = toks.next() {} else {
todo!("expected `{{` after mixin args")
}
FuncArgs(args)
}
fn eat_call_args<I: Iterator<Item = Token>>(toks: &mut Peekable<I>) -> CallArgs {
let mut args: Vec<CallArg> = Vec::new();
devour_whitespace(toks);
let mut name: Option<String> = None;
let mut val = Vec::new();
while let Some(Token { kind, pos }) = toks.next() {
match kind {
TokenKind::Variable(v) => name = Some(v),
TokenKind::Symbol(Symbol::Colon) => {
todo!("handle default values")
// let mut val: Vec<Token> = Vec::new();
// while let Some(Token { kind, .. }) = toks.next() {
// match &kind {
// _ => {}
// }
// }
}
TokenKind::Symbol(Symbol::CloseParen) => {
args.push(CallArg {
name: name.clone(),
val: val.clone(),
});
break;
},
TokenKind::Symbol(Symbol::Comma) => {
args.push(CallArg {
name: name.clone(),
val: val.clone(),
});
if let Some(ref mut s) = name {
s.clear();
}
val.clear();
}
_ => val.push(Token { kind, pos }),
}
devour_whitespace(toks);
}
CallArgs(args)
}
fn parse_mixin<I: Iterator<Item = Token>>( fn parse_mixin<I: Iterator<Item = Token>>(
toks: &mut Peekable<I>, toks: &mut Peekable<I>,
scope: Scope, scope: Scope,

View File

@ -1,9 +1,11 @@
use crate::common::Scope;
use crate::selector::Selector;
use crate::{eat_expr, Expr, RuleSet, Stmt, Token};
use std::iter::Peekable; use std::iter::Peekable;
use std::vec::IntoIter; use std::vec::IntoIter;
use crate::common::Scope;
use crate::function::{CallArgs, FuncArgs};
use crate::selector::Selector;
use crate::{eat_expr, Expr, RuleSet, Stmt, Token};
#[derive(Debug, Clone)] #[derive(Debug, Clone)]
pub struct Mixin { pub struct Mixin {
scope: Scope, scope: Scope,
@ -29,7 +31,9 @@ impl Mixin {
todo!("keyword args") todo!("keyword args")
} else { } else {
// dbg!(&self.args.0[idx].name.clone()); // dbg!(&self.args.0[idx].name.clone());
self.scope.vars.insert(self.args.0[idx].name.clone(), arg.val.clone()); self.scope
.vars
.insert(self.args.0[idx].name.clone(), arg.val.clone());
} }
} }
self self
@ -67,43 +71,3 @@ impl Mixin {
stmts stmts
} }
} }
#[derive(Debug, Clone, Eq, PartialEq)]
pub struct FuncArgs(pub Vec<FuncArg>);
#[derive(Debug, Clone, Eq, PartialEq)]
pub struct FuncArg {
pub name: String,
pub default: Option<Vec<Token>>,
}
impl FuncArgs {
pub const fn new() -> Self {
FuncArgs(Vec::new())
}
}
#[derive(Debug, Clone)]
pub struct CallArgs(pub Vec<CallArg>);
#[derive(Debug, Clone)]
pub struct CallArg {
pub name: Option<String>,
pub val: Vec<Token>,
}
impl CallArg {
pub fn is_named(&self) -> bool {
self.name.is_some()
}
}
impl CallArgs {
pub const fn new() -> Self {
CallArgs(Vec::new())
}
pub fn is_empty(&self) -> bool {
self.0.is_empty()
}
}