simplify declaration of string fns
This commit is contained in:
parent
26aabb42ad
commit
22098ca684
@ -29,6 +29,7 @@ pub(crate) struct Builtin(
|
|||||||
pub fn(CallArgs, &Scope, &Selector) -> SassResult<Value>,
|
pub fn(CallArgs, &Scope, &Selector) -> SassResult<Value>,
|
||||||
usize,
|
usize,
|
||||||
);
|
);
|
||||||
|
|
||||||
impl Builtin {
|
impl Builtin {
|
||||||
pub fn new(body: fn(CallArgs, &Scope, &Selector) -> SassResult<Value>) -> Builtin {
|
pub fn new(body: fn(CallArgs, &Scope, &Selector) -> SassResult<Value>) -> Builtin {
|
||||||
let count = FUNCTION_COUNT.fetch_add(1, Ordering::Relaxed);
|
let count = FUNCTION_COUNT.fetch_add(1, Ordering::Relaxed);
|
||||||
|
@ -7,361 +7,381 @@ use num_traits::{Signed, ToPrimitive, Zero};
|
|||||||
use rand::{distributions::Alphanumeric, thread_rng, Rng};
|
use rand::{distributions::Alphanumeric, thread_rng, Rng};
|
||||||
|
|
||||||
use super::Builtin;
|
use super::Builtin;
|
||||||
|
use crate::args::CallArgs;
|
||||||
use crate::common::QuoteKind;
|
use crate::common::QuoteKind;
|
||||||
|
use crate::error::SassResult;
|
||||||
|
use crate::scope::Scope;
|
||||||
|
use crate::selector::Selector;
|
||||||
use crate::unit::Unit;
|
use crate::unit::Unit;
|
||||||
use crate::value::{Number, Value};
|
use crate::value::{Number, Value};
|
||||||
|
|
||||||
pub(crate) fn register(f: &mut GlobalFunctionMap) {
|
pub(crate) fn register(f: &mut GlobalFunctionMap) {
|
||||||
f.insert(
|
fn to_upper_case(
|
||||||
"to-upper-case",
|
mut args: CallArgs,
|
||||||
Builtin::new(|mut args, scope, super_selector| {
|
scope: &Scope,
|
||||||
args.max_args(1)?;
|
super_selector: &Selector,
|
||||||
match arg!(args, scope, super_selector, 0, "string") {
|
) -> SassResult<Value> {
|
||||||
Value::Ident(mut i, q) => {
|
args.max_args(1)?;
|
||||||
i.make_ascii_uppercase();
|
match arg!(args, scope, super_selector, 0, "string") {
|
||||||
Ok(Value::Ident(i, q))
|
Value::Ident(mut i, q) => {
|
||||||
}
|
i.make_ascii_uppercase();
|
||||||
v => Err((
|
Ok(Value::Ident(i, q))
|
||||||
|
}
|
||||||
|
v => Err((
|
||||||
|
format!(
|
||||||
|
"$string: {} is not a string.",
|
||||||
|
v.to_css_string(args.span())?
|
||||||
|
),
|
||||||
|
args.span(),
|
||||||
|
)
|
||||||
|
.into()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn to_lower_case(
|
||||||
|
mut args: CallArgs,
|
||||||
|
scope: &Scope,
|
||||||
|
super_selector: &Selector,
|
||||||
|
) -> SassResult<Value> {
|
||||||
|
args.max_args(1)?;
|
||||||
|
match arg!(args, scope, super_selector, 0, "string") {
|
||||||
|
Value::Ident(mut i, q) => {
|
||||||
|
i.make_ascii_lowercase();
|
||||||
|
Ok(Value::Ident(i, q))
|
||||||
|
}
|
||||||
|
v => Err((
|
||||||
|
format!(
|
||||||
|
"$string: {} is not a string.",
|
||||||
|
v.to_css_string(args.span())?
|
||||||
|
),
|
||||||
|
args.span(),
|
||||||
|
)
|
||||||
|
.into()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn str_length(
|
||||||
|
mut args: CallArgs,
|
||||||
|
scope: &Scope,
|
||||||
|
super_selector: &Selector,
|
||||||
|
) -> SassResult<Value> {
|
||||||
|
args.max_args(1)?;
|
||||||
|
match arg!(args, scope, super_selector, 0, "string") {
|
||||||
|
Value::Ident(i, _) => Ok(Value::Dimension(
|
||||||
|
Number::from(i.chars().count()),
|
||||||
|
Unit::None,
|
||||||
|
)),
|
||||||
|
v => Err((
|
||||||
|
format!(
|
||||||
|
"$string: {} is not a string.",
|
||||||
|
v.to_css_string(args.span())?
|
||||||
|
),
|
||||||
|
args.span(),
|
||||||
|
)
|
||||||
|
.into()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn quote(mut args: CallArgs, scope: &Scope, super_selector: &Selector) -> SassResult<Value> {
|
||||||
|
args.max_args(1)?;
|
||||||
|
match arg!(args, scope, super_selector, 0, "string") {
|
||||||
|
Value::Ident(i, _) => Ok(Value::Ident(i, QuoteKind::Quoted)),
|
||||||
|
v => Err((
|
||||||
|
format!(
|
||||||
|
"$string: {} is not a string.",
|
||||||
|
v.to_css_string(args.span())?
|
||||||
|
),
|
||||||
|
args.span(),
|
||||||
|
)
|
||||||
|
.into()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn unquote(mut args: CallArgs, scope: &Scope, super_selector: &Selector) -> SassResult<Value> {
|
||||||
|
args.max_args(1)?;
|
||||||
|
match arg!(args, scope, super_selector, 0, "string") {
|
||||||
|
i @ Value::Ident(..) => Ok(i.unquote()),
|
||||||
|
v => Err((
|
||||||
|
format!(
|
||||||
|
"$string: {} is not a string.",
|
||||||
|
v.to_css_string(args.span())?
|
||||||
|
),
|
||||||
|
args.span(),
|
||||||
|
)
|
||||||
|
.into()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn str_slice(
|
||||||
|
mut args: CallArgs,
|
||||||
|
scope: &Scope,
|
||||||
|
super_selector: &Selector,
|
||||||
|
) -> SassResult<Value> {
|
||||||
|
args.max_args(3)?;
|
||||||
|
let (string, quotes) = match arg!(args, scope, super_selector, 0, "string") {
|
||||||
|
Value::Ident(s, q) => (s, q),
|
||||||
|
v => {
|
||||||
|
return Err((
|
||||||
format!(
|
format!(
|
||||||
"$string: {} is not a string.",
|
"$string: {} is not a string.",
|
||||||
v.to_css_string(args.span())?
|
v.to_css_string(args.span())?
|
||||||
),
|
),
|
||||||
args.span(),
|
args.span(),
|
||||||
)
|
)
|
||||||
.into()),
|
.into())
|
||||||
}
|
}
|
||||||
}),
|
};
|
||||||
);
|
let str_len = string.chars().count();
|
||||||
f.insert(
|
let start = match arg!(args, scope, super_selector, 1, "start-at") {
|
||||||
"to-lower-case",
|
Value::Dimension(n, Unit::None) if n.is_decimal() => {
|
||||||
Builtin::new(|mut args, scope, super_selector| {
|
return Err((format!("{} is not an int.", n), args.span()).into())
|
||||||
args.max_args(1)?;
|
}
|
||||||
match arg!(args, scope, super_selector, 0, "string") {
|
Value::Dimension(n, Unit::None) if n.is_positive() => {
|
||||||
Value::Ident(mut i, q) => {
|
n.to_integer().to_usize().unwrap_or(str_len + 1)
|
||||||
i.make_ascii_lowercase();
|
}
|
||||||
Ok(Value::Ident(i, q))
|
Value::Dimension(n, Unit::None) if n.is_zero() => 1_usize,
|
||||||
}
|
Value::Dimension(n, Unit::None) if n < -Number::from(str_len) => 1_usize,
|
||||||
v => Err((
|
Value::Dimension(n, Unit::None) => (BigInt::from(str_len + 1) + n.to_integer())
|
||||||
|
.to_usize()
|
||||||
|
.unwrap(),
|
||||||
|
v @ Value::Dimension(..) => {
|
||||||
|
return Err((
|
||||||
|
format!(
|
||||||
|
"$start: Expected {} to have no units.",
|
||||||
|
v.to_css_string(args.span())?
|
||||||
|
),
|
||||||
|
args.span(),
|
||||||
|
)
|
||||||
|
.into())
|
||||||
|
}
|
||||||
|
v => {
|
||||||
|
return Err((
|
||||||
|
format!(
|
||||||
|
"$start-at: {} is not a number.",
|
||||||
|
v.to_css_string(args.span())?
|
||||||
|
),
|
||||||
|
args.span(),
|
||||||
|
)
|
||||||
|
.into())
|
||||||
|
}
|
||||||
|
};
|
||||||
|
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), args.span()).into())
|
||||||
|
}
|
||||||
|
Value::Dimension(n, Unit::None) if n.is_positive() => {
|
||||||
|
n.to_integer().to_usize().unwrap_or(str_len + 1)
|
||||||
|
}
|
||||||
|
Value::Dimension(n, Unit::None) if n.is_zero() => 0_usize,
|
||||||
|
Value::Dimension(n, Unit::None) if n < -Number::from(str_len) => 0_usize,
|
||||||
|
Value::Dimension(n, Unit::None) => (BigInt::from(str_len + 1) + n.to_integer())
|
||||||
|
.to_usize()
|
||||||
|
.unwrap_or(str_len + 1),
|
||||||
|
v @ Value::Dimension(..) => {
|
||||||
|
return Err((
|
||||||
|
format!(
|
||||||
|
"$end: Expected {} to have no units.",
|
||||||
|
v.to_css_string(args.span())?
|
||||||
|
),
|
||||||
|
args.span(),
|
||||||
|
)
|
||||||
|
.into())
|
||||||
|
}
|
||||||
|
Value::Null => str_len,
|
||||||
|
v => {
|
||||||
|
return Err((
|
||||||
|
format!(
|
||||||
|
"$end-at: {} is not a number.",
|
||||||
|
v.to_css_string(args.span())?
|
||||||
|
),
|
||||||
|
args.span(),
|
||||||
|
)
|
||||||
|
.into())
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
if end > str_len {
|
||||||
|
end = str_len;
|
||||||
|
}
|
||||||
|
|
||||||
|
if start > end || start > str_len {
|
||||||
|
Ok(Value::Ident(String::new(), quotes))
|
||||||
|
} else {
|
||||||
|
Ok(Value::Ident(
|
||||||
|
string
|
||||||
|
.chars()
|
||||||
|
.skip(start - 1)
|
||||||
|
.take(end - start + 1)
|
||||||
|
.collect(),
|
||||||
|
quotes,
|
||||||
|
))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn str_index(
|
||||||
|
mut args: CallArgs,
|
||||||
|
scope: &Scope,
|
||||||
|
super_selector: &Selector,
|
||||||
|
) -> SassResult<Value> {
|
||||||
|
args.max_args(2)?;
|
||||||
|
let s1 = match arg!(args, scope, super_selector, 0, "string") {
|
||||||
|
Value::Ident(i, _) => i,
|
||||||
|
v => {
|
||||||
|
return Err((
|
||||||
format!(
|
format!(
|
||||||
"$string: {} is not a string.",
|
"$string: {} is not a string.",
|
||||||
v.to_css_string(args.span())?
|
v.to_css_string(args.span())?
|
||||||
),
|
),
|
||||||
args.span(),
|
args.span(),
|
||||||
)
|
)
|
||||||
.into()),
|
.into())
|
||||||
}
|
}
|
||||||
}),
|
};
|
||||||
);
|
|
||||||
f.insert(
|
let substr = match arg!(args, scope, super_selector, 1, "substring") {
|
||||||
"str-length",
|
Value::Ident(i, _) => i,
|
||||||
Builtin::new(|mut args, scope, super_selector| {
|
v => {
|
||||||
args.max_args(1)?;
|
return Err((
|
||||||
match arg!(args, scope, super_selector, 0, "string") {
|
format!(
|
||||||
Value::Ident(i, _) => Ok(Value::Dimension(
|
"$substring: {} is not a string.",
|
||||||
Number::from(i.chars().count()),
|
v.to_css_string(args.span())?
|
||||||
Unit::None,
|
),
|
||||||
)),
|
args.span(),
|
||||||
v => Err((
|
)
|
||||||
|
.into())
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
Ok(match s1.find(&substr) {
|
||||||
|
Some(v) => Value::Dimension(Number::from(v + 1), Unit::None),
|
||||||
|
None => Value::Null,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
fn str_insert(
|
||||||
|
mut args: CallArgs,
|
||||||
|
scope: &Scope,
|
||||||
|
super_selector: &Selector,
|
||||||
|
) -> SassResult<Value> {
|
||||||
|
args.max_args(3)?;
|
||||||
|
let (s1, quotes) = match arg!(args, scope, super_selector, 0, "string") {
|
||||||
|
Value::Ident(i, q) => (i, q),
|
||||||
|
v => {
|
||||||
|
return Err((
|
||||||
format!(
|
format!(
|
||||||
"$string: {} is not a string.",
|
"$string: {} is not a string.",
|
||||||
v.to_css_string(args.span())?
|
v.to_css_string(args.span())?
|
||||||
),
|
),
|
||||||
args.span(),
|
args.span(),
|
||||||
)
|
)
|
||||||
.into()),
|
.into())
|
||||||
}
|
}
|
||||||
}),
|
};
|
||||||
);
|
|
||||||
f.insert(
|
let substr = match arg!(args, scope, super_selector, 1, "insert") {
|
||||||
"quote",
|
Value::Ident(i, _) => i,
|
||||||
Builtin::new(|mut args, scope, super_selector| {
|
v => {
|
||||||
args.max_args(1)?;
|
return Err((
|
||||||
match arg!(args, scope, super_selector, 0, "string") {
|
|
||||||
Value::Ident(i, _) => Ok(Value::Ident(i, QuoteKind::Quoted)),
|
|
||||||
v => Err((
|
|
||||||
format!(
|
format!(
|
||||||
"$string: {} is not a string.",
|
"$insert: {} is not a string.",
|
||||||
v.to_css_string(args.span())?
|
v.to_css_string(args.span())?
|
||||||
),
|
),
|
||||||
args.span(),
|
args.span(),
|
||||||
)
|
)
|
||||||
.into()),
|
.into())
|
||||||
}
|
}
|
||||||
}),
|
};
|
||||||
);
|
|
||||||
f.insert(
|
let index = match arg!(args, scope, super_selector, 2, "index") {
|
||||||
"unquote",
|
Value::Dimension(n, Unit::None) if n.is_decimal() => {
|
||||||
Builtin::new(|mut args, scope, super_selector| {
|
return Err((format!("$index: {} is not an int.", n), args.span()).into())
|
||||||
args.max_args(1)?;
|
}
|
||||||
match arg!(args, scope, super_selector, 0, "string") {
|
Value::Dimension(n, Unit::None) => n,
|
||||||
i @ Value::Ident(..) => Ok(i.unquote()),
|
v @ Value::Dimension(..) => {
|
||||||
v => Err((
|
return Err((
|
||||||
format!(
|
format!(
|
||||||
"$string: {} is not a string.",
|
"$index: Expected {} to have no units.",
|
||||||
v.to_css_string(args.span())?
|
v.to_css_string(args.span())?
|
||||||
),
|
),
|
||||||
args.span(),
|
args.span(),
|
||||||
)
|
)
|
||||||
.into()),
|
.into())
|
||||||
}
|
}
|
||||||
}),
|
v => {
|
||||||
);
|
return Err((
|
||||||
f.insert(
|
format!("$index: {} is not a number.", v.to_css_string(args.span())?),
|
||||||
"str-slice",
|
args.span(),
|
||||||
Builtin::new(|mut args, scope, super_selector| {
|
)
|
||||||
args.max_args(3)?;
|
.into())
|
||||||
let (string, quotes) = match arg!(args, scope, super_selector, 0, "string") {
|
}
|
||||||
Value::Ident(s, q) => (s, q),
|
};
|
||||||
v => {
|
|
||||||
return Err((
|
if s1.is_empty() {
|
||||||
format!(
|
return Ok(Value::Ident(substr, quotes));
|
||||||
"$string: {} is not a string.",
|
}
|
||||||
v.to_css_string(args.span())?
|
|
||||||
),
|
let len = s1.chars().count();
|
||||||
args.span(),
|
|
||||||
)
|
// Insert substring at char position, rather than byte position
|
||||||
.into())
|
let insert = |idx, s1: String, s2| {
|
||||||
}
|
s1.chars()
|
||||||
};
|
.enumerate()
|
||||||
let str_len = string.chars().count();
|
.map(|(i, c)| {
|
||||||
let start = match arg!(args, scope, super_selector, 1, "start-at") {
|
if i + 1 == idx {
|
||||||
Value::Dimension(n, Unit::None) if n.is_decimal() => {
|
c.to_string() + s2
|
||||||
return Err((format!("{} is not an int.", n), args.span()).into())
|
} else if idx == 0 && i == 0 {
|
||||||
}
|
s2.to_string() + &c.to_string()
|
||||||
Value::Dimension(n, Unit::None) if n.is_positive() => {
|
} else {
|
||||||
n.to_integer().to_usize().unwrap_or(str_len + 1)
|
c.to_string()
|
||||||
}
|
}
|
||||||
Value::Dimension(n, Unit::None) if n.is_zero() => 1_usize,
|
})
|
||||||
Value::Dimension(n, Unit::None) if n < -Number::from(str_len) => 1_usize,
|
.collect::<String>()
|
||||||
Value::Dimension(n, Unit::None) => (BigInt::from(str_len + 1) + n.to_integer())
|
};
|
||||||
|
|
||||||
|
let string = if index.is_positive() {
|
||||||
|
insert(
|
||||||
|
index
|
||||||
|
.to_integer()
|
||||||
.to_usize()
|
.to_usize()
|
||||||
.unwrap(),
|
.unwrap_or(len + 1)
|
||||||
v @ Value::Dimension(..) => {
|
.min(len + 1)
|
||||||
return Err((
|
- 1,
|
||||||
format!(
|
s1,
|
||||||
"$start: Expected {} to have no units.",
|
&substr,
|
||||||
v.to_css_string(args.span())?
|
)
|
||||||
),
|
} else if index.is_zero() {
|
||||||
args.span(),
|
insert(0, s1, &substr)
|
||||||
)
|
} else {
|
||||||
.into())
|
let idx = index.abs().to_integer().to_usize().unwrap_or(len + 1);
|
||||||
}
|
if idx > len {
|
||||||
v => {
|
|
||||||
return Err((
|
|
||||||
format!(
|
|
||||||
"$start-at: {} is not a number.",
|
|
||||||
v.to_css_string(args.span())?
|
|
||||||
),
|
|
||||||
args.span(),
|
|
||||||
)
|
|
||||||
.into())
|
|
||||||
}
|
|
||||||
};
|
|
||||||
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), args.span()).into())
|
|
||||||
}
|
|
||||||
Value::Dimension(n, Unit::None) if n.is_positive() => {
|
|
||||||
n.to_integer().to_usize().unwrap_or(str_len + 1)
|
|
||||||
}
|
|
||||||
Value::Dimension(n, Unit::None) if n.is_zero() => 0_usize,
|
|
||||||
Value::Dimension(n, Unit::None) if n < -Number::from(str_len) => 0_usize,
|
|
||||||
Value::Dimension(n, Unit::None) => (BigInt::from(str_len + 1) + n.to_integer())
|
|
||||||
.to_usize()
|
|
||||||
.unwrap_or(str_len + 1),
|
|
||||||
v @ Value::Dimension(..) => {
|
|
||||||
return Err((
|
|
||||||
format!(
|
|
||||||
"$end: Expected {} to have no units.",
|
|
||||||
v.to_css_string(args.span())?
|
|
||||||
),
|
|
||||||
args.span(),
|
|
||||||
)
|
|
||||||
.into())
|
|
||||||
}
|
|
||||||
Value::Null => str_len,
|
|
||||||
v => {
|
|
||||||
return Err((
|
|
||||||
format!(
|
|
||||||
"$end-at: {} is not a number.",
|
|
||||||
v.to_css_string(args.span())?
|
|
||||||
),
|
|
||||||
args.span(),
|
|
||||||
)
|
|
||||||
.into())
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
if end > str_len {
|
|
||||||
end = str_len;
|
|
||||||
}
|
|
||||||
|
|
||||||
if start > end || start > str_len {
|
|
||||||
Ok(Value::Ident(String::new(), quotes))
|
|
||||||
} else {
|
|
||||||
Ok(Value::Ident(
|
|
||||||
string
|
|
||||||
.chars()
|
|
||||||
.skip(start - 1)
|
|
||||||
.take(end - start + 1)
|
|
||||||
.collect(),
|
|
||||||
quotes,
|
|
||||||
))
|
|
||||||
}
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
f.insert(
|
|
||||||
"str-index",
|
|
||||||
Builtin::new(|mut args, scope, super_selector| {
|
|
||||||
args.max_args(2)?;
|
|
||||||
let s1 = match arg!(args, scope, super_selector, 0, "string") {
|
|
||||||
Value::Ident(i, _) => i,
|
|
||||||
v => {
|
|
||||||
return Err((
|
|
||||||
format!(
|
|
||||||
"$string: {} is not a string.",
|
|
||||||
v.to_css_string(args.span())?
|
|
||||||
),
|
|
||||||
args.span(),
|
|
||||||
)
|
|
||||||
.into())
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
let substr = match arg!(args, scope, super_selector, 1, "substring") {
|
|
||||||
Value::Ident(i, _) => i,
|
|
||||||
v => {
|
|
||||||
return Err((
|
|
||||||
format!(
|
|
||||||
"$substring: {} is not a string.",
|
|
||||||
v.to_css_string(args.span())?
|
|
||||||
),
|
|
||||||
args.span(),
|
|
||||||
)
|
|
||||||
.into())
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
Ok(match s1.find(&substr) {
|
|
||||||
Some(v) => Value::Dimension(Number::from(v + 1), Unit::None),
|
|
||||||
None => Value::Null,
|
|
||||||
})
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
f.insert(
|
|
||||||
"str-insert",
|
|
||||||
Builtin::new(|mut args, scope, super_selector| {
|
|
||||||
args.max_args(3)?;
|
|
||||||
let (s1, quotes) = match arg!(args, scope, super_selector, 0, "string") {
|
|
||||||
Value::Ident(i, q) => (i, q),
|
|
||||||
v => {
|
|
||||||
return Err((
|
|
||||||
format!(
|
|
||||||
"$string: {} is not a string.",
|
|
||||||
v.to_css_string(args.span())?
|
|
||||||
),
|
|
||||||
args.span(),
|
|
||||||
)
|
|
||||||
.into())
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
let substr = match arg!(args, scope, super_selector, 1, "insert") {
|
|
||||||
Value::Ident(i, _) => i,
|
|
||||||
v => {
|
|
||||||
return Err((
|
|
||||||
format!(
|
|
||||||
"$insert: {} is not a string.",
|
|
||||||
v.to_css_string(args.span())?
|
|
||||||
),
|
|
||||||
args.span(),
|
|
||||||
)
|
|
||||||
.into())
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
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), args.span()).into())
|
|
||||||
}
|
|
||||||
Value::Dimension(n, Unit::None) => n,
|
|
||||||
v @ Value::Dimension(..) => {
|
|
||||||
return Err((
|
|
||||||
format!(
|
|
||||||
"$index: Expected {} to have no units.",
|
|
||||||
v.to_css_string(args.span())?
|
|
||||||
),
|
|
||||||
args.span(),
|
|
||||||
)
|
|
||||||
.into())
|
|
||||||
}
|
|
||||||
v => {
|
|
||||||
return Err((
|
|
||||||
format!("$index: {} is not a number.", v.to_css_string(args.span())?),
|
|
||||||
args.span(),
|
|
||||||
)
|
|
||||||
.into())
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
if s1.is_empty() {
|
|
||||||
return Ok(Value::Ident(substr, quotes));
|
|
||||||
}
|
|
||||||
|
|
||||||
let len = s1.chars().count();
|
|
||||||
|
|
||||||
// Insert substring at char position, rather than byte position
|
|
||||||
let insert = |idx, s1: String, s2| {
|
|
||||||
s1.chars()
|
|
||||||
.enumerate()
|
|
||||||
.map(|(i, c)| {
|
|
||||||
if i + 1 == idx {
|
|
||||||
c.to_string() + s2
|
|
||||||
} else if idx == 0 && i == 0 {
|
|
||||||
s2.to_string() + &c.to_string()
|
|
||||||
} else {
|
|
||||||
c.to_string()
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.collect::<String>()
|
|
||||||
};
|
|
||||||
|
|
||||||
let string = if index.is_positive() {
|
|
||||||
insert(
|
|
||||||
index
|
|
||||||
.to_integer()
|
|
||||||
.to_usize()
|
|
||||||
.unwrap_or(len + 1)
|
|
||||||
.min(len + 1)
|
|
||||||
- 1,
|
|
||||||
s1,
|
|
||||||
&substr,
|
|
||||||
)
|
|
||||||
} else if index.is_zero() {
|
|
||||||
insert(0, s1, &substr)
|
insert(0, s1, &substr)
|
||||||
} else {
|
} else {
|
||||||
let idx = index.abs().to_integer().to_usize().unwrap_or(len + 1);
|
insert(len - idx + 1, s1, &substr)
|
||||||
if idx > len {
|
}
|
||||||
insert(0, s1, &substr)
|
};
|
||||||
} else {
|
|
||||||
insert(len - idx + 1, s1, &substr)
|
Ok(Value::Ident(string, quotes))
|
||||||
}
|
}
|
||||||
};
|
|
||||||
|
|
||||||
Ok(Value::Ident(string, quotes))
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
#[cfg(feature = "random")]
|
#[cfg(feature = "random")]
|
||||||
f.insert(
|
fn unique_id(args: CallArgs, _: &Scope, _: &Selector) -> SassResult<Value> {
|
||||||
"unique-id",
|
args.max_args(0)?;
|
||||||
Builtin::new(|args, _, _| {
|
let mut rng = thread_rng();
|
||||||
args.max_args(0)?;
|
let string = std::iter::repeat(())
|
||||||
let mut rng = thread_rng();
|
.map(|()| rng.sample(Alphanumeric))
|
||||||
let string = std::iter::repeat(())
|
.take(7)
|
||||||
.map(|()| rng.sample(Alphanumeric))
|
.collect();
|
||||||
.take(7)
|
Ok(Value::Ident(string, QuoteKind::None))
|
||||||
.collect();
|
}
|
||||||
Ok(Value::Ident(string, QuoteKind::None))
|
|
||||||
}),
|
f.insert("to-upper-case", Builtin::new(to_upper_case));
|
||||||
);
|
f.insert("to-lower-case", Builtin::new(to_lower_case));
|
||||||
|
f.insert("str-length", Builtin::new(str_length));
|
||||||
|
f.insert("quote", Builtin::new(quote));
|
||||||
|
f.insert("unquote", Builtin::new(unquote));
|
||||||
|
f.insert("str-slice", Builtin::new(str_slice));
|
||||||
|
f.insert("str-index", Builtin::new(str_index));
|
||||||
|
f.insert("str-insert", Builtin::new(str_insert));
|
||||||
|
#[cfg(feature = "random")]
|
||||||
|
f.insert("unique-id", Builtin::new(unique_id));
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user