handle builtin functions in get-function
This commit is contained in:
parent
3f98d1abca
commit
81c85a6f86
@ -11,7 +11,7 @@ use crate::value::{Number, Value};
|
||||
pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
f.insert(
|
||||
"hsl".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
if args.is_empty() {
|
||||
return Err("Missing argument $channels.".into());
|
||||
}
|
||||
@ -89,7 +89,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"hsla".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
if args.is_empty() {
|
||||
return Err("Missing argument $channels.".into());
|
||||
}
|
||||
@ -167,7 +167,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"hue".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "color") {
|
||||
Value::Color(c) => Ok(Value::Dimension(c.hue(), Unit::Deg)),
|
||||
@ -177,7 +177,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"saturation".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "color") {
|
||||
Value::Color(c) => Ok(Value::Dimension(c.saturation(), Unit::Percent)),
|
||||
@ -187,7 +187,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"lightness".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "color") {
|
||||
Value::Color(c) => Ok(Value::Dimension(c.lightness(), Unit::Percent)),
|
||||
@ -197,7 +197,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"adjust-hue".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 2);
|
||||
let color = match arg!(args, 0, "color") {
|
||||
Value::Color(c) => c,
|
||||
@ -212,7 +212,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"lighten".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 2);
|
||||
let color = match arg!(args, 0, "color") {
|
||||
Value::Color(c) => c,
|
||||
@ -227,7 +227,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"darken".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 2);
|
||||
let color = match arg!(args, 0, "color") {
|
||||
Value::Color(c) => c,
|
||||
@ -242,7 +242,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"saturate".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 2);
|
||||
if args.len() == 1 {
|
||||
return Ok(Value::Ident(
|
||||
@ -270,7 +270,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"desaturate".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 2);
|
||||
let color = match arg!(args, 0, "color") {
|
||||
Value::Color(c) => c,
|
||||
@ -285,7 +285,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"grayscale".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 1);
|
||||
let color = match arg!(args, 0, "color") {
|
||||
Value::Color(c) => c,
|
||||
@ -302,7 +302,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"complement".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 1);
|
||||
let color = match arg!(args, 0, "color") {
|
||||
Value::Color(c) => c,
|
||||
@ -313,7 +313,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"invert".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 2);
|
||||
let weight = match arg!(
|
||||
args,
|
||||
|
@ -9,7 +9,7 @@ use crate::value::Value;
|
||||
pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
f.insert(
|
||||
"alpha".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "color") {
|
||||
Value::Color(c) => Ok(Value::Dimension(c.alpha(), Unit::None)),
|
||||
@ -19,7 +19,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"opacity".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "color") {
|
||||
Value::Color(c) => Ok(Value::Dimension(c.alpha(), Unit::None)),
|
||||
@ -33,7 +33,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"opacify".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 2);
|
||||
let color = match arg!(args, 0, "color") {
|
||||
Value::Color(c) => c,
|
||||
@ -48,7 +48,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"fade-in".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 2);
|
||||
let color = match arg!(args, 0, "color") {
|
||||
Value::Color(c) => c,
|
||||
@ -63,7 +63,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"transparentize".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 2);
|
||||
let color = match arg!(args, 0, "color") {
|
||||
Value::Color(c) => c,
|
||||
@ -78,7 +78,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"fade-out".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 2);
|
||||
let color = match arg!(args, 0, "color") {
|
||||
Value::Color(c) => c,
|
||||
|
@ -31,7 +31,7 @@ macro_rules! opt_hsl {
|
||||
}
|
||||
|
||||
pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
f.insert("change-color".to_owned(), Box::new(|mut args, _| {
|
||||
f.insert("change-color".to_owned(), Builtin::new(|mut args, _| {
|
||||
if args.get_positional(1).is_some() {
|
||||
return Err("Only one positional argument is allowed. All other arguments must be passed by name.".into());
|
||||
}
|
||||
@ -73,7 +73,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
}));
|
||||
f.insert(
|
||||
"adjust-color".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
let color = match arg!(args, 0, "color") {
|
||||
Value::Color(c) => c,
|
||||
v => return Err(format!("$color: {} is not a color.", v).into()),
|
||||
@ -123,7 +123,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"scale-color".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
let color = match arg!(args, 0, "color") {
|
||||
Value::Color(c) => c,
|
||||
v => return Err(format!("$color: {} is not a color.", v).into()),
|
||||
@ -209,7 +209,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"ie-hex-str".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 1);
|
||||
let color = match arg!(args, 0, "color") {
|
||||
Value::Color(c) => c,
|
||||
|
@ -10,7 +10,7 @@ use crate::value::{Number, Value};
|
||||
pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
f.insert(
|
||||
"rgb".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
if args.is_empty() {
|
||||
return Err("Missing argument $channels.".into());
|
||||
}
|
||||
@ -132,7 +132,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"rgba".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
if args.is_empty() {
|
||||
return Err("Missing argument $channels.".into());
|
||||
}
|
||||
@ -254,7 +254,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"red".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "color") {
|
||||
Value::Color(c) => Ok(Value::Dimension(c.red(), Unit::None)),
|
||||
@ -264,7 +264,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"green".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "color") {
|
||||
Value::Color(c) => Ok(Value::Dimension(c.green(), Unit::None)),
|
||||
@ -274,7 +274,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"blue".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "color") {
|
||||
Value::Color(c) => Ok(Value::Dimension(c.blue(), Unit::None)),
|
||||
@ -284,7 +284,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"mix".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 3);
|
||||
let color1 = match arg!(args, 0, "color1") {
|
||||
Value::Color(c) => c,
|
||||
|
@ -10,7 +10,7 @@ use crate::value::{Number, Value};
|
||||
pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
f.insert(
|
||||
"length".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 1);
|
||||
let len = match arg!(args, 0, "list") {
|
||||
Value::List(v, ..) => Number::from(v.len()),
|
||||
@ -22,7 +22,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"nth".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 2);
|
||||
let list = match arg!(args, 0, "list") {
|
||||
Value::List(v, ..) => v,
|
||||
@ -60,7 +60,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"list-separator".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 1);
|
||||
Ok(Value::Ident(
|
||||
match arg!(args, 0, "list") {
|
||||
@ -74,7 +74,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"set-nth".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 3);
|
||||
let (mut list, sep, brackets) = match arg!(args, 0, "list") {
|
||||
Value::List(v, sep, b) => (v, sep, b),
|
||||
@ -115,7 +115,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"append".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 3);
|
||||
let (mut list, sep, brackets) = match arg!(args, 0, "list") {
|
||||
Value::List(v, sep, b) => (v, sep, b),
|
||||
@ -145,7 +145,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"join".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 4);
|
||||
let (mut list1, sep1, brackets) = match arg!(args, 0, "list1") {
|
||||
Value::List(v, sep, brackets) => (v, sep, brackets),
|
||||
@ -204,7 +204,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"is-bracketed".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 1);
|
||||
Ok(Value::bool(match arg!(args, 0, "list") {
|
||||
Value::List(.., brackets) => match brackets {
|
||||
@ -217,7 +217,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"index".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 2);
|
||||
let list = match arg!(args, 0, "list") {
|
||||
Value::List(v, ..) => v,
|
||||
@ -242,7 +242,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"zip".to_owned(),
|
||||
Box::new(|args, _| {
|
||||
Builtin::new(|args, _| {
|
||||
let lists = args
|
||||
.get_variadic()?
|
||||
.into_iter()
|
||||
|
@ -7,7 +7,7 @@ use crate::value::{SassMap, Value};
|
||||
pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
f.insert(
|
||||
"map-get".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 2);
|
||||
let key = arg!(args, 1, "key");
|
||||
let map = match arg!(args, 0, "map") {
|
||||
@ -20,7 +20,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"map-has-key".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 2);
|
||||
let key = arg!(args, 1, "key");
|
||||
let map = match arg!(args, 0, "map") {
|
||||
@ -33,7 +33,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"map-keys".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 1);
|
||||
let map = match arg!(args, 0, "map") {
|
||||
Value::Map(m) => m,
|
||||
@ -49,7 +49,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"map-values".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 1);
|
||||
let map = match arg!(args, 0, "map") {
|
||||
Value::Map(m) => m,
|
||||
@ -65,7 +65,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"map-merge".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 2);
|
||||
let mut map1 = match arg!(args, 0, "map1") {
|
||||
Value::Map(m) => m,
|
||||
@ -83,7 +83,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"map-remove".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
let mut map = match arg!(args, 0, "map") {
|
||||
Value::Map(m) => m,
|
||||
Value::List(v, ..) if v.is_empty() => SassMap::new(),
|
||||
|
@ -7,7 +7,7 @@ use crate::value::{Number, Value};
|
||||
pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
f.insert(
|
||||
"percentage".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 1);
|
||||
let num = match arg!(args, 0, "number") {
|
||||
Value::Dimension(n, Unit::None) => n * Number::from(100),
|
||||
@ -21,7 +21,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"round".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "number") {
|
||||
Value::Dimension(n, u) => Ok(Value::Dimension(n.round(), u)),
|
||||
@ -31,7 +31,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"ceil".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "number") {
|
||||
Value::Dimension(n, u) => Ok(Value::Dimension(n.ceil(), u)),
|
||||
@ -41,7 +41,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"floor".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "number") {
|
||||
Value::Dimension(n, u) => Ok(Value::Dimension(n.floor(), u)),
|
||||
@ -51,7 +51,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"abs".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "number") {
|
||||
Value::Dimension(n, u) => Ok(Value::Dimension(n.abs(), u)),
|
||||
@ -61,7 +61,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"comparable".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 2);
|
||||
let unit1 = match arg!(args, 0, "number1") {
|
||||
Value::Dimension(_, u) => u,
|
||||
|
@ -4,12 +4,12 @@ use super::{Builtin, GLOBAL_FUNCTIONS};
|
||||
use crate::common::{Brackets, QuoteKind};
|
||||
use crate::scope::global_var_exists;
|
||||
use crate::unit::Unit;
|
||||
use crate::value::Value;
|
||||
use crate::value::{SassFunction, Value};
|
||||
|
||||
pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
f.insert(
|
||||
"if".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 3);
|
||||
if arg!(args, 0, "condition").is_true()? {
|
||||
Ok(arg!(args, 1, "if-true"))
|
||||
@ -20,7 +20,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"feature-exists".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "feature") {
|
||||
Value::Ident(s, _) => match s.as_str() {
|
||||
@ -47,7 +47,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"unit".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 1);
|
||||
let unit = match arg!(args, 0, "number") {
|
||||
Value::Dimension(_, u) => u.to_string(),
|
||||
@ -58,7 +58,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"type-of".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 1);
|
||||
let value = arg!(args, 0, "value");
|
||||
Ok(Value::Ident(value.kind()?.to_owned(), QuoteKind::None))
|
||||
@ -66,7 +66,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"unitless".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "number") {
|
||||
Value::Dimension(_, Unit::None) => Ok(Value::True),
|
||||
@ -77,7 +77,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"inspect".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 1);
|
||||
Ok(Value::Ident(
|
||||
match arg!(args, 0, "value") {
|
||||
@ -85,6 +85,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
Brackets::None => "()".to_string(),
|
||||
Brackets::Bracketed => "[]".to_string(),
|
||||
},
|
||||
Value::Function(f) => format!("get-function(\"{}\")", f.name()),
|
||||
v => v.to_string(),
|
||||
},
|
||||
QuoteKind::None,
|
||||
@ -93,7 +94,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"variable-exists".to_owned(),
|
||||
Box::new(|mut args, scope| {
|
||||
Builtin::new(|mut args, scope| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "name") {
|
||||
Value::Ident(s, _) => Ok(Value::bool(scope.var_exists(&s))),
|
||||
@ -103,7 +104,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"global-variable-exists".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "name") {
|
||||
Value::Ident(s, _) => Ok(Value::bool(global_var_exists(&s))),
|
||||
@ -113,7 +114,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"mixin-exists".to_owned(),
|
||||
Box::new(|mut args, scope| {
|
||||
Builtin::new(|mut args, scope| {
|
||||
max_args!(args, 2);
|
||||
match arg!(args, 0, "name") {
|
||||
Value::Ident(s, _) => Ok(Value::bool(scope.mixin_exists(&s))),
|
||||
@ -123,7 +124,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"function-exists".to_owned(),
|
||||
Box::new(|mut args, scope| {
|
||||
Builtin::new(|mut args, scope| {
|
||||
max_args!(args, 2);
|
||||
match arg!(args, 0, "name") {
|
||||
Value::Ident(s, _) => Ok(Value::bool(
|
||||
@ -135,7 +136,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"get-function".to_owned(),
|
||||
Box::new(|mut args, scope| {
|
||||
Builtin::new(|mut args, scope| {
|
||||
max_args!(args, 2);
|
||||
let name = match arg!(args, 0, "name") {
|
||||
Value::Ident(s, _) => s,
|
||||
@ -143,10 +144,18 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
};
|
||||
let css = arg!(args, 1, "css" = Value::False).is_true()?;
|
||||
|
||||
Ok(Value::Function(Box::new(scope.get_fn(&name)?), css))
|
||||
let func = match scope.get_fn(&name) {
|
||||
Ok(f) => SassFunction::UserDefined(Box::new(f), name),
|
||||
Err(e) => match GLOBAL_FUNCTIONS.get(&name) {
|
||||
Some(f) => SassFunction::Builtin(f.clone(), name),
|
||||
None => return Err(e),
|
||||
},
|
||||
};
|
||||
|
||||
Ok(Value::Function(func))
|
||||
}),
|
||||
);
|
||||
f.insert("call".to_owned(), Box::new(|_args, _scope| {
|
||||
f.insert("call".to_owned(), Builtin::new(|_args, _scope| {
|
||||
todo!("builtin function `call()` is blocked on refactoring how call args are stored and parsed")
|
||||
// let func = arg!(args, 0, "function").to_string();
|
||||
// let func = match scope.get_fn(&func) {
|
||||
|
@ -1,5 +1,6 @@
|
||||
use once_cell::sync::Lazy;
|
||||
use std::collections::HashMap;
|
||||
use std::sync::atomic::{AtomicUsize, Ordering};
|
||||
|
||||
use crate::args::CallArgs;
|
||||
use crate::error::SassResult;
|
||||
@ -17,7 +18,23 @@ mod meta;
|
||||
mod selector;
|
||||
mod string;
|
||||
|
||||
pub(crate) type Builtin = Box<dyn Fn(CallArgs, &Scope) -> SassResult<Value> + Send + Sync>;
|
||||
static FUNCTION_COUNT: AtomicUsize = AtomicUsize::new(0);
|
||||
|
||||
// TODO: impl Fn
|
||||
#[derive(Clone)]
|
||||
pub(crate) struct Builtin(pub fn(CallArgs, &Scope) -> SassResult<Value>, usize);
|
||||
impl Builtin {
|
||||
pub fn new(body: fn(CallArgs, &Scope) -> SassResult<Value>) -> Builtin {
|
||||
let count = FUNCTION_COUNT.fetch_add(1, Ordering::Relaxed);
|
||||
Self(body, count)
|
||||
}
|
||||
}
|
||||
|
||||
impl PartialEq for Builtin {
|
||||
fn eq(&self, other: &Self) -> bool {
|
||||
self.1 == other.1
|
||||
}
|
||||
}
|
||||
|
||||
pub(crate) static GLOBAL_FUNCTIONS: Lazy<HashMap<String, Builtin>> = Lazy::new(|| {
|
||||
let mut m = HashMap::new();
|
||||
|
@ -11,7 +11,7 @@ use crate::value::{Number, Value};
|
||||
pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
f.insert(
|
||||
"to-upper-case".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "string") {
|
||||
Value::Ident(i, q) => Ok(Value::Ident(i.to_ascii_uppercase(), q)),
|
||||
@ -21,7 +21,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"to-lower-case".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "string") {
|
||||
Value::Ident(i, q) => Ok(Value::Ident(i.to_ascii_lowercase(), q)),
|
||||
@ -31,7 +31,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"str-length".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "string") {
|
||||
Value::Ident(i, _) => Ok(Value::Dimension(
|
||||
@ -44,7 +44,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"quote".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "string") {
|
||||
Value::Ident(i, _) => Ok(Value::Ident(i, QuoteKind::Double)),
|
||||
@ -54,7 +54,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"unquote".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 1);
|
||||
match arg!(args, 0, "string") {
|
||||
i @ Value::Ident(..) => Ok(i.unquote()),
|
||||
@ -64,7 +64,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"str-slice".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 3);
|
||||
let (string, quotes) = match arg!(args, 0, "string") {
|
||||
Value::Ident(s, q) => (s, q),
|
||||
@ -127,7 +127,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"str-index".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 2);
|
||||
let s1 = match arg!(args, 0, "string") {
|
||||
Value::Ident(i, _) => i,
|
||||
@ -147,7 +147,7 @@ pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||
);
|
||||
f.insert(
|
||||
"str-insert".to_owned(),
|
||||
Box::new(|mut args, _| {
|
||||
Builtin::new(|mut args, _| {
|
||||
max_args!(args, 3);
|
||||
let (s1, quotes) = match arg!(args, 0, "string") {
|
||||
Value::Ident(i, q) => (i, q.normalize()),
|
||||
|
42
src/value/function.rs
Normal file
42
src/value/function.rs
Normal file
@ -0,0 +1,42 @@
|
||||
use std::fmt;
|
||||
|
||||
use crate::atrule::Function;
|
||||
use crate::builtin::Builtin;
|
||||
|
||||
#[derive(Clone)]
|
||||
pub(crate) enum SassFunction {
|
||||
Builtin(Builtin, String),
|
||||
UserDefined(Box<Function>, String),
|
||||
}
|
||||
|
||||
impl SassFunction {
|
||||
pub fn name(&self) -> &str {
|
||||
match self {
|
||||
Self::Builtin(_, name) => name,
|
||||
Self::UserDefined(_, name) => name,
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl fmt::Debug for SassFunction {
|
||||
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
||||
todo!()
|
||||
}
|
||||
}
|
||||
|
||||
impl PartialEq for SassFunction {
|
||||
fn eq(&self, other: &Self) -> bool {
|
||||
match self {
|
||||
Self::UserDefined(f, ..) => match other {
|
||||
Self::UserDefined(f2, ..) => f == f2,
|
||||
Self::Builtin(..) => false,
|
||||
},
|
||||
Self::Builtin(f, ..) => match other {
|
||||
Self::UserDefined(..) => false,
|
||||
Self::Builtin(f2, ..) => f == f2,
|
||||
},
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl Eq for SassFunction {}
|
@ -2,15 +2,16 @@ use std::cmp::Ordering;
|
||||
use std::fmt::{self, Display, Write};
|
||||
use std::iter::Iterator;
|
||||
|
||||
use crate::atrule::Function;
|
||||
use crate::color::Color;
|
||||
use crate::common::{Brackets, ListSeparator, Op, QuoteKind};
|
||||
use crate::error::SassResult;
|
||||
use crate::unit::{Unit, UNIT_CONVERSION_TABLE};
|
||||
|
||||
pub(crate) use function::SassFunction;
|
||||
pub(crate) use map::SassMap;
|
||||
pub(crate) use number::Number;
|
||||
|
||||
mod function;
|
||||
mod map;
|
||||
mod number;
|
||||
mod ops;
|
||||
@ -32,7 +33,7 @@ pub(crate) enum Value {
|
||||
Map(SassMap),
|
||||
ArgList(Vec<Value>),
|
||||
/// Returned by `get-function()`
|
||||
Function(Box<Function>, bool),
|
||||
Function(SassFunction),
|
||||
}
|
||||
|
||||
impl Display for Value {
|
||||
|
@ -368,7 +368,7 @@ impl Value {
|
||||
Ok(f) => f,
|
||||
Err(_) => match GLOBAL_FUNCTIONS.get(&s) {
|
||||
Some(f) => {
|
||||
return Ok(IntermediateValue::Value(f(
|
||||
return Ok(IntermediateValue::Value(f.0(
|
||||
eat_call_args(toks, scope, super_selector)?,
|
||||
scope,
|
||||
)?))
|
||||
|
@ -32,3 +32,23 @@ test!(
|
||||
a {b: type-of(get-function(user-defined));}",
|
||||
"a {\n b: function;\n}\n"
|
||||
);
|
||||
test!(
|
||||
type_of_builtin_function,
|
||||
"a {b: type-of(get-function(lighten));}",
|
||||
"a {\n b: function;\n}\n"
|
||||
);
|
||||
test!(
|
||||
same_builtin_function_is_equal,
|
||||
"a {b: get-function(lighten) == get-function(lighten);}",
|
||||
"a {\n b: true;\n}\n"
|
||||
);
|
||||
test!(
|
||||
different_builtin_function_not_equal,
|
||||
"a {b: get-function(lighten) == get-function(darken);}",
|
||||
"a {\n b: false;\n}\n"
|
||||
);
|
||||
test!(
|
||||
inspect_builtin_function,
|
||||
"a {b: inspect(get-function(lighten));}",
|
||||
"a {\n b: get-function(\"lighten\");\n}\n"
|
||||
);
|
||||
|
Loading…
x
Reference in New Issue
Block a user