support $keys...
argument for map.get
(#83)
per the sass-lang docs [1], the user should be able to invoke `map.get($my-map, "key1", "key2")` to perform a nested lookup of the two keys. the current implementation fails if provided more than two arguments to `map.get`. this change implements the nested get. fixes #80. [1] https://sass-lang.com/documentation/modules/map/
This commit is contained in:
parent
346b8c127b
commit
beb64abac4
@ -233,10 +233,14 @@ impl ArgumentResult {
|
|||||||
pub(crate) fn min_args(&self, min: usize) -> SassResult<()> {
|
pub(crate) fn min_args(&self, min: usize) -> SassResult<()> {
|
||||||
let len = self.len();
|
let len = self.len();
|
||||||
if len < min {
|
if len < min {
|
||||||
if min == 1 {
|
let phrase = match min {
|
||||||
return Err(("At least one argument must be passed.", self.span()).into());
|
1 => "one argument",
|
||||||
}
|
2 => "two arguments",
|
||||||
todo!("min args greater than one")
|
3 => "three arguments",
|
||||||
|
_ => todo!("min args greater than three"),
|
||||||
|
};
|
||||||
|
|
||||||
|
return Err((format!("At least {phrase} must be passed."), self.span()).into());
|
||||||
}
|
}
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
@ -1,12 +1,43 @@
|
|||||||
use crate::builtin::builtin_imports::*;
|
use crate::builtin::builtin_imports::*;
|
||||||
|
|
||||||
|
/// map.get($map, $key, $keys...)
|
||||||
|
/// map-get($map, $key, $keys...)
|
||||||
|
///
|
||||||
|
/// If $keys is empty, returns the value in $map associated with $key.
|
||||||
|
/// If $map doesn’t have a value associated with $key, returns null.
|
||||||
|
/// If $keys is not empty, follows the set of keys including $key and
|
||||||
|
/// excluding the last key in $keys, from left to right, to find the
|
||||||
|
/// nested map targeted for searching.
|
||||||
|
/// Returns the value in the targeted map associated with the last key
|
||||||
|
/// in $keys.
|
||||||
|
/// Returns null if the map does not have a value associated with the
|
||||||
|
/// key, or if any key in $keys is missing from a map or references a
|
||||||
|
/// value that is not a map.
|
||||||
|
///
|
||||||
|
/// https://sass-lang.com/documentation/modules/map/
|
||||||
pub(crate) fn map_get(mut args: ArgumentResult, visitor: &mut Visitor) -> SassResult<Value> {
|
pub(crate) fn map_get(mut args: ArgumentResult, visitor: &mut Visitor) -> SassResult<Value> {
|
||||||
args.max_args(2)?;
|
|
||||||
let key = args.get_err(1, "key")?;
|
let key = args.get_err(1, "key")?;
|
||||||
let map = args
|
let map = args
|
||||||
.get_err(0, "map")?
|
.get_err(0, "map")?
|
||||||
.assert_map_with_name("map", args.span())?;
|
.assert_map_with_name("map", args.span())?;
|
||||||
Ok(map.get(&key).unwrap_or(Value::Null))
|
|
||||||
|
// since we already extracted the map and first key,
|
||||||
|
// neither will be returned in the variadic args list
|
||||||
|
let keys = args.get_variadic()?;
|
||||||
|
|
||||||
|
let mut val = map.get(&key).unwrap_or(Value::Null);
|
||||||
|
for key in keys {
|
||||||
|
// if at any point we find a value that's not a map,
|
||||||
|
// we return null
|
||||||
|
let val_map = match val.try_map() {
|
||||||
|
Some(val_map) => val_map,
|
||||||
|
None => return Ok(Value::Null),
|
||||||
|
};
|
||||||
|
|
||||||
|
val = val_map.get(&key).unwrap_or(Value::Null);
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(val)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) fn map_has_key(mut args: ArgumentResult, visitor: &mut Visitor) -> SassResult<Value> {
|
pub(crate) fn map_has_key(mut args: ArgumentResult, visitor: &mut Visitor) -> SassResult<Value> {
|
||||||
|
@ -5,6 +5,11 @@ test!(
|
|||||||
"a {\n color: map-get((a: b), a);\n}\n",
|
"a {\n color: map-get((a: b), a);\n}\n",
|
||||||
"a {\n color: b;\n}\n"
|
"a {\n color: b;\n}\n"
|
||||||
);
|
);
|
||||||
|
test!(
|
||||||
|
map_get_key_exists_named,
|
||||||
|
"a {\n color: map-get($map: (a: b), $key: a);\n}\n",
|
||||||
|
"a {\n color: b;\n}\n"
|
||||||
|
);
|
||||||
test!(
|
test!(
|
||||||
map_get_key_does_not_exist,
|
map_get_key_does_not_exist,
|
||||||
"a {\n color: map-get((a: b), foo);\n}\n",
|
"a {\n color: map-get((a: b), foo);\n}\n",
|
||||||
@ -19,6 +24,36 @@ error!(
|
|||||||
map_get_non_map,
|
map_get_non_map,
|
||||||
"a {\n color: map-get(foo, foo);\n}\n", "Error: $map: foo is not a map."
|
"a {\n color: map-get(foo, foo);\n}\n", "Error: $map: foo is not a map."
|
||||||
);
|
);
|
||||||
|
test!(
|
||||||
|
map_get_nested,
|
||||||
|
"a {\n color: map-get((a: (b: (c: d))), a, b, c);\n}\n",
|
||||||
|
"a {\n color: d;\n}\n"
|
||||||
|
);
|
||||||
|
// it's an odd thing to do, but the spec suggests that the user
|
||||||
|
// can call the function like:
|
||||||
|
// map.get("key2", "key3", $map: $my-map, $key: "key1")
|
||||||
|
// in this case we are to use the named argument $key as the
|
||||||
|
// first key and use the positional arguments at the front as
|
||||||
|
// $keys. this test verifies this behavior.
|
||||||
|
test!(
|
||||||
|
map_get_nested_named_and_positional,
|
||||||
|
"a {\n color: map-get(b, c, $map: (a: (b: (c: d))), $key: a);\n}\n",
|
||||||
|
"a {\n color: d;\n}\n"
|
||||||
|
);
|
||||||
|
test!(
|
||||||
|
map_get_nested_key_does_not_exist,
|
||||||
|
"a {\n color: map-get((a: (b: (c: d))), a, d, e, f);\n}\n",
|
||||||
|
""
|
||||||
|
);
|
||||||
|
test!(
|
||||||
|
map_get_nested_non_map,
|
||||||
|
"a {\n color: map-get((a: (b: c)), a, b, c, d);\n}\n",
|
||||||
|
""
|
||||||
|
);
|
||||||
|
error!(
|
||||||
|
map_get_no_args,
|
||||||
|
"a {\n color: map-get();\n}\n", "Error: Missing argument $key."
|
||||||
|
);
|
||||||
error!(
|
error!(
|
||||||
map_get_one_arg,
|
map_get_one_arg,
|
||||||
"a {\n color: map-get(1);\n}\n", "Error: Missing argument $key."
|
"a {\n color: map-get(1);\n}\n", "Error: Missing argument $key."
|
||||||
|
Loading…
x
Reference in New Issue
Block a user