args and default values are lazily evaluated
This commit is contained in:
parent
ece0f5afde
commit
b7be1705a2
95
src/args.rs
95
src/args.rs
@ -18,7 +18,7 @@ pub(crate) struct FuncArgs(pub Vec<FuncArg>);
|
||||
#[derive(Debug, Clone, Eq, PartialEq)]
|
||||
pub(crate) struct FuncArg {
|
||||
pub name: String,
|
||||
pub default: Option<Value>,
|
||||
pub default: Option<Vec<Token>>,
|
||||
pub is_variadic: bool,
|
||||
}
|
||||
|
||||
@ -29,7 +29,7 @@ impl FuncArgs {
|
||||
}
|
||||
|
||||
#[derive(Debug, Clone)]
|
||||
pub(crate) struct CallArgs(HashMap<CallArg, Value>);
|
||||
pub(crate) struct CallArgs(HashMap<CallArg, Vec<Token>>);
|
||||
|
||||
#[derive(Debug, Clone, Hash, Eq, PartialEq)]
|
||||
enum CallArg {
|
||||
@ -58,25 +58,58 @@ impl CallArgs {
|
||||
CallArgs(HashMap::new())
|
||||
}
|
||||
|
||||
#[allow(dead_code)]
|
||||
pub fn get_named(&self, val: String) -> Option<&Value> {
|
||||
self.0.get(&CallArg::Named(val))
|
||||
/// Get argument by name
|
||||
///
|
||||
/// Removes the argument
|
||||
pub fn get_named(
|
||||
&mut self,
|
||||
val: String,
|
||||
scope: &Scope,
|
||||
super_selector: &Selector,
|
||||
) -> Option<SassResult<Value>> {
|
||||
match self.0.remove(&CallArg::Named(val)) {
|
||||
Some(v) => Some(Value::from_tokens(
|
||||
&mut v.into_iter().peekable(),
|
||||
scope,
|
||||
super_selector,
|
||||
)),
|
||||
None => None,
|
||||
}
|
||||
}
|
||||
|
||||
pub fn get_positional(&self, val: usize) -> Option<&Value> {
|
||||
self.0.get(&CallArg::Positional(val))
|
||||
/// Get a positional argument by 0-indexed position
|
||||
///
|
||||
/// Removes the argument
|
||||
pub fn get_positional(
|
||||
&mut self,
|
||||
val: usize,
|
||||
scope: &Scope,
|
||||
super_selector: &Selector,
|
||||
) -> Option<SassResult<Value>> {
|
||||
match self.0.remove(&CallArg::Positional(val)) {
|
||||
Some(v) => Some(Value::from_tokens(
|
||||
&mut v.into_iter().peekable(),
|
||||
scope,
|
||||
super_selector,
|
||||
)),
|
||||
None => None,
|
||||
}
|
||||
}
|
||||
|
||||
pub fn get_variadic(self) -> SassResult<Vec<Value>> {
|
||||
pub fn get_variadic(self, scope: &Scope, super_selector: &Selector) -> SassResult<Vec<Value>> {
|
||||
let mut vals = Vec::new();
|
||||
let mut args = self
|
||||
.0
|
||||
.into_iter()
|
||||
.map(|(a, v)| Ok((a.position()?, v)))
|
||||
.collect::<SassResult<Vec<(usize, Value)>>>()?;
|
||||
.collect::<SassResult<Vec<(usize, Vec<Token>)>>>()?;
|
||||
args.sort_by(|(a1, _), (a2, _)| a1.cmp(a2));
|
||||
for arg in args {
|
||||
vals.push(arg.1);
|
||||
vals.push(Value::from_tokens(
|
||||
&mut arg.1.into_iter().peekable(),
|
||||
scope,
|
||||
super_selector,
|
||||
)?);
|
||||
}
|
||||
Ok(vals)
|
||||
}
|
||||
@ -97,14 +130,6 @@ impl CallArgs {
|
||||
pub fn is_empty(&self) -> bool {
|
||||
self.0.len() == 0
|
||||
}
|
||||
|
||||
pub fn remove_named(&mut self, s: String) -> Option<Value> {
|
||||
self.0.remove(&CallArg::Named(s))
|
||||
}
|
||||
|
||||
pub fn remove_positional(&mut self, s: usize) -> Option<Value> {
|
||||
self.0.remove(&CallArg::Positional(s))
|
||||
}
|
||||
}
|
||||
|
||||
pub(crate) fn eat_func_args<I: Iterator<Item = Token>>(
|
||||
@ -137,11 +162,7 @@ pub(crate) fn eat_func_args<I: Iterator<Item = Token>>(
|
||||
toks.next();
|
||||
args.push(FuncArg {
|
||||
name: name.replace('_', "-"),
|
||||
default: Some(Value::from_tokens(
|
||||
&mut default.into_iter().peekable(),
|
||||
scope,
|
||||
super_selector,
|
||||
)?),
|
||||
default: Some(default),
|
||||
is_variadic,
|
||||
});
|
||||
break;
|
||||
@ -149,11 +170,7 @@ pub(crate) fn eat_func_args<I: Iterator<Item = Token>>(
|
||||
')' => {
|
||||
args.push(FuncArg {
|
||||
name: name.replace('_', "-"),
|
||||
default: Some(Value::from_tokens(
|
||||
&mut default.into_iter().peekable(),
|
||||
scope,
|
||||
super_selector,
|
||||
)?),
|
||||
default: Some(default),
|
||||
is_variadic,
|
||||
});
|
||||
break;
|
||||
@ -181,11 +198,7 @@ pub(crate) fn eat_func_args<I: Iterator<Item = Token>>(
|
||||
|
||||
args.push(FuncArg {
|
||||
name: name.replace('_', "-"),
|
||||
default: Some(Value::from_tokens(
|
||||
&mut default.into_iter().peekable(),
|
||||
scope,
|
||||
super_selector,
|
||||
)?),
|
||||
default: Some(default),
|
||||
is_variadic,
|
||||
});
|
||||
break;
|
||||
@ -196,11 +209,7 @@ pub(crate) fn eat_func_args<I: Iterator<Item = Token>>(
|
||||
default: if default.is_empty() {
|
||||
None
|
||||
} else {
|
||||
Some(Value::from_tokens(
|
||||
&mut default.into_iter().peekable(),
|
||||
scope,
|
||||
super_selector,
|
||||
)?)
|
||||
Some(default)
|
||||
},
|
||||
is_variadic,
|
||||
});
|
||||
@ -228,7 +237,7 @@ pub(crate) fn eat_call_args<I: Iterator<Item = Token>>(
|
||||
scope: &Scope,
|
||||
super_selector: &Selector,
|
||||
) -> SassResult<CallArgs> {
|
||||
let mut args: HashMap<CallArg, Value> = HashMap::new();
|
||||
let mut args: HashMap<CallArg, Vec<Token>> = HashMap::new();
|
||||
devour_whitespace_or_comment(toks)?;
|
||||
let mut name = String::new();
|
||||
let mut val: Vec<Token> = Vec::new();
|
||||
@ -264,7 +273,7 @@ pub(crate) fn eat_call_args<I: Iterator<Item = Token>>(
|
||||
} else {
|
||||
CallArg::Named(name.replace('_', "-"))
|
||||
},
|
||||
Value::from_tokens(&mut val.into_iter().peekable(), scope, super_selector)?,
|
||||
val,
|
||||
);
|
||||
return Ok(CallArgs(args));
|
||||
}
|
||||
@ -291,11 +300,7 @@ pub(crate) fn eat_call_args<I: Iterator<Item = Token>>(
|
||||
} else {
|
||||
CallArg::Named(name.replace('_', "-"))
|
||||
},
|
||||
Value::from_tokens(
|
||||
&mut val.clone().into_iter().peekable(),
|
||||
scope,
|
||||
super_selector,
|
||||
)?,
|
||||
val.clone(),
|
||||
);
|
||||
val.clear();
|
||||
devour_whitespace(toks);
|
||||
|
@ -59,19 +59,30 @@ impl Function {
|
||||
Ok((name, Function::new(scope, args, body, pos)))
|
||||
}
|
||||
|
||||
pub fn args(mut self, mut args: CallArgs) -> SassResult<Function> {
|
||||
pub fn args(
|
||||
mut self,
|
||||
mut args: CallArgs,
|
||||
scope: &Scope,
|
||||
super_selector: &Selector,
|
||||
) -> SassResult<Function> {
|
||||
for (idx, arg) in self.args.0.iter().enumerate() {
|
||||
if arg.is_variadic {
|
||||
self.scope
|
||||
.insert_var(&arg.name, Value::ArgList(args.get_variadic()?))?;
|
||||
self.scope.insert_var(
|
||||
&arg.name,
|
||||
Value::ArgList(args.get_variadic(scope, super_selector)?),
|
||||
)?;
|
||||
break;
|
||||
}
|
||||
let val = match args.remove_positional(idx) {
|
||||
Some(v) => v,
|
||||
None => match args.remove_named(arg.name.clone()) {
|
||||
Some(v) => v,
|
||||
let val = match args.get_positional(idx, scope, super_selector) {
|
||||
Some(v) => v?,
|
||||
None => match args.get_named(arg.name.clone(), scope, super_selector) {
|
||||
Some(v) => v?,
|
||||
None => match &arg.default {
|
||||
Some(v) => v.clone(),
|
||||
Some(v) => Value::from_tokens(
|
||||
&mut v.iter().cloned().peekable(),
|
||||
scope,
|
||||
super_selector,
|
||||
)?,
|
||||
None => return Err(format!("Missing argument ${}.", &arg.name).into()),
|
||||
},
|
||||
},
|
||||
|
@ -60,19 +60,30 @@ impl Mixin {
|
||||
self
|
||||
}
|
||||
|
||||
pub fn args(mut self, mut args: CallArgs) -> SassResult<Mixin> {
|
||||
pub fn args(
|
||||
mut self,
|
||||
mut args: CallArgs,
|
||||
scope: &Scope,
|
||||
super_selector: &Selector,
|
||||
) -> SassResult<Mixin> {
|
||||
for (idx, arg) in self.args.0.iter().enumerate() {
|
||||
if arg.is_variadic {
|
||||
self.scope
|
||||
.insert_var(&arg.name, Value::ArgList(args.get_variadic()?))?;
|
||||
self.scope.insert_var(
|
||||
&arg.name,
|
||||
Value::ArgList(args.get_variadic(scope, super_selector)?),
|
||||
)?;
|
||||
break;
|
||||
}
|
||||
let val = match args.remove_positional(idx) {
|
||||
Some(v) => v,
|
||||
None => match args.remove_named(arg.name.clone()) {
|
||||
Some(v) => v,
|
||||
let val = match args.get_positional(idx, scope, super_selector) {
|
||||
Some(v) => v?,
|
||||
None => match args.get_named(arg.name.clone(), scope, super_selector) {
|
||||
Some(v) => v?,
|
||||
None => match &arg.default {
|
||||
Some(v) => v.clone(),
|
||||
Some(v) => Value::from_tokens(
|
||||
&mut v.iter().cloned().peekable(),
|
||||
scope,
|
||||
super_selector,
|
||||
)?,
|
||||
None => return Err(format!("Missing argument ${}.", &arg.name).into()),
|
||||
},
|
||||
},
|
||||
@ -176,6 +187,9 @@ pub(crate) fn eat_include<I: Iterator<Item = Token>>(
|
||||
|
||||
let mixin = scope.get_mixin(&name)?.clone();
|
||||
|
||||
let rules = mixin.args(args)?.content(content).call(super_selector)?;
|
||||
let rules = mixin
|
||||
.args(args, scope, super_selector)?
|
||||
.content(content)
|
||||
.call(super_selector)?;
|
||||
Ok(rules)
|
||||
}
|
||||
|
@ -11,13 +11,13 @@ use crate::value::{Number, Value};
|
||||
pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
f.insert(
|
||||
"hsl".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
if args.is_empty() {
|
||||
return Err("Missing argument $channels.".into());
|
||||
}
|
||||
|
||||
if args.len() == 1 {
|
||||
let mut channels = match arg!(args, 0, "channels") {
|
||||
let mut channels = match arg!(args, scope, super_selector, 0, "channels") {
|
||||
Value::List(v, ..) => v,
|
||||
_ => return Err("Missing argument $channels.".into()),
|
||||
};
|
||||
@ -55,20 +55,22 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
Number::one(),
|
||||
)))
|
||||
} else {
|
||||
let hue = match arg!(args, 0, "hue") {
|
||||
let hue = match arg!(args, scope, super_selector, 0, "hue") {
|
||||
Value::Dimension(n, _) => n,
|
||||
v => return Err(format!("$hue: {} is not a number.", v).into()),
|
||||
};
|
||||
let saturation = match arg!(args, 1, "saturation") {
|
||||
let saturation = match arg!(args, scope, super_selector, 1, "saturation") {
|
||||
Value::Dimension(n, _) => n / Number::from(100),
|
||||
v => return Err(format!("$saturation: {} is not a number.", v).into()),
|
||||
};
|
||||
let lightness = match arg!(args, 2, "lightness") {
|
||||
let lightness = match arg!(args, scope, super_selector, 2, "lightness") {
|
||||
Value::Dimension(n, _) => n / Number::from(100),
|
||||
v => return Err(format!("$lightness: {} is not a number.", v).into()),
|
||||
};
|
||||
let alpha = match arg!(
|
||||
args,
|
||||
scope,
|
||||
super_selector,
|
||||
3,
|
||||
"alpha" = Value::Dimension(Number::one(), Unit::None)
|
||||
) {
|
||||
@ -89,13 +91,13 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"hsla".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
if args.is_empty() {
|
||||
return Err("Missing argument $channels.".into());
|
||||
}
|
||||
|
||||
if args.len() == 1 {
|
||||
let mut channels = match arg!(args, 0, "channels") {
|
||||
let mut channels = match arg!(args, scope, super_selector, 0, "channels") {
|
||||
Value::List(v, ..) => v,
|
||||
_ => return Err("Missing argument $channels.".into()),
|
||||
};
|
||||
@ -133,20 +135,22 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
Number::one(),
|
||||
)))
|
||||
} else {
|
||||
let hue = match arg!(args, 0, "hue") {
|
||||
let hue = match arg!(args, scope, super_selector, 0, "hue") {
|
||||
Value::Dimension(n, _) => n,
|
||||
v => return Err(format!("$hue: {} is not a number.", v).into()),
|
||||
};
|
||||
let saturation = match arg!(args, 1, "saturation") {
|
||||
let saturation = match arg!(args, scope, super_selector, 1, "saturation") {
|
||||
Value::Dimension(n, _) => n / Number::from(100),
|
||||
v => return Err(format!("$saturation: {} is not a number.", v).into()),
|
||||
};
|
||||
let lightness = match arg!(args, 2, "lightness") {
|
||||
let lightness = match arg!(args, scope, super_selector, 2, "lightness") {
|
||||
Value::Dimension(n, _) => n / Number::from(100),
|
||||
v => return Err(format!("$lightness: {} is not a number.", v).into()),
|
||||
};
|
||||
let alpha = match arg!(
|
||||
args,
|
||||
scope,
|
||||
super_selector,
|
||||
3,
|
||||
"alpha" = Value::Dimension(Number::one(), Unit::None)
|
||||
) {
|
||||
@ -167,9 +171,9 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"hue".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "color") {
|
||||
match arg!(args, scope, super_selector, 0, "color") {
|
||||
Value::Color(c) => Ok(Value::Dimension(c.hue(), Unit::Deg)),
|
||||
v => Err(format!("$color: {} is not a color.", v).into()),
|
||||
}
|
||||
@ -177,9 +181,9 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"saturation".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "color") {
|
||||
match arg!(args, scope, super_selector, 0, "color") {
|
||||
Value::Color(c) => Ok(Value::Dimension(c.saturation(), Unit::Percent)),
|
||||
v => Err(format!("$color: {} is not a color.", v).into()),
|
||||
}
|
||||
@ -187,9 +191,9 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"lightness".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "color") {
|
||||
match arg!(args, scope, super_selector, 0, "color") {
|
||||
Value::Color(c) => Ok(Value::Dimension(c.lightness(), Unit::Percent)),
|
||||
v => Err(format!("$color: {} is not a color.", v).into()),
|
||||
}
|
||||
@ -197,13 +201,13 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"adjust-hue".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 2);
|
||||
let color = match arg!(args, 0, "color") {
|
||||
let color = match arg!(args, scope, super_selector, 0, "color") {
|
||||
Value::Color(c) => c,
|
||||
v => return Err(format!("$color: {} is not a color.", v).into()),
|
||||
};
|
||||
let degrees = match arg!(args, 1, "degrees") {
|
||||
let degrees = match arg!(args, scope, super_selector, 1, "degrees") {
|
||||
Value::Dimension(n, _) => n,
|
||||
v => return Err(format!("$degrees: {} is not a number.", v).into()),
|
||||
};
|
||||
@ -212,13 +216,13 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"lighten".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 2);
|
||||
let color = match arg!(args, 0, "color") {
|
||||
let color = match arg!(args, scope, super_selector, 0, "color") {
|
||||
Value::Color(c) => c,
|
||||
v => return Err(format!("$color: {} is not a color.", v).into()),
|
||||
};
|
||||
let amount = match arg!(args, 1, "amount") {
|
||||
let amount = match arg!(args, scope, super_selector, 1, "amount") {
|
||||
Value::Dimension(n, u) => bound!("amount", n, u, 0, 100) / Number::from(100),
|
||||
v => return Err(format!("$amount: {} is not a number.", v).into()),
|
||||
};
|
||||
@ -227,13 +231,13 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"darken".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 2);
|
||||
let color = match arg!(args, 0, "color") {
|
||||
let color = match arg!(args, scope, super_selector, 0, "color") {
|
||||
Value::Color(c) => c,
|
||||
v => return Err(format!("$color: {} is not a color.", v).into()),
|
||||
};
|
||||
let amount = match arg!(args, 1, "amount") {
|
||||
let amount = match arg!(args, scope, super_selector, 1, "amount") {
|
||||
Value::Dimension(n, u) => bound!("amount", n, u, 0, 100) / Number::from(100),
|
||||
v => return Err(format!("$amount: {} is not a number.", v).into()),
|
||||
};
|
||||
@ -242,20 +246,23 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"saturate".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 2);
|
||||
if args.len() == 1 {
|
||||
return Ok(Value::Ident(
|
||||
format!("saturate({})", arg!(args, 0, "amount")),
|
||||
format!(
|
||||
"saturate({})",
|
||||
arg!(args, scope, super_selector, 0, "amount")
|
||||
),
|
||||
QuoteKind::None,
|
||||
));
|
||||
}
|
||||
|
||||
let amount = match arg!(args, 1, "amount") {
|
||||
let amount = match arg!(args, scope, super_selector, 1, "amount") {
|
||||
Value::Dimension(n, u) => bound!("amount", n, u, 0, 100) / Number::from(100),
|
||||
v => return Err(format!("$amount: {} is not a number.", v).into()),
|
||||
};
|
||||
let color = match arg!(args, 0, "color") {
|
||||
let color = match arg!(args, scope, super_selector, 0, "color") {
|
||||
Value::Color(c) => c,
|
||||
Value::Dimension(n, u) => {
|
||||
return Ok(Value::Ident(
|
||||
@ -270,13 +277,13 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"desaturate".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 2);
|
||||
let color = match arg!(args, 0, "color") {
|
||||
let color = match arg!(args, scope, super_selector, 0, "color") {
|
||||
Value::Color(c) => c,
|
||||
v => return Err(format!("$color: {} is not a color.", v).into()),
|
||||
};
|
||||
let amount = match arg!(args, 1, "amount") {
|
||||
let amount = match arg!(args, scope, super_selector, 1, "amount") {
|
||||
Value::Dimension(n, u) => bound!("amount", n, u, 0, 100) / Number::from(100),
|
||||
v => return Err(format!("$amount: {} is not a number.", v).into()),
|
||||
};
|
||||
@ -285,9 +292,9 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"grayscale".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 1);
|
||||
let color = match arg!(args, 0, "color") {
|
||||
let color = match arg!(args, scope, super_selector, 0, "color") {
|
||||
Value::Color(c) => c,
|
||||
Value::Dimension(n, u) => {
|
||||
return Ok(Value::Ident(
|
||||
@ -302,9 +309,9 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"complement".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 1);
|
||||
let color = match arg!(args, 0, "color") {
|
||||
let color = match arg!(args, scope, super_selector, 0, "color") {
|
||||
Value::Color(c) => c,
|
||||
v => return Err(format!("$color: {} is not a color.", v).into()),
|
||||
};
|
||||
@ -313,17 +320,19 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"invert".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 2);
|
||||
let weight = match arg!(
|
||||
args,
|
||||
scope,
|
||||
super_selector,
|
||||
1,
|
||||
"weight" = Value::Dimension(Number::from(100), Unit::Percent)
|
||||
) {
|
||||
Value::Dimension(n, u) => bound!("weight", n, u, 0, 100) / Number::from(100),
|
||||
v => return Err(format!("$weight: {} is not a number.", v).into()),
|
||||
};
|
||||
match arg!(args, 0, "color") {
|
||||
match arg!(args, scope, super_selector, 0, "color") {
|
||||
Value::Color(c) => Ok(Value::Color(c.invert(weight))),
|
||||
Value::Dimension(n, Unit::Percent) => {
|
||||
Ok(Value::Ident(format!("invert({}%)", n), QuoteKind::None))
|
||||
|
@ -9,9 +9,9 @@ use crate::value::Value;
|
||||
pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
f.insert(
|
||||
"alpha".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "color") {
|
||||
match arg!(args, scope, super_selector, 0, "color") {
|
||||
Value::Color(c) => Ok(Value::Dimension(c.alpha(), Unit::None)),
|
||||
v => Err(format!("$color: {} is not a color.", v).into()),
|
||||
}
|
||||
@ -19,9 +19,9 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"opacity".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "color") {
|
||||
match arg!(args, scope, super_selector, 0, "color") {
|
||||
Value::Color(c) => Ok(Value::Dimension(c.alpha(), Unit::None)),
|
||||
Value::Dimension(num, unit) => Ok(Value::Ident(
|
||||
format!("opacity({}{})", num, unit),
|
||||
@ -33,13 +33,13 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"opacify".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 2);
|
||||
let color = match arg!(args, 0, "color") {
|
||||
let color = match arg!(args, scope, super_selector, 0, "color") {
|
||||
Value::Color(c) => c,
|
||||
v => return Err(format!("$color: {} is not a color.", v).into()),
|
||||
};
|
||||
let amount = match arg!(args, 1, "amount") {
|
||||
let amount = match arg!(args, scope, super_selector, 1, "amount") {
|
||||
Value::Dimension(n, u) => bound!("amount", n, u, 0, 1),
|
||||
v => return Err(format!("$amount: {} is not a number.", v).into()),
|
||||
};
|
||||
@ -48,13 +48,13 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"fade-in".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 2);
|
||||
let color = match arg!(args, 0, "color") {
|
||||
let color = match arg!(args, scope, super_selector, 0, "color") {
|
||||
Value::Color(c) => c,
|
||||
v => return Err(format!("$color: {} is not a color.", v).into()),
|
||||
};
|
||||
let amount = match arg!(args, 1, "amount") {
|
||||
let amount = match arg!(args, scope, super_selector, 1, "amount") {
|
||||
Value::Dimension(n, u) => bound!("amount", n, u, 0, 1),
|
||||
v => return Err(format!("$amount: {} is not a number.", v).into()),
|
||||
};
|
||||
@ -63,13 +63,13 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"transparentize".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 2);
|
||||
let color = match arg!(args, 0, "color") {
|
||||
let color = match arg!(args, scope, super_selector, 0, "color") {
|
||||
Value::Color(c) => c,
|
||||
v => return Err(format!("$color: {} is not a color.", v).into()),
|
||||
};
|
||||
let amount = match arg!(args, 1, "amount") {
|
||||
let amount = match arg!(args, scope, super_selector, 1, "amount") {
|
||||
Value::Dimension(n, u) => bound!("amount", n, u, 0, 1),
|
||||
v => return Err(format!("$amount: {} is not a number.", v).into()),
|
||||
};
|
||||
@ -78,13 +78,13 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"fade-out".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 2);
|
||||
let color = match arg!(args, 0, "color") {
|
||||
let color = match arg!(args, scope, super_selector, 0, "color") {
|
||||
Value::Color(c) => c,
|
||||
v => return Err(format!("$color: {} is not a color.", v).into()),
|
||||
};
|
||||
let amount = match arg!(args, 1, "amount") {
|
||||
let amount = match arg!(args, scope, super_selector, 1, "amount") {
|
||||
Value::Dimension(n, u) => bound!("amount", n, u, 0, 1),
|
||||
v => return Err(format!("$amount: {} is not a number.", v).into()),
|
||||
};
|
||||
|
@ -9,9 +9,9 @@ use crate::unit::Unit;
|
||||
use crate::value::{Number, Value};
|
||||
|
||||
macro_rules! opt_rgba {
|
||||
($args:ident, $name:ident, $arg:literal, $low:literal, $high:literal) => {
|
||||
($args:ident, $name:ident, $arg:literal, $low:literal, $high:literal, $scope:ident, $super_selector:ident) => {
|
||||
let x = $low;
|
||||
let $name = match named_arg!($args, $arg = Value::Null) {
|
||||
let $name = match named_arg!($args, $scope, $super_selector, $arg = Value::Null) {
|
||||
Value::Dimension(n, u) => Some(bound!($arg, n, u, x, $high)),
|
||||
Value::Null => None,
|
||||
v => return Err(format!("${}: {} is not a number.", $arg, v).into()),
|
||||
@ -20,9 +20,9 @@ macro_rules! opt_rgba {
|
||||
}
|
||||
|
||||
macro_rules! opt_hsl {
|
||||
($args:ident, $name:ident, $arg:literal, $low:literal, $high:literal) => {
|
||||
($args:ident, $name:ident, $arg:literal, $low:literal, $high:literal, $scope:ident, $super_selector:ident) => {
|
||||
let x = $low;
|
||||
let $name = match named_arg!($args, $arg = Value::Null) {
|
||||
let $name = match named_arg!($args, $scope, $super_selector, $arg = Value::Null) {
|
||||
Value::Dimension(n, u) => Some(bound!($arg, n, u, x, $high) / Number::from(100)),
|
||||
Value::Null => None,
|
||||
v => return Err(format!("${}: {} is not a number.", $arg, v).into()),
|
||||
@ -31,33 +31,33 @@ macro_rules! opt_hsl {
|
||||
}
|
||||
|
||||
pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
f.insert("change-color".to_owned(), Builtin::new(|mut args, _| {
|
||||
if args.get_positional(1).is_some() {
|
||||
f.insert("change-color".to_owned(), Builtin::new(|mut args, scope, super_selector| {
|
||||
if args.get_positional(1, scope, super_selector).is_some() {
|
||||
return Err("Only one positional argument is allowed. All other arguments must be passed by name.".into());
|
||||
}
|
||||
|
||||
let color = match arg!(args, 0, "color") {
|
||||
let color = match arg!(args, scope, super_selector, 0, "color") {
|
||||
Value::Color(c) => c,
|
||||
v => return Err(format!("$color: {} is not a color.", v).into()),
|
||||
};
|
||||
|
||||
opt_rgba!(args, alpha, "alpha", 0, 1);
|
||||
opt_rgba!(args, red, "red", 0, 255);
|
||||
opt_rgba!(args, green, "green", 0, 255);
|
||||
opt_rgba!(args, blue, "blue", 0, 255);
|
||||
opt_rgba!(args, alpha, "alpha", 0, 1, scope, super_selector);
|
||||
opt_rgba!(args, red, "red", 0, 255, scope, super_selector);
|
||||
opt_rgba!(args, green, "green", 0, 255, scope, super_selector);
|
||||
opt_rgba!(args, blue, "blue", 0, 255, scope, super_selector);
|
||||
|
||||
if red.is_some() || green.is_some() || blue.is_some() {
|
||||
return Ok(Value::Color(Color::from_rgba(red.unwrap_or(color.red()), green.unwrap_or(color.green()), blue.unwrap_or(color.blue()), alpha.unwrap_or(color.alpha()))))
|
||||
}
|
||||
|
||||
let hue = match named_arg!(args, "hue"=Value::Null) {
|
||||
let hue = match named_arg!(args, scope, super_selector, "hue"=Value::Null) {
|
||||
Value::Dimension(n, _) => Some(n),
|
||||
Value::Null => None,
|
||||
v => return Err(format!("$hue: {} is not a number.", v).into()),
|
||||
};
|
||||
|
||||
opt_hsl!(args, saturation, "saturation", 0, 100);
|
||||
opt_hsl!(args, luminance, "lightness", 0, 100);
|
||||
opt_hsl!(args, saturation, "saturation", 0, 100, scope, super_selector);
|
||||
opt_hsl!(args, luminance, "lightness", 0, 100, scope, super_selector);
|
||||
|
||||
if hue.is_some() || saturation.is_some() || luminance.is_some() {
|
||||
// Color::as_hsla() returns more exact values than Color::hue(), etc.
|
||||
@ -73,16 +73,16 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
}));
|
||||
f.insert(
|
||||
"adjust-color".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
let color = match arg!(args, 0, "color") {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
let color = match arg!(args, scope, super_selector, 0, "color") {
|
||||
Value::Color(c) => c,
|
||||
v => return Err(format!("$color: {} is not a color.", v).into()),
|
||||
};
|
||||
|
||||
opt_rgba!(args, alpha, "alpha", -1, 1);
|
||||
opt_rgba!(args, red, "red", -255, 255);
|
||||
opt_rgba!(args, green, "green", -255, 255);
|
||||
opt_rgba!(args, blue, "blue", -255, 255);
|
||||
opt_rgba!(args, alpha, "alpha", -1, 1, scope, super_selector);
|
||||
opt_rgba!(args, red, "red", -255, 255, scope, super_selector);
|
||||
opt_rgba!(args, green, "green", -255, 255, scope, super_selector);
|
||||
opt_rgba!(args, blue, "blue", -255, 255, scope, super_selector);
|
||||
|
||||
if red.is_some() || green.is_some() || blue.is_some() {
|
||||
return Ok(Value::Color(Color::from_rgba(
|
||||
@ -93,14 +93,30 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
)));
|
||||
}
|
||||
|
||||
let hue = match named_arg!(args, "hue" = Value::Null) {
|
||||
let hue = match named_arg!(args, scope, super_selector, "hue" = Value::Null) {
|
||||
Value::Dimension(n, _) => Some(n),
|
||||
Value::Null => None,
|
||||
v => return Err(format!("$hue: {} is not a number.", v).into()),
|
||||
};
|
||||
|
||||
opt_hsl!(args, saturation, "saturation", -100, 100);
|
||||
opt_hsl!(args, luminance, "lightness", -100, 100);
|
||||
opt_hsl!(
|
||||
args,
|
||||
saturation,
|
||||
"saturation",
|
||||
-100,
|
||||
100,
|
||||
scope,
|
||||
super_selector
|
||||
);
|
||||
opt_hsl!(
|
||||
args,
|
||||
luminance,
|
||||
"lightness",
|
||||
-100,
|
||||
100,
|
||||
scope,
|
||||
super_selector
|
||||
);
|
||||
|
||||
if hue.is_some() || saturation.is_some() || luminance.is_some() {
|
||||
// Color::as_hsla() returns more exact values than Color::hue(), etc.
|
||||
@ -123,16 +139,16 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"scale-color".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
let color = match arg!(args, 0, "color") {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
let color = match arg!(args, scope, super_selector, 0, "color") {
|
||||
Value::Color(c) => c,
|
||||
v => return Err(format!("$color: {} is not a color.", v).into()),
|
||||
};
|
||||
|
||||
macro_rules! opt_scale_arg {
|
||||
($args:ident, $name:ident, $arg:literal, $low:literal, $high:literal) => {
|
||||
($args:ident, $name:ident, $arg:literal, $low:literal, $high:literal, $scope:ident, $super_selector:ident) => {
|
||||
let x = $low;
|
||||
let $name = match named_arg!($args, $arg = Value::Null) {
|
||||
let $name = match named_arg!($args, $scope, $super_selector, $arg = Value::Null) {
|
||||
Value::Dimension(n, Unit::Percent) => {
|
||||
Some(bound!($arg, n, Unit::Percent, x, $high) / Number::from(100))
|
||||
}
|
||||
@ -147,10 +163,10 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
};
|
||||
}
|
||||
|
||||
opt_scale_arg!(args, alpha, "alpha", -100, 100);
|
||||
opt_scale_arg!(args, red, "red", -100, 100);
|
||||
opt_scale_arg!(args, green, "green", -100, 100);
|
||||
opt_scale_arg!(args, blue, "blue", -100, 100);
|
||||
opt_scale_arg!(args, alpha, "alpha", -100, 100, scope, super_selector);
|
||||
opt_scale_arg!(args, red, "red", -100, 100, scope, super_selector);
|
||||
opt_scale_arg!(args, green, "green", -100, 100, scope, super_selector);
|
||||
opt_scale_arg!(args, blue, "blue", -100, 100, scope, super_selector);
|
||||
|
||||
if red.is_some() || green.is_some() || blue.is_some() {
|
||||
return Ok(Value::Color(Color::from_rgba(
|
||||
@ -177,8 +193,8 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
)));
|
||||
}
|
||||
|
||||
opt_scale_arg!(args, saturation, "saturation", -100, 100);
|
||||
opt_scale_arg!(args, luminance, "lightness", -100, 100);
|
||||
opt_scale_arg!(args, saturation, "saturation", -100, 100, scope, super_selector);
|
||||
opt_scale_arg!(args, luminance, "lightness", -100, 100, scope, super_selector);
|
||||
|
||||
if saturation.is_some() || luminance.is_some() {
|
||||
// Color::as_hsla() returns more exact values than Color::hue(), etc.
|
||||
@ -209,9 +225,9 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"ie-hex-str".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 1);
|
||||
let color = match arg!(args, 0, "color") {
|
||||
let color = match arg!(args, scope, super_selector, 0, "color") {
|
||||
Value::Color(c) => c,
|
||||
v => return Err(format!("$color: {} is not a color.", v).into()),
|
||||
};
|
||||
|
@ -10,13 +10,13 @@ use crate::value::{Number, Value};
|
||||
pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
f.insert(
|
||||
"rgb".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
if args.is_empty() {
|
||||
return Err("Missing argument $channels.".into());
|
||||
}
|
||||
|
||||
if args.len() == 1 {
|
||||
let mut channels = match arg!(args, 0, "channels") {
|
||||
let mut channels = match arg!(args, scope, super_selector, 0, "channels") {
|
||||
Value::List(v, ..) => v,
|
||||
_ => return Err("Missing argument $channels.".into()),
|
||||
};
|
||||
@ -60,11 +60,11 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
|
||||
Ok(Value::Color(color))
|
||||
} else if args.len() == 2 {
|
||||
let color = match arg!(args, 0, "color") {
|
||||
let color = match arg!(args, scope, super_selector, 0, "color") {
|
||||
Value::Color(c) => c,
|
||||
v => return Err(format!("$color: {} is not a color.", v).into()),
|
||||
};
|
||||
let alpha = match arg!(args, 1, "alpha") {
|
||||
let alpha = match arg!(args, scope, super_selector, 1, "alpha") {
|
||||
Value::Dimension(n, Unit::None) => n,
|
||||
Value::Dimension(n, Unit::Percent) => n / Number::from(100),
|
||||
v @ Value::Dimension(..) => {
|
||||
@ -76,7 +76,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
};
|
||||
Ok(Value::Color(color.with_alpha(alpha)))
|
||||
} else {
|
||||
let red = match arg!(args, 0, "red") {
|
||||
let red = match arg!(args, scope, super_selector, 0, "red") {
|
||||
Value::Dimension(n, Unit::None) => n,
|
||||
Value::Dimension(n, Unit::Percent) => {
|
||||
(n / Number::from(100)) * Number::from(255)
|
||||
@ -88,7 +88,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
}
|
||||
v => return Err(format!("$red: {} is not a number.", v).into()),
|
||||
};
|
||||
let green = match arg!(args, 1, "green") {
|
||||
let green = match arg!(args, scope, super_selector, 1, "green") {
|
||||
Value::Dimension(n, Unit::None) => n,
|
||||
Value::Dimension(n, Unit::Percent) => {
|
||||
(n / Number::from(100)) * Number::from(255)
|
||||
@ -100,7 +100,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
}
|
||||
v => return Err(format!("$green: {} is not a number.", v).into()),
|
||||
};
|
||||
let blue = match arg!(args, 2, "blue") {
|
||||
let blue = match arg!(args, scope, super_selector, 2, "blue") {
|
||||
Value::Dimension(n, Unit::None) => n,
|
||||
Value::Dimension(n, Unit::Percent) => {
|
||||
(n / Number::from(100)) * Number::from(255)
|
||||
@ -114,6 +114,8 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
};
|
||||
let alpha = match arg!(
|
||||
args,
|
||||
scope,
|
||||
super_selector,
|
||||
3,
|
||||
"alpha" = Value::Dimension(Number::one(), Unit::None)
|
||||
) {
|
||||
@ -132,13 +134,13 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"rgba".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
if args.is_empty() {
|
||||
return Err("Missing argument $channels.".into());
|
||||
}
|
||||
|
||||
if args.len() == 1 {
|
||||
let mut channels = match arg!(args, 0, "channels") {
|
||||
let mut channels = match arg!(args, scope, super_selector, 0, "channels") {
|
||||
Value::List(v, ..) => v,
|
||||
_ => return Err("Missing argument $channels.".into()),
|
||||
};
|
||||
@ -182,11 +184,11 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
|
||||
Ok(Value::Color(color))
|
||||
} else if args.len() == 2 {
|
||||
let color = match arg!(args, 0, "color") {
|
||||
let color = match arg!(args, scope, super_selector, 0, "color") {
|
||||
Value::Color(c) => c,
|
||||
v => return Err(format!("$color: {} is not a color.", v).into()),
|
||||
};
|
||||
let alpha = match arg!(args, 1, "alpha") {
|
||||
let alpha = match arg!(args, scope, super_selector, 1, "alpha") {
|
||||
Value::Dimension(n, Unit::None) => n,
|
||||
Value::Dimension(n, Unit::Percent) => n / Number::from(100),
|
||||
v @ Value::Dimension(..) => {
|
||||
@ -198,7 +200,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
};
|
||||
Ok(Value::Color(color.with_alpha(alpha)))
|
||||
} else {
|
||||
let red = match arg!(args, 0, "red") {
|
||||
let red = match arg!(args, scope, super_selector, 0, "red") {
|
||||
Value::Dimension(n, Unit::None) => n,
|
||||
Value::Dimension(n, Unit::Percent) => {
|
||||
(n / Number::from(100)) * Number::from(255)
|
||||
@ -210,7 +212,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
}
|
||||
v => return Err(format!("$red: {} is not a number.", v).into()),
|
||||
};
|
||||
let green = match arg!(args, 1, "green") {
|
||||
let green = match arg!(args, scope, super_selector, 1, "green") {
|
||||
Value::Dimension(n, Unit::None) => n,
|
||||
Value::Dimension(n, Unit::Percent) => {
|
||||
(n / Number::from(100)) * Number::from(255)
|
||||
@ -222,7 +224,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
}
|
||||
v => return Err(format!("$green: {} is not a number.", v).into()),
|
||||
};
|
||||
let blue = match arg!(args, 2, "blue") {
|
||||
let blue = match arg!(args, scope, super_selector, 2, "blue") {
|
||||
Value::Dimension(n, Unit::None) => n,
|
||||
Value::Dimension(n, Unit::Percent) => {
|
||||
(n / Number::from(100)) * Number::from(255)
|
||||
@ -236,6 +238,8 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
};
|
||||
let alpha = match arg!(
|
||||
args,
|
||||
scope,
|
||||
super_selector,
|
||||
3,
|
||||
"alpha" = Value::Dimension(Number::one(), Unit::None)
|
||||
) {
|
||||
@ -254,9 +258,9 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"red".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "color") {
|
||||
match arg!(args, scope, super_selector, 0, "color") {
|
||||
Value::Color(c) => Ok(Value::Dimension(c.red(), Unit::None)),
|
||||
v => Err(format!("$color: {} is not a color.", v).into()),
|
||||
}
|
||||
@ -264,9 +268,9 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"green".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "color") {
|
||||
match arg!(args, scope, super_selector, 0, "color") {
|
||||
Value::Color(c) => Ok(Value::Dimension(c.green(), Unit::None)),
|
||||
v => Err(format!("$color: {} is not a color.", v).into()),
|
||||
}
|
||||
@ -274,9 +278,9 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"blue".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "color") {
|
||||
match arg!(args, scope, super_selector, 0, "color") {
|
||||
Value::Color(c) => Ok(Value::Dimension(c.blue(), Unit::None)),
|
||||
v => Err(format!("$color: {} is not a color.", v).into()),
|
||||
}
|
||||
@ -284,20 +288,22 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"mix".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 3);
|
||||
let color1 = match arg!(args, 0, "color1") {
|
||||
let color1 = match arg!(args, scope, super_selector, 0, "color1") {
|
||||
Value::Color(c) => c,
|
||||
v => return Err(format!("$color1: {} is not a color.", v).into()),
|
||||
};
|
||||
|
||||
let color2 = match arg!(args, 1, "color2") {
|
||||
let color2 = match arg!(args, scope, super_selector, 1, "color2") {
|
||||
Value::Color(c) => c,
|
||||
v => return Err(format!("$color2: {} is not a color.", v).into()),
|
||||
};
|
||||
|
||||
let weight = match arg!(
|
||||
args,
|
||||
scope,
|
||||
super_selector,
|
||||
2,
|
||||
"weight" = Value::Dimension(Number::from(50), Unit::None)
|
||||
) {
|
||||
|
@ -10,9 +10,9 @@ use crate::value::{Number, Value};
|
||||
pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
f.insert(
|
||||
"length".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 1);
|
||||
let len = match arg!(args, 0, "list") {
|
||||
let len = match arg!(args, scope, super_selector, 0, "list") {
|
||||
Value::List(v, ..) => Number::from(v.len()),
|
||||
Value::Map(m) => Number::from(m.len()),
|
||||
_ => Number::one(),
|
||||
@ -22,14 +22,14 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"nth".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 2);
|
||||
let list = match arg!(args, 0, "list") {
|
||||
let list = match arg!(args, scope, super_selector, 0, "list") {
|
||||
Value::List(v, ..) => v,
|
||||
Value::Map(m) => m.entries(),
|
||||
v => vec![v],
|
||||
};
|
||||
let n = match arg!(args, 1, "n") {
|
||||
let n = match arg!(args, scope, super_selector, 1, "n") {
|
||||
Value::Dimension(num, _) => num,
|
||||
v => return Err(format!("$n: {} is not a number.", v).into()),
|
||||
};
|
||||
@ -60,10 +60,10 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"list-separator".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 1);
|
||||
Ok(Value::Ident(
|
||||
match arg!(args, 0, "list") {
|
||||
match arg!(args, scope, super_selector, 0, "list") {
|
||||
Value::List(_, sep, ..) => sep.name(),
|
||||
_ => ListSeparator::Space.name(),
|
||||
}
|
||||
@ -74,14 +74,14 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"set-nth".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 3);
|
||||
let (mut list, sep, brackets) = match arg!(args, 0, "list") {
|
||||
let (mut list, sep, brackets) = match arg!(args, scope, super_selector, 0, "list") {
|
||||
Value::List(v, sep, b) => (v, sep, b),
|
||||
Value::Map(m) => (m.entries(), ListSeparator::Comma, Brackets::None),
|
||||
v => (vec![v], ListSeparator::Space, Brackets::None),
|
||||
};
|
||||
let n = match arg!(args, 1, "n") {
|
||||
let n = match arg!(args, scope, super_selector, 1, "n") {
|
||||
Value::Dimension(num, _) => num,
|
||||
v => return Err(format!("$n: {} is not a number.", v).into()),
|
||||
};
|
||||
@ -102,7 +102,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
return Err(format!("$n: {} is not an int.", n).into());
|
||||
}
|
||||
|
||||
let val = arg!(args, 2, "value");
|
||||
let val = arg!(args, scope, super_selector, 2, "value");
|
||||
|
||||
if n.is_positive() {
|
||||
list[n.to_integer().to_usize().unwrap() - 1] = val;
|
||||
@ -115,15 +115,17 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"append".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 3);
|
||||
let (mut list, sep, brackets) = match arg!(args, 0, "list") {
|
||||
let (mut list, sep, brackets) = match arg!(args, scope, super_selector, 0, "list") {
|
||||
Value::List(v, sep, b) => (v, sep, b),
|
||||
v => (vec![v], ListSeparator::Space, Brackets::None),
|
||||
};
|
||||
let val = arg!(args, 1, "val");
|
||||
let val = arg!(args, scope, super_selector, 1, "val");
|
||||
let sep = match arg!(
|
||||
args,
|
||||
scope,
|
||||
super_selector,
|
||||
2,
|
||||
"separator" = Value::Ident("auto".to_owned(), QuoteKind::None)
|
||||
) {
|
||||
@ -145,20 +147,22 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"join".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 4);
|
||||
let (mut list1, sep1, brackets) = match arg!(args, 0, "list1") {
|
||||
let (mut list1, sep1, brackets) = match arg!(args, scope, super_selector, 0, "list1") {
|
||||
Value::List(v, sep, brackets) => (v, sep, brackets),
|
||||
Value::Map(m) => (m.entries(), ListSeparator::Comma, Brackets::None),
|
||||
v => (vec![v], ListSeparator::Space, Brackets::None),
|
||||
};
|
||||
let (list2, sep2) = match arg!(args, 1, "list2") {
|
||||
let (list2, sep2) = match arg!(args, scope, super_selector, 1, "list2") {
|
||||
Value::List(v, sep, ..) => (v, sep),
|
||||
Value::Map(m) => (m.entries(), ListSeparator::Comma),
|
||||
v => (vec![v], ListSeparator::Space),
|
||||
};
|
||||
let sep = match arg!(
|
||||
args,
|
||||
scope,
|
||||
super_selector,
|
||||
2,
|
||||
"separator" = Value::Ident("auto".to_owned(), QuoteKind::None)
|
||||
) {
|
||||
@ -181,6 +185,8 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
|
||||
let brackets = match arg!(
|
||||
args,
|
||||
scope,
|
||||
super_selector,
|
||||
3,
|
||||
"bracketed" = Value::Ident("auto".to_owned(), QuoteKind::None)
|
||||
) {
|
||||
@ -204,27 +210,29 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"is-bracketed".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 1);
|
||||
Ok(Value::bool(match arg!(args, 0, "list") {
|
||||
Value::List(.., brackets) => match brackets {
|
||||
Brackets::Bracketed => true,
|
||||
Brackets::None => false,
|
||||
Ok(Value::bool(
|
||||
match arg!(args, scope, super_selector, 0, "list") {
|
||||
Value::List(.., brackets) => match brackets {
|
||||
Brackets::Bracketed => true,
|
||||
Brackets::None => false,
|
||||
},
|
||||
_ => false,
|
||||
},
|
||||
_ => false,
|
||||
}))
|
||||
))
|
||||
}),
|
||||
);
|
||||
f.insert(
|
||||
"index".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 2);
|
||||
let list = match arg!(args, 0, "list") {
|
||||
let list = match arg!(args, scope, super_selector, 0, "list") {
|
||||
Value::List(v, ..) => v,
|
||||
Value::Map(m) => m.entries(),
|
||||
v => vec![v],
|
||||
};
|
||||
let value = arg!(args, 1, "value");
|
||||
let value = arg!(args, scope, super_selector, 1, "value");
|
||||
// TODO: find a way around this unwrap.
|
||||
// It should be impossible to hit as the arg is
|
||||
// evaluated prior to checking equality, but
|
||||
@ -242,9 +250,9 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"zip".to_owned(),
|
||||
Builtin::new(|args, _| {
|
||||
Builtin::new(|args, scope, super_selector| {
|
||||
let lists = args
|
||||
.get_variadic()?
|
||||
.get_variadic(scope, super_selector)?
|
||||
.into_iter()
|
||||
.map(|x| match x {
|
||||
Value::List(v, ..) => v,
|
||||
|
@ -1,18 +1,18 @@
|
||||
macro_rules! arg {
|
||||
($args:ident, $idx:literal, $name:literal) => {
|
||||
match $args.remove_positional($idx) {
|
||||
Some(v) => v.eval()?,
|
||||
None => match $args.remove_named($name.to_owned()) {
|
||||
Some(v) => v.eval()?,
|
||||
($args:ident, $scope:ident, $super_selector:ident, $idx:literal, $name:literal) => {
|
||||
match $args.get_positional($idx, $scope, $super_selector) {
|
||||
Some(v) => v?.eval()?,
|
||||
None => match $args.get_named($name.to_owned(), $scope, $super_selector) {
|
||||
Some(v) => v?.eval()?,
|
||||
None => return Err(concat!("Missing argument $", $name, ".").into()),
|
||||
},
|
||||
};
|
||||
};
|
||||
($args:ident, $idx:literal, $name:literal=$default:expr) => {
|
||||
match $args.remove_positional($idx) {
|
||||
Some(v) => v.eval()?,
|
||||
None => match $args.remove_named($name.to_owned()) {
|
||||
Some(v) => v.eval()?,
|
||||
($args:ident, $scope:ident, $super_selector:ident, $idx:literal, $name:literal=$default:expr) => {
|
||||
match $args.get_positional($idx, $scope, $super_selector) {
|
||||
Some(v) => v?.eval()?,
|
||||
None => match $args.get_named($name.to_owned(), $scope, $super_selector) {
|
||||
Some(v) => v?.eval()?,
|
||||
None => $default,
|
||||
},
|
||||
};
|
||||
@ -20,15 +20,15 @@ macro_rules! arg {
|
||||
}
|
||||
|
||||
macro_rules! named_arg {
|
||||
($args:ident, $name:literal) => {
|
||||
match $args.remove_named($name.to_owned()) {
|
||||
Some(v) => v.eval()?,
|
||||
($args:ident, $scope:ident, $super_selector:ident, $name:literal) => {
|
||||
match $args.get_named($name.to_owned(), $scope, $super_selector) {
|
||||
Some(v) => v?.eval()?,
|
||||
None => return Err(concat!("Missing argument $", $name, ".").into()),
|
||||
};
|
||||
};
|
||||
($args:ident, $name:literal=$default:expr) => {
|
||||
match $args.remove_named($name.to_owned()) {
|
||||
Some(v) => v.eval()?,
|
||||
($args:ident, $scope:ident, $super_selector:ident, $name:literal=$default:expr) => {
|
||||
match $args.get_named($name.to_owned(), $scope, $super_selector) {
|
||||
Some(v) => v?.eval()?,
|
||||
None => $default,
|
||||
};
|
||||
};
|
||||
|
@ -7,10 +7,10 @@ use crate::value::{SassMap, Value};
|
||||
pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
f.insert(
|
||||
"map-get".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 2);
|
||||
let key = arg!(args, 1, "key");
|
||||
let map = match arg!(args, 0, "map") {
|
||||
let key = arg!(args, scope, super_selector, 1, "key");
|
||||
let map = match arg!(args, scope, super_selector, 0, "map") {
|
||||
Value::Map(m) => m,
|
||||
Value::List(v, ..) if v.is_empty() => SassMap::new(),
|
||||
v => return Err(format!("$map: {} is not a map.", v).into()),
|
||||
@ -20,10 +20,10 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"map-has-key".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 2);
|
||||
let key = arg!(args, 1, "key");
|
||||
let map = match arg!(args, 0, "map") {
|
||||
let key = arg!(args, scope, super_selector, 1, "key");
|
||||
let map = match arg!(args, scope, super_selector, 0, "map") {
|
||||
Value::Map(m) => m,
|
||||
Value::List(v, ..) if v.is_empty() => SassMap::new(),
|
||||
v => return Err(format!("$map: {} is not a map.", v).into()),
|
||||
@ -33,9 +33,9 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"map-keys".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 1);
|
||||
let map = match arg!(args, 0, "map") {
|
||||
let map = match arg!(args, scope, super_selector, 0, "map") {
|
||||
Value::Map(m) => m,
|
||||
Value::List(v, ..) if v.is_empty() => SassMap::new(),
|
||||
v => return Err(format!("$map: {} is not a map.", v).into()),
|
||||
@ -49,9 +49,9 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"map-values".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 1);
|
||||
let map = match arg!(args, 0, "map") {
|
||||
let map = match arg!(args, scope, super_selector, 0, "map") {
|
||||
Value::Map(m) => m,
|
||||
Value::List(v, ..) if v.is_empty() => SassMap::new(),
|
||||
v => return Err(format!("$map: {} is not a map.", v).into()),
|
||||
@ -65,14 +65,14 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"map-merge".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 2);
|
||||
let mut map1 = match arg!(args, 0, "map1") {
|
||||
let mut map1 = match arg!(args, scope, super_selector, 0, "map1") {
|
||||
Value::Map(m) => m,
|
||||
Value::List(v, ..) if v.is_empty() => SassMap::new(),
|
||||
v => return Err(format!("$map1: {} is not a map.", v).into()),
|
||||
};
|
||||
let map2 = match arg!(args, 1, "map2") {
|
||||
let map2 = match arg!(args, scope, super_selector, 1, "map2") {
|
||||
Value::Map(m) => m,
|
||||
Value::List(v, ..) if v.is_empty() => SassMap::new(),
|
||||
v => return Err(format!("$map2: {} is not a map.", v).into()),
|
||||
@ -83,13 +83,13 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"map-remove".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
let mut map = match arg!(args, 0, "map") {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
let mut map = match arg!(args, scope, super_selector, 0, "map") {
|
||||
Value::Map(m) => m,
|
||||
Value::List(v, ..) if v.is_empty() => SassMap::new(),
|
||||
v => return Err(format!("$map: {} is not a map.", v).into()),
|
||||
};
|
||||
let keys = args.get_variadic()?;
|
||||
let keys = args.get_variadic(scope, super_selector)?;
|
||||
for key in keys {
|
||||
map.remove(&key);
|
||||
}
|
||||
|
@ -7,9 +7,9 @@ use crate::value::{Number, Value};
|
||||
pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
f.insert(
|
||||
"percentage".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 1);
|
||||
let num = match arg!(args, 0, "number") {
|
||||
let num = match arg!(args, scope, super_selector, 0, "number") {
|
||||
Value::Dimension(n, Unit::None) => n * Number::from(100),
|
||||
v @ Value::Dimension(..) => {
|
||||
return Err(format!("$number: Expected {} to have no units.", v).into())
|
||||
@ -21,9 +21,9 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"round".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "number") {
|
||||
match arg!(args, scope, super_selector, 0, "number") {
|
||||
Value::Dimension(n, u) => Ok(Value::Dimension(n.round(), u)),
|
||||
v => Err(format!("$number: {} is not a number.", v).into()),
|
||||
}
|
||||
@ -31,9 +31,9 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"ceil".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "number") {
|
||||
match arg!(args, scope, super_selector, 0, "number") {
|
||||
Value::Dimension(n, u) => Ok(Value::Dimension(n.ceil(), u)),
|
||||
v => Err(format!("$number: {} is not a number.", v).into()),
|
||||
}
|
||||
@ -41,9 +41,9 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"floor".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "number") {
|
||||
match arg!(args, scope, super_selector, 0, "number") {
|
||||
Value::Dimension(n, u) => Ok(Value::Dimension(n.floor(), u)),
|
||||
v => Err(format!("$number: {} is not a number.", v).into()),
|
||||
}
|
||||
@ -51,9 +51,9 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"abs".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "number") {
|
||||
match arg!(args, scope, super_selector, 0, "number") {
|
||||
Value::Dimension(n, u) => Ok(Value::Dimension(n.abs(), u)),
|
||||
v => Err(format!("$number: {} is not a number.", v).into()),
|
||||
}
|
||||
@ -61,13 +61,13 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"comparable".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 2);
|
||||
let unit1 = match arg!(args, 0, "number1") {
|
||||
let unit1 = match arg!(args, scope, super_selector, 0, "number1") {
|
||||
Value::Dimension(_, u) => u,
|
||||
v => return Err(format!("$number1: {} is not a number.", v).into()),
|
||||
};
|
||||
let unit2 = match arg!(args, 1, "number2") {
|
||||
let unit2 = match arg!(args, scope, super_selector, 1, "number2") {
|
||||
Value::Dimension(_, u) => u,
|
||||
v => return Err(format!("$number2: {} is not a number.", v).into()),
|
||||
};
|
||||
|
@ -9,20 +9,20 @@ use crate::value::{SassFunction, Value};
|
||||
pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
f.insert(
|
||||
"if".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 3);
|
||||
if arg!(args, 0, "condition").is_true()? {
|
||||
Ok(arg!(args, 1, "if-true"))
|
||||
if arg!(args, scope, super_selector, 0, "condition").is_true()? {
|
||||
Ok(arg!(args, scope, super_selector, 1, "if-true"))
|
||||
} else {
|
||||
Ok(arg!(args, 2, "if-false"))
|
||||
Ok(arg!(args, scope, super_selector, 2, "if-false"))
|
||||
}
|
||||
}),
|
||||
);
|
||||
f.insert(
|
||||
"feature-exists".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "feature") {
|
||||
match arg!(args, scope, super_selector, 0, "feature") {
|
||||
Value::Ident(s, _) => match s.as_str() {
|
||||
// A local variable will shadow a global variable unless
|
||||
// `!global` is used.
|
||||
@ -47,9 +47,9 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"unit".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 1);
|
||||
let unit = match arg!(args, 0, "number") {
|
||||
let unit = match arg!(args, scope, super_selector, 0, "number") {
|
||||
Value::Dimension(_, u) => u.to_string(),
|
||||
v => return Err(format!("$number: {} is not a number.", v).into()),
|
||||
};
|
||||
@ -58,17 +58,17 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"type-of".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 1);
|
||||
let value = arg!(args, 0, "value");
|
||||
let value = arg!(args, scope, super_selector, 0, "value");
|
||||
Ok(Value::Ident(value.kind()?.to_owned(), QuoteKind::None))
|
||||
}),
|
||||
);
|
||||
f.insert(
|
||||
"unitless".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "number") {
|
||||
match arg!(args, scope, super_selector, 0, "number") {
|
||||
Value::Dimension(_, Unit::None) => Ok(Value::True),
|
||||
Value::Dimension(_, _) => Ok(Value::False),
|
||||
_ => Ok(Value::True),
|
||||
@ -77,10 +77,10 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"inspect".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 1);
|
||||
Ok(Value::Ident(
|
||||
match arg!(args, 0, "value") {
|
||||
match arg!(args, scope, super_selector, 0, "value") {
|
||||
Value::List(v, _, brackets) if v.is_empty() => match brackets {
|
||||
Brackets::None => "()".to_string(),
|
||||
Brackets::Bracketed => "[]".to_string(),
|
||||
@ -94,9 +94,9 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"variable-exists".to_owned(),
|
||||
Builtin::new(|mut args, scope| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "name") {
|
||||
match arg!(args, scope, super_selector, 0, "name") {
|
||||
Value::Ident(s, _) => Ok(Value::bool(scope.var_exists(&s))),
|
||||
v => Err(format!("$name: {} is not a string.", v).into()),
|
||||
}
|
||||
@ -104,9 +104,9 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"global-variable-exists".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "name") {
|
||||
match arg!(args, scope, super_selector, 0, "name") {
|
||||
Value::Ident(s, _) => Ok(Value::bool(global_var_exists(&s))),
|
||||
v => Err(format!("$name: {} is not a string.", v).into()),
|
||||
}
|
||||
@ -114,9 +114,9 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"mixin-exists".to_owned(),
|
||||
Builtin::new(|mut args, scope| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 2);
|
||||
match arg!(args, 0, "name") {
|
||||
match arg!(args, scope, super_selector, 0, "name") {
|
||||
Value::Ident(s, _) => Ok(Value::bool(scope.mixin_exists(&s))),
|
||||
v => Err(format!("$name: {} is not a string.", v).into()),
|
||||
}
|
||||
@ -124,9 +124,9 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"function-exists".to_owned(),
|
||||
Builtin::new(|mut args, scope| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 2);
|
||||
match arg!(args, 0, "name") {
|
||||
match arg!(args, scope, super_selector, 0, "name") {
|
||||
Value::Ident(s, _) => Ok(Value::bool(
|
||||
scope.fn_exists(&s) || GLOBAL_FUNCTIONS.contains_key(&s),
|
||||
)),
|
||||
@ -136,14 +136,14 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"get-function".to_owned(),
|
||||
Builtin::new(|mut args, scope| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 3);
|
||||
let name = match arg!(args, 0, "name") {
|
||||
let name = match arg!(args, scope, super_selector, 0, "name") {
|
||||
Value::Ident(s, _) => s,
|
||||
v => return Err(format!("$name: {} is not a string.", v).into()),
|
||||
};
|
||||
let css = arg!(args, 1, "css" = Value::False).is_true()?;
|
||||
let module = match arg!(args, 2, "module" = Value::Null) {
|
||||
let css = arg!(args, scope, super_selector, 1, "css" = Value::False).is_true()?;
|
||||
let module = match arg!(args, scope, super_selector, 2, "module" = Value::Null) {
|
||||
Value::Ident(s, ..) => Some(s),
|
||||
Value::Null => None,
|
||||
v => return Err(format!("$module: {} is not a string.", v).into()),
|
||||
@ -166,12 +166,12 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"call".to_owned(),
|
||||
Builtin::new(|mut args, scope| {
|
||||
let func = match arg!(args, 0, "function") {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
let func = match arg!(args, scope, super_selector, 0, "function") {
|
||||
Value::Function(f) => f,
|
||||
v => return Err(format!("$function: {} is not a function reference.", v).into()),
|
||||
};
|
||||
func.call(args.decrement(), scope)
|
||||
func.call(args.decrement(), scope, super_selector)
|
||||
}),
|
||||
);
|
||||
}
|
||||
|
@ -5,6 +5,7 @@ use std::sync::atomic::{AtomicUsize, Ordering};
|
||||
use crate::args::CallArgs;
|
||||
use crate::error::SassResult;
|
||||
use crate::scope::Scope;
|
||||
use crate::selector::Selector;
|
||||
use crate::value::Value;
|
||||
|
||||
#[macro_use]
|
||||
@ -22,9 +23,12 @@ static FUNCTION_COUNT: AtomicUsize = AtomicUsize::new(0);
|
||||
|
||||
// TODO: impl Fn
|
||||
#[derive(Clone)]
|
||||
pub(crate) struct Builtin(pub fn(CallArgs, &Scope) -> SassResult<Value>, usize);
|
||||
pub(crate) struct Builtin(
|
||||
pub fn(CallArgs, &Scope, &Selector) -> SassResult<Value>,
|
||||
usize,
|
||||
);
|
||||
impl Builtin {
|
||||
pub fn new(body: fn(CallArgs, &Scope) -> SassResult<Value>) -> Builtin {
|
||||
pub fn new(body: fn(CallArgs, &Scope, &Selector) -> SassResult<Value>) -> Builtin {
|
||||
let count = FUNCTION_COUNT.fetch_add(1, Ordering::Relaxed);
|
||||
Self(body, count)
|
||||
}
|
||||
|
@ -11,9 +11,9 @@ use crate::value::{Number, Value};
|
||||
pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
f.insert(
|
||||
"to-upper-case".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "string") {
|
||||
match arg!(args, scope, super_selector, 0, "string") {
|
||||
Value::Ident(i, q) => Ok(Value::Ident(i.to_ascii_uppercase(), q)),
|
||||
v => Err(format!("$string: {} is not a string.", v).into()),
|
||||
}
|
||||
@ -21,9 +21,9 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"to-lower-case".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "string") {
|
||||
match arg!(args, scope, super_selector, 0, "string") {
|
||||
Value::Ident(i, q) => Ok(Value::Ident(i.to_ascii_lowercase(), q)),
|
||||
v => Err(format!("$string: {} is not a string.", v).into()),
|
||||
}
|
||||
@ -31,9 +31,9 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"str-length".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "string") {
|
||||
match arg!(args, scope, super_selector, 0, "string") {
|
||||
Value::Ident(i, _) => Ok(Value::Dimension(
|
||||
Number::from(i.chars().count()),
|
||||
Unit::None,
|
||||
@ -44,9 +44,9 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"quote".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "string") {
|
||||
match arg!(args, scope, super_selector, 0, "string") {
|
||||
Value::Ident(i, _) => Ok(Value::Ident(i, QuoteKind::Double)),
|
||||
v => Err(format!("$string: {} is not a string.", v).into()),
|
||||
}
|
||||
@ -54,9 +54,9 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"unquote".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "string") {
|
||||
match arg!(args, scope, super_selector, 0, "string") {
|
||||
i @ Value::Ident(..) => Ok(i.unquote()),
|
||||
v => Err(format!("$string: {} is not a string.", v).into()),
|
||||
}
|
||||
@ -64,14 +64,14 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"str-slice".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 3);
|
||||
let (string, quotes) = match arg!(args, 0, "string") {
|
||||
let (string, quotes) = match arg!(args, scope, super_selector, 0, "string") {
|
||||
Value::Ident(s, q) => (s, q),
|
||||
v => return Err(format!("$string: {} is not a string.", v).into()),
|
||||
};
|
||||
let str_len = string.chars().count();
|
||||
let start = match arg!(args, 1, "start-at") {
|
||||
let start = match arg!(args, scope, super_selector, 1, "start-at") {
|
||||
Value::Dimension(n, Unit::None) if n.is_decimal() => {
|
||||
return Err(format!("{} is not an int.", n).into())
|
||||
}
|
||||
@ -88,7 +88,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
}
|
||||
v => return Err(format!("$start-at: {} is not a number.", v).into()),
|
||||
};
|
||||
let mut end = match arg!(args, 2, "end-at" = Value::Null) {
|
||||
let mut end = match arg!(args, scope, super_selector, 2, "end-at" = Value::Null) {
|
||||
Value::Dimension(n, Unit::None) if n.is_decimal() => {
|
||||
return Err(format!("{} is not an int.", n).into())
|
||||
}
|
||||
@ -127,14 +127,14 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"str-index".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 2);
|
||||
let s1 = match arg!(args, 0, "string") {
|
||||
let s1 = match arg!(args, scope, super_selector, 0, "string") {
|
||||
Value::Ident(i, _) => i,
|
||||
v => return Err(format!("$string: {} is not a string.", v).into()),
|
||||
};
|
||||
|
||||
let substr = match arg!(args, 1, "substring") {
|
||||
let substr = match arg!(args, scope, super_selector, 1, "substring") {
|
||||
Value::Ident(i, _) => i,
|
||||
v => return Err(format!("$substring: {} is not a string.", v).into()),
|
||||
};
|
||||
@ -147,19 +147,19 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"str-insert".to_owned(),
|
||||
Builtin::new(|mut args, _| {
|
||||
Builtin::new(|mut args, scope, super_selector| {
|
||||
max_args!(args, 3);
|
||||
let (s1, quotes) = match arg!(args, 0, "string") {
|
||||
let (s1, quotes) = match arg!(args, scope, super_selector, 0, "string") {
|
||||
Value::Ident(i, q) => (i, q.normalize()),
|
||||
v => return Err(format!("$string: {} is not a string.", v).into()),
|
||||
};
|
||||
|
||||
let substr = match arg!(args, 1, "insert") {
|
||||
let substr = match arg!(args, scope, super_selector, 1, "insert") {
|
||||
Value::Ident(i, _) => i,
|
||||
v => return Err(format!("$insert: {} is not a string.", v).into()),
|
||||
};
|
||||
|
||||
let index = match arg!(args, 2, "index") {
|
||||
let index = match arg!(args, scope, super_selector, 2, "index") {
|
||||
Value::Dimension(n, Unit::None) if n.is_decimal() => {
|
||||
return Err(format!("$index: {} is not an int.", n).into())
|
||||
}
|
||||
|
@ -29,11 +29,19 @@ impl SassFunction {
|
||||
}
|
||||
}
|
||||
|
||||
pub fn call(self, args: CallArgs, scope: &Scope) -> SassResult<Value> {
|
||||
pub fn call(
|
||||
self,
|
||||
args: CallArgs,
|
||||
scope: &Scope,
|
||||
super_selector: &Selector,
|
||||
) -> SassResult<Value> {
|
||||
match self {
|
||||
Self::Builtin(f, ..) => f.0(args, scope),
|
||||
Self::Builtin(f, ..) => f.0(args, scope, super_selector),
|
||||
// todo: superselector
|
||||
Self::UserDefined(f, ..) => f.clone().args(args)?.call(&Selector::new(), f.body()),
|
||||
Self::UserDefined(f, ..) => f
|
||||
.clone()
|
||||
.args(args, scope, super_selector)?
|
||||
.call(&Selector::new(), f.body()),
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -371,6 +371,7 @@ impl Value {
|
||||
return Ok(IntermediateValue::Value(f.0(
|
||||
eat_call_args(toks, scope, super_selector)?,
|
||||
scope,
|
||||
super_selector,
|
||||
)?))
|
||||
}
|
||||
None => {
|
||||
@ -408,7 +409,11 @@ impl Value {
|
||||
};
|
||||
Ok(IntermediateValue::Value(
|
||||
func.clone()
|
||||
.args(eat_call_args(toks, scope, super_selector)?)?
|
||||
.args(
|
||||
eat_call_args(toks, scope, super_selector)?,
|
||||
scope,
|
||||
super_selector,
|
||||
)?
|
||||
.call(super_selector, func.body())?,
|
||||
))
|
||||
}
|
||||
|
@ -25,3 +25,8 @@ test!(
|
||||
"@function foo($a...) {\n @return $a;\n}\n\na {\n color: foo(1, 2, 3, 4, 5);\n}\n",
|
||||
"a {\n color: 1, 2, 3, 4, 5;\n}\n"
|
||||
);
|
||||
test!(
|
||||
default_args_are_lazily_evaluated,
|
||||
"$da: a;\n\n@mixin foo($x: $da) {\n color: $x;\n}\n$da: b;\n\na {\n @include foo();\n}\n",
|
||||
"a {\n color: b;\n}\n"
|
||||
);
|
||||
|
Loading…
x
Reference in New Issue
Block a user