Refactor function parsing into separate module
This commit is contained in:
parent
364480dc6f
commit
cee40858ee
140
src/function.rs
Normal file
140
src/function.rs
Normal 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)
|
||||||
|
}
|
98
src/main.rs
98
src/main.rs
@ -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,
|
||||||
|
52
src/mixin.rs
52
src/mixin.rs
@ -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()
|
|
||||||
}
|
|
||||||
}
|
|
Loading…
x
Reference in New Issue
Block a user