grass/src/builtin/list.rs

334 lines
11 KiB
Rust
Raw Normal View History

use std::collections::HashMap;
2020-02-02 21:09:29 -05:00
2020-03-22 22:28:54 -04:00
use num_traits::{One, Signed, ToPrimitive, Zero};
2020-03-20 12:09:08 -04:00
2020-02-14 10:10:51 -05:00
use super::Builtin;
2020-04-30 14:47:07 -04:00
use crate::args::CallArgs;
use crate::common::{Brackets, ListSeparator, QuoteKind};
2020-04-13 12:28:40 -04:00
use crate::error::SassResult;
2020-04-30 14:47:07 -04:00
use crate::scope::Scope;
use crate::selector::Selector;
use crate::unit::Unit;
2020-02-14 10:10:51 -05:00
use crate::value::{Number, Value};
pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
2020-04-30 14:47:07 -04:00
fn length(mut args: CallArgs, scope: &Scope, super_selector: &Selector) -> SassResult<Value> {
args.max_args(1)?;
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(),
};
Ok(Value::Dimension(len, Unit::None))
}
2020-03-20 12:09:08 -04:00
2020-04-30 14:47:07 -04:00
fn nth(mut args: CallArgs, scope: &Scope, super_selector: &Selector) -> SassResult<Value> {
args.max_args(2)?;
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, scope, super_selector, 1, "n") {
Value::Dimension(num, _) => num,
v => {
2020-04-12 19:37:12 -04:00
return Err((
2020-04-30 14:47:07 -04:00
format!("$n: {} is not a number.", v.to_css_string(args.span())?),
2020-04-12 19:37:12 -04:00
args.span(),
2020-03-20 12:09:08 -04:00
)
2020-04-30 14:47:07 -04:00
.into())
2020-03-20 12:09:08 -04:00
}
2020-04-30 14:47:07 -04:00
};
2020-03-20 12:09:08 -04:00
2020-04-30 14:47:07 -04:00
if n.is_zero() {
return Err(("$n: List index may not be 0.", args.span()).into());
}
2020-03-20 12:09:08 -04:00
2020-04-30 14:47:07 -04:00
if n.abs() > Number::from(list.len()) {
return Err((
format!(
"$n: Invalid index {} for a list with {} elements.",
n,
list.len()
),
args.span(),
)
.into());
}
2020-03-20 19:27:26 -04:00
2020-04-30 14:47:07 -04:00
if n.is_decimal() {
return Err((format!("$n: {} is not an int.", n), args.span()).into());
}
if n.is_positive() {
Ok(list[n.to_integer().to_usize().unwrap() - 1].clone())
} else {
Ok(list[list.len() - n.abs().to_integer().to_usize().unwrap()].clone())
}
}
2020-03-20 19:27:26 -04:00
2020-04-30 14:47:07 -04:00
fn list_separator(
mut args: CallArgs,
scope: &Scope,
super_selector: &Selector,
) -> SassResult<Value> {
args.max_args(1)?;
Ok(Value::Ident(
match arg!(args, scope, super_selector, 0, "list") {
Value::List(_, sep, ..) => sep.name(),
_ => ListSeparator::Space.name(),
}
.to_owned(),
QuoteKind::None,
))
}
2020-03-20 19:27:26 -04:00
2020-04-30 14:47:07 -04:00
fn set_nth(mut args: CallArgs, scope: &Scope, super_selector: &Selector) -> SassResult<Value> {
args.max_args(3)?;
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, scope, super_selector, 1, "n") {
Value::Dimension(num, _) => num,
v => {
2020-04-12 19:37:12 -04:00
return Err((
2020-04-30 14:47:07 -04:00
format!("$n: {} is not a number.", v.to_css_string(args.span())?),
2020-04-12 19:37:12 -04:00
args.span(),
)
2020-04-30 14:47:07 -04:00
.into())
2020-03-20 19:27:26 -04:00
}
2020-04-30 14:47:07 -04:00
};
2020-03-20 19:27:26 -04:00
2020-04-30 14:47:07 -04:00
if n.is_zero() {
return Err(("$n: List index may not be 0.", args.span()).into());
}
2020-03-20 19:27:26 -04:00
2020-04-30 14:47:07 -04:00
let len = list.len();
2020-03-20 19:27:26 -04:00
2020-04-30 14:47:07 -04:00
if n.abs() > Number::from(len) {
return Err((
format!("$n: Invalid index {} for a list with {} elements.", n, len),
args.span(),
)
.into());
}
2020-03-20 19:27:26 -04:00
2020-04-30 14:47:07 -04:00
if n.is_decimal() {
return Err((format!("$n: {} is not an int.", n), args.span()).into());
}
let val = arg!(args, scope, super_selector, 2, "value");
if n.is_positive() {
list[n.to_integer().to_usize().unwrap() - 1] = val;
} else {
list[len - n.abs().to_integer().to_usize().unwrap()] = val;
}
Ok(Value::List(list, sep, brackets))
}
fn append(mut args: CallArgs, scope: &Scope, super_selector: &Selector) -> SassResult<Value> {
args.max_args(3)?;
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, scope, super_selector, 1, "val");
let sep = match arg!(
args,
scope,
super_selector,
2,
"separator" = Value::Ident("auto".to_owned(), QuoteKind::None)
) {
Value::Ident(s, ..) => match s.as_str() {
"auto" => sep,
"comma" => ListSeparator::Comma,
"space" => ListSeparator::Space,
_ => {
2020-04-12 19:37:12 -04:00
return Err((
2020-04-30 14:47:07 -04:00
"$separator: Must be \"space\", \"comma\", or \"auto\".",
2020-04-12 19:37:12 -04:00
args.span(),
)
.into())
}
2020-04-30 14:47:07 -04:00
},
v => {
return Err((
format!(
"$separator: {} is not a string.",
v.to_css_string(args.span())?
),
args.span(),
)
.into())
}
};
list.push(val);
Ok(Value::List(list, sep, brackets))
}
fn join(mut args: CallArgs, scope: &Scope, super_selector: &Selector) -> SassResult<Value> {
args.max_args(4)?;
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, 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)
) {
Value::Ident(s, ..) => match s.as_str() {
"auto" => {
if list1.is_empty() || (list1.len() == 1 && sep1 == ListSeparator::Space) {
sep2
} else {
sep1
2020-03-22 14:04:16 -04:00
}
2020-04-30 14:47:07 -04:00
}
"comma" => ListSeparator::Comma,
"space" => ListSeparator::Space,
_ => {
2020-04-12 19:37:12 -04:00
return Err((
2020-04-30 14:47:07 -04:00
"$separator: Must be \"space\", \"comma\", or \"auto\".",
2020-04-12 19:37:12 -04:00
args.span(),
)
.into())
}
2020-04-30 14:47:07 -04:00
},
v => {
return Err((
format!(
"$separator: {} is not a string.",
v.to_css_string(args.span())?
),
args.span(),
)
.into())
}
};
let brackets = match arg!(
args,
scope,
super_selector,
3,
"bracketed" = Value::Ident("auto".to_owned(), QuoteKind::None)
) {
Value::Ident(s, ..) => match s.as_str() {
"auto" => brackets,
_ => Brackets::Bracketed,
},
v => {
if v.is_true(args.span())? {
Brackets::Bracketed
} else {
Brackets::None
2020-04-02 18:04:44 -04:00
}
2020-04-30 14:47:07 -04:00
}
};
list1.extend(list2);
Ok(Value::List(list1, sep, brackets))
}
fn is_bracketed(
mut args: CallArgs,
scope: &Scope,
super_selector: &Selector,
) -> SassResult<Value> {
args.max_args(1)?;
Ok(Value::bool(
match arg!(args, scope, super_selector, 0, "list") {
Value::List(.., brackets) => match brackets {
Brackets::Bracketed => true,
Brackets::None => false,
},
2020-04-30 14:47:07 -04:00
_ => false,
},
))
}
fn index(mut args: CallArgs, scope: &Scope, super_selector: &Selector) -> SassResult<Value> {
args.max_args(2)?;
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, 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
// it is still dirty.
// Potential input to fuzz: index(1px 1in 1cm, 96px + 1rem)
let index = match list
.into_iter()
.position(|v| v.equals(value.clone(), args.span()).unwrap())
{
Some(v) => Number::from(v + 1),
None => return Ok(Value::Null),
};
Ok(Value::Dimension(index, Unit::None))
}
fn zip(args: CallArgs, scope: &Scope, super_selector: &Selector) -> SassResult<Value> {
let span = args.span();
let lists = args
.get_variadic(scope, super_selector)?
.into_iter()
.map(|x| {
Ok(match x.node.eval(span)?.node {
Value::List(v, ..) => v,
Value::Map(m) => m.entries(),
v => vec![v],
2020-04-03 15:54:59 -04:00
})
2020-04-30 14:47:07 -04:00
})
.collect::<SassResult<Vec<Vec<Value>>>>()?;
2020-04-03 15:54:59 -04:00
2020-04-30 14:47:07 -04:00
let len = lists.iter().map(Vec::len).min().unwrap_or(0);
2020-04-03 15:54:59 -04:00
2020-04-30 14:47:07 -04:00
if len == 0 {
return Ok(Value::List(
Vec::new(),
ListSeparator::Comma,
Brackets::None,
));
}
2020-04-03 15:54:59 -04:00
2020-04-30 14:47:07 -04:00
let result = (0..len)
.map(|i| {
let items = lists.iter().map(|v| v[i].clone()).collect();
Value::List(items, ListSeparator::Space, Brackets::None)
})
.collect();
Ok(Value::List(result, ListSeparator::Comma, Brackets::None))
}
2020-04-03 15:54:59 -04:00
2020-04-30 14:47:07 -04:00
f.insert("length".to_owned(), Builtin::new(length));
f.insert("nth".to_owned(), Builtin::new(nth));
f.insert("list-separator".to_owned(), Builtin::new(list_separator));
f.insert("set-nth".to_owned(), Builtin::new(set_nth));
f.insert("append".to_owned(), Builtin::new(append));
f.insert("join".to_owned(), Builtin::new(join));
f.insert("is-bracketed".to_owned(), Builtin::new(is_bracketed));
f.insert("index".to_owned(), Builtin::new(index));
f.insert("zip".to_owned(), Builtin::new(zip));
2020-02-14 10:10:51 -05:00
}