handle special functions in 3args rgba
This commit is contained in:
parent
97425665f9
commit
b5a0957aa7
@ -9,147 +9,334 @@ use crate::unit::Unit;
|
|||||||
use crate::value::{Number, Value};
|
use crate::value::{Number, Value};
|
||||||
|
|
||||||
pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
pub(crate) fn register(f: &mut HashMap<String, Builtin>) {
|
||||||
let rgb = Builtin::new(|mut args, scope, super_selector| {
|
f.insert(
|
||||||
if args.is_empty() {
|
"rgb".to_owned(),
|
||||||
return Err("Missing argument $channels.".into());
|
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, scope, super_selector, 0, "channels") {
|
|
||||||
Value::List(v, ..) => v,
|
|
||||||
_ => return Err("Missing argument $channels.".into()),
|
|
||||||
};
|
|
||||||
|
|
||||||
if channels.len() > 3 {
|
|
||||||
return Err(format!(
|
|
||||||
"Only 3 elements allowed, but {} were passed.",
|
|
||||||
channels.len()
|
|
||||||
)
|
|
||||||
.into());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
let blue = match channels.pop() {
|
if args.len() == 1 {
|
||||||
Some(Value::Dimension(n, Unit::None)) => n,
|
let mut channels = match arg!(args, scope, super_selector, 0, "channels") {
|
||||||
Some(Value::Dimension(n, Unit::Percent)) => {
|
Value::List(v, ..) => v,
|
||||||
(n / Number::from(100)) * Number::from(255)
|
_ => return Err("Missing argument $channels.".into()),
|
||||||
}
|
};
|
||||||
Some(v) => return Err(format!("$blue: {} is not a number.", v).into()),
|
|
||||||
None => return Err("Missing element $blue.".into()),
|
|
||||||
};
|
|
||||||
|
|
||||||
let green = match channels.pop() {
|
if channels.len() > 3 {
|
||||||
Some(Value::Dimension(n, Unit::None)) => n,
|
return Err(format!(
|
||||||
Some(Value::Dimension(n, Unit::Percent)) => {
|
"Only 3 elements allowed, but {} were passed.",
|
||||||
(n / Number::from(100)) * Number::from(255)
|
channels.len()
|
||||||
|
)
|
||||||
|
.into());
|
||||||
}
|
}
|
||||||
Some(v) => return Err(format!("$green: {} is not a number.", v).into()),
|
|
||||||
None => return Err("Missing element $green.".into()),
|
|
||||||
};
|
|
||||||
|
|
||||||
let red = match channels.pop() {
|
let blue = match channels.pop() {
|
||||||
Some(Value::Dimension(n, Unit::None)) => n,
|
Some(Value::Dimension(n, Unit::None)) => n,
|
||||||
Some(Value::Dimension(n, Unit::Percent)) => {
|
Some(Value::Dimension(n, Unit::Percent)) => {
|
||||||
(n / Number::from(100)) * Number::from(255)
|
(n / Number::from(100)) * Number::from(255)
|
||||||
}
|
|
||||||
Some(v) => return Err(format!("$red: {} is not a number.", v).into()),
|
|
||||||
None => return Err("Missing element $red.".into()),
|
|
||||||
};
|
|
||||||
|
|
||||||
let color = Color::from_rgba(red, green, blue, Number::one());
|
|
||||||
|
|
||||||
Ok(Value::Color(color))
|
|
||||||
} else if args.len() == 2 {
|
|
||||||
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, scope, super_selector, 1, "alpha") {
|
|
||||||
Value::Dimension(n, Unit::None) => n,
|
|
||||||
Value::Dimension(n, Unit::Percent) => n / Number::from(100),
|
|
||||||
v @ Value::Dimension(..) => {
|
|
||||||
return Err(format!("$alpha: Expected {} to have no units or \"%\".", v).into())
|
|
||||||
}
|
|
||||||
v => return Err(format!("$alpha: {} is not a number.", v).into()),
|
|
||||||
};
|
|
||||||
Ok(Value::Color(color.with_alpha(alpha)))
|
|
||||||
} else {
|
|
||||||
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),
|
|
||||||
v @ Value::Dimension(..) => {
|
|
||||||
return Err(format!("$red: Expected {} to have no units or \"%\".", v).into())
|
|
||||||
}
|
|
||||||
v if v.is_special_function() => {
|
|
||||||
let green = arg!(args, scope, super_selector, 1, "green");
|
|
||||||
let blue = arg!(args, scope, super_selector, 2, "blue");
|
|
||||||
let mut string = format!("rgb({}, {}, {}", v, green, blue);
|
|
||||||
if !args.is_empty() {
|
|
||||||
string.push_str(", ");
|
|
||||||
string.push_str(&arg!(args, scope, super_selector, 3, "alpha").to_string());
|
|
||||||
}
|
}
|
||||||
string.push(')');
|
Some(v) => return Err(format!("$blue: {} is not a number.", v).into()),
|
||||||
return Ok(Value::Ident(string, QuoteKind::None));
|
None => return Err("Missing element $blue.".into()),
|
||||||
}
|
};
|
||||||
v => return Err(format!("$red: {} is not a number.", v).into()),
|
|
||||||
};
|
let green = match channels.pop() {
|
||||||
let green = match arg!(args, scope, super_selector, 1, "green") {
|
Some(Value::Dimension(n, Unit::None)) => n,
|
||||||
Value::Dimension(n, Unit::None) => n,
|
Some(Value::Dimension(n, Unit::Percent)) => {
|
||||||
Value::Dimension(n, Unit::Percent) => (n / Number::from(100)) * Number::from(255),
|
(n / Number::from(100)) * Number::from(255)
|
||||||
v @ Value::Dimension(..) => {
|
|
||||||
return Err(format!("$green: Expected {} to have no units or \"%\".", v).into())
|
|
||||||
}
|
|
||||||
v if v.is_special_function() => {
|
|
||||||
let blue = arg!(args, scope, super_selector, 2, "blue");
|
|
||||||
let mut string = format!("rgb({}, {}, {}", red, v, blue);
|
|
||||||
if !args.is_empty() {
|
|
||||||
string.push_str(", ");
|
|
||||||
string.push_str(&arg!(args, scope, super_selector, 3, "alpha").to_string());
|
|
||||||
}
|
}
|
||||||
string.push(')');
|
Some(v) => return Err(format!("$green: {} is not a number.", v).into()),
|
||||||
return Ok(Value::Ident(string, QuoteKind::None));
|
None => return Err("Missing element $green.".into()),
|
||||||
}
|
};
|
||||||
v => return Err(format!("$green: {} is not a number.", v).into()),
|
|
||||||
};
|
let red = match channels.pop() {
|
||||||
let blue = match arg!(args, scope, super_selector, 2, "blue") {
|
Some(Value::Dimension(n, Unit::None)) => n,
|
||||||
Value::Dimension(n, Unit::None) => n,
|
Some(Value::Dimension(n, Unit::Percent)) => {
|
||||||
Value::Dimension(n, Unit::Percent) => (n / Number::from(100)) * Number::from(255),
|
(n / Number::from(100)) * Number::from(255)
|
||||||
v @ Value::Dimension(..) => {
|
|
||||||
return Err(format!("$blue: Expected {} to have no units or \"%\".", v).into())
|
|
||||||
}
|
|
||||||
v if v.is_special_function() => {
|
|
||||||
let mut string = format!("rgb({}, {}, {}", red, green, v);
|
|
||||||
if !args.is_empty() {
|
|
||||||
string.push_str(", ");
|
|
||||||
string.push_str(&arg!(args, scope, super_selector, 3, "alpha").to_string());
|
|
||||||
}
|
}
|
||||||
string.push(')');
|
Some(v) => return Err(format!("$red: {} is not a number.", v).into()),
|
||||||
return Ok(Value::Ident(string, QuoteKind::None));
|
None => return Err("Missing element $red.".into()),
|
||||||
|
};
|
||||||
|
|
||||||
|
let color = Color::from_rgba(red, green, blue, Number::one());
|
||||||
|
|
||||||
|
Ok(Value::Color(color))
|
||||||
|
} else if args.len() == 2 {
|
||||||
|
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, scope, super_selector, 1, "alpha") {
|
||||||
|
Value::Dimension(n, Unit::None) => n,
|
||||||
|
Value::Dimension(n, Unit::Percent) => n / Number::from(100),
|
||||||
|
v @ Value::Dimension(..) => {
|
||||||
|
return Err(
|
||||||
|
format!("$alpha: Expected {} to have no units or \"%\".", v).into()
|
||||||
|
)
|
||||||
|
}
|
||||||
|
v => return Err(format!("$alpha: {} is not a number.", v).into()),
|
||||||
|
};
|
||||||
|
Ok(Value::Color(color.with_alpha(alpha)))
|
||||||
|
} else {
|
||||||
|
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)
|
||||||
|
}
|
||||||
|
v @ Value::Dimension(..) => {
|
||||||
|
return Err(
|
||||||
|
format!("$red: Expected {} to have no units or \"%\".", v).into()
|
||||||
|
)
|
||||||
|
}
|
||||||
|
v if v.is_special_function() => {
|
||||||
|
let green = arg!(args, scope, super_selector, 1, "green");
|
||||||
|
let blue = arg!(args, scope, super_selector, 2, "blue");
|
||||||
|
let mut string = format!("rgb({}, {}, {}", v, green, blue);
|
||||||
|
if !args.is_empty() {
|
||||||
|
string.push_str(", ");
|
||||||
|
string.push_str(
|
||||||
|
&arg!(args, scope, super_selector, 3, "alpha").to_string(),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
string.push(')');
|
||||||
|
return Ok(Value::Ident(string, QuoteKind::None));
|
||||||
|
}
|
||||||
|
v => return Err(format!("$red: {} is not a number.", v).into()),
|
||||||
|
};
|
||||||
|
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)
|
||||||
|
}
|
||||||
|
v @ Value::Dimension(..) => {
|
||||||
|
return Err(
|
||||||
|
format!("$green: Expected {} to have no units or \"%\".", v).into()
|
||||||
|
)
|
||||||
|
}
|
||||||
|
v if v.is_special_function() => {
|
||||||
|
let blue = arg!(args, scope, super_selector, 2, "blue");
|
||||||
|
let mut string = format!("rgb({}, {}, {}", red, v, blue);
|
||||||
|
if !args.is_empty() {
|
||||||
|
string.push_str(", ");
|
||||||
|
string.push_str(
|
||||||
|
&arg!(args, scope, super_selector, 3, "alpha").to_string(),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
string.push(')');
|
||||||
|
return Ok(Value::Ident(string, QuoteKind::None));
|
||||||
|
}
|
||||||
|
v => return Err(format!("$green: {} is not a number.", v).into()),
|
||||||
|
};
|
||||||
|
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)
|
||||||
|
}
|
||||||
|
v @ Value::Dimension(..) => {
|
||||||
|
return Err(
|
||||||
|
format!("$blue: Expected {} to have no units or \"%\".", v).into()
|
||||||
|
)
|
||||||
|
}
|
||||||
|
v if v.is_special_function() => {
|
||||||
|
let mut string = format!("rgb({}, {}, {}", red, green, v);
|
||||||
|
if !args.is_empty() {
|
||||||
|
string.push_str(", ");
|
||||||
|
string.push_str(
|
||||||
|
&arg!(args, scope, super_selector, 3, "alpha").to_string(),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
string.push(')');
|
||||||
|
return Ok(Value::Ident(string, QuoteKind::None));
|
||||||
|
}
|
||||||
|
v => return Err(format!("$blue: {} is not a number.", v).into()),
|
||||||
|
};
|
||||||
|
let alpha = match arg!(
|
||||||
|
args,
|
||||||
|
scope,
|
||||||
|
super_selector,
|
||||||
|
3,
|
||||||
|
"alpha" = Value::Dimension(Number::one(), Unit::None)
|
||||||
|
) {
|
||||||
|
Value::Dimension(n, Unit::None) => n,
|
||||||
|
Value::Dimension(n, Unit::Percent) => n / Number::from(100),
|
||||||
|
v @ Value::Dimension(..) => {
|
||||||
|
return Err(
|
||||||
|
format!("$alpha: Expected {} to have no units or \"%\".", v).into()
|
||||||
|
)
|
||||||
|
}
|
||||||
|
v if v.is_special_function() => {
|
||||||
|
let string = format!("rgb({}, {}, {}, {})", red, green, blue, v);
|
||||||
|
return Ok(Value::Ident(string, QuoteKind::None));
|
||||||
|
}
|
||||||
|
v => return Err(format!("$alpha: {} is not a number.", v).into()),
|
||||||
|
};
|
||||||
|
Ok(Value::Color(Color::from_rgba(red, green, blue, alpha)))
|
||||||
|
}
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
f.insert(
|
||||||
|
"rgba".to_owned(),
|
||||||
|
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, scope, super_selector, 0, "channels") {
|
||||||
|
Value::List(v, ..) => v,
|
||||||
|
_ => return Err("Missing argument $channels.".into()),
|
||||||
|
};
|
||||||
|
|
||||||
|
if channels.len() > 3 {
|
||||||
|
return Err(format!(
|
||||||
|
"Only 3 elements allowed, but {} were passed.",
|
||||||
|
channels.len()
|
||||||
|
)
|
||||||
|
.into());
|
||||||
}
|
}
|
||||||
v => return Err(format!("$blue: {} is not a number.", v).into()),
|
|
||||||
};
|
let blue = match channels.pop() {
|
||||||
let alpha = match arg!(
|
Some(Value::Dimension(n, Unit::None)) => n,
|
||||||
args,
|
Some(Value::Dimension(n, Unit::Percent)) => {
|
||||||
scope,
|
(n / Number::from(100)) * Number::from(255)
|
||||||
super_selector,
|
}
|
||||||
3,
|
Some(v) => return Err(format!("$blue: {} is not a number.", v).into()),
|
||||||
"alpha" = Value::Dimension(Number::one(), Unit::None)
|
None => return Err("Missing element $blue.".into()),
|
||||||
) {
|
};
|
||||||
Value::Dimension(n, Unit::None) => n,
|
|
||||||
Value::Dimension(n, Unit::Percent) => n / Number::from(100),
|
let green = match channels.pop() {
|
||||||
v @ Value::Dimension(..) => {
|
Some(Value::Dimension(n, Unit::None)) => n,
|
||||||
return Err(format!("$alpha: Expected {} to have no units or \"%\".", v).into())
|
Some(Value::Dimension(n, Unit::Percent)) => {
|
||||||
}
|
(n / Number::from(100)) * Number::from(255)
|
||||||
v if v.is_special_function() => {
|
}
|
||||||
let string = format!("rgb({}, {}, {}, {})", red, green, blue, v);
|
Some(v) => return Err(format!("$green: {} is not a number.", v).into()),
|
||||||
return Ok(Value::Ident(string, QuoteKind::None));
|
None => return Err("Missing element $green.".into()),
|
||||||
}
|
};
|
||||||
v => return Err(format!("$alpha: {} is not a number.", v).into()),
|
|
||||||
};
|
let red = match channels.pop() {
|
||||||
Ok(Value::Color(Color::from_rgba(red, green, blue, alpha)))
|
Some(Value::Dimension(n, Unit::None)) => n,
|
||||||
}
|
Some(Value::Dimension(n, Unit::Percent)) => {
|
||||||
});
|
(n / Number::from(100)) * Number::from(255)
|
||||||
f.insert("rgb".to_owned(), rgb.clone());
|
}
|
||||||
f.insert("rgba".to_owned(), rgb);
|
Some(v) => return Err(format!("$red: {} is not a number.", v).into()),
|
||||||
|
None => return Err("Missing element $red.".into()),
|
||||||
|
};
|
||||||
|
|
||||||
|
let color = Color::from_rgba(red, green, blue, Number::one());
|
||||||
|
|
||||||
|
Ok(Value::Color(color))
|
||||||
|
} else if args.len() == 2 {
|
||||||
|
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, scope, super_selector, 1, "alpha") {
|
||||||
|
Value::Dimension(n, Unit::None) => n,
|
||||||
|
Value::Dimension(n, Unit::Percent) => n / Number::from(100),
|
||||||
|
v @ Value::Dimension(..) => {
|
||||||
|
return Err(
|
||||||
|
format!("$alpha: Expected {} to have no units or \"%\".", v).into()
|
||||||
|
)
|
||||||
|
}
|
||||||
|
v => return Err(format!("$alpha: {} is not a number.", v).into()),
|
||||||
|
};
|
||||||
|
Ok(Value::Color(color.with_alpha(alpha)))
|
||||||
|
} else {
|
||||||
|
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)
|
||||||
|
}
|
||||||
|
v @ Value::Dimension(..) => {
|
||||||
|
return Err(
|
||||||
|
format!("$red: Expected {} to have no units or \"%\".", v).into()
|
||||||
|
)
|
||||||
|
}
|
||||||
|
v if v.is_special_function() => {
|
||||||
|
let green = arg!(args, scope, super_selector, 1, "green");
|
||||||
|
let blue = arg!(args, scope, super_selector, 2, "blue");
|
||||||
|
let mut string = format!("rgba({}, {}, {}", v, green, blue);
|
||||||
|
if !args.is_empty() {
|
||||||
|
string.push_str(", ");
|
||||||
|
string.push_str(
|
||||||
|
&arg!(args, scope, super_selector, 3, "alpha").to_string(),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
string.push(')');
|
||||||
|
return Ok(Value::Ident(string, QuoteKind::None));
|
||||||
|
}
|
||||||
|
v => return Err(format!("$red: {} is not a number.", v).into()),
|
||||||
|
};
|
||||||
|
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)
|
||||||
|
}
|
||||||
|
v @ Value::Dimension(..) => {
|
||||||
|
return Err(
|
||||||
|
format!("$green: Expected {} to have no units or \"%\".", v).into()
|
||||||
|
)
|
||||||
|
}
|
||||||
|
v if v.is_special_function() => {
|
||||||
|
let blue = arg!(args, scope, super_selector, 2, "blue");
|
||||||
|
let mut string = format!("rgba({}, {}, {}", red, v, blue);
|
||||||
|
if !args.is_empty() {
|
||||||
|
string.push_str(", ");
|
||||||
|
string.push_str(
|
||||||
|
&arg!(args, scope, super_selector, 3, "alpha").to_string(),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
string.push(')');
|
||||||
|
return Ok(Value::Ident(string, QuoteKind::None));
|
||||||
|
}
|
||||||
|
v => return Err(format!("$green: {} is not a number.", v).into()),
|
||||||
|
};
|
||||||
|
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)
|
||||||
|
}
|
||||||
|
v @ Value::Dimension(..) => {
|
||||||
|
return Err(
|
||||||
|
format!("$blue: Expected {} to have no units or \"%\".", v).into()
|
||||||
|
)
|
||||||
|
}
|
||||||
|
v if v.is_special_function() => {
|
||||||
|
let mut string = format!("rgba({}, {}, {}", red, green, v);
|
||||||
|
if !args.is_empty() {
|
||||||
|
string.push_str(", ");
|
||||||
|
string.push_str(
|
||||||
|
&arg!(args, scope, super_selector, 3, "alpha").to_string(),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
string.push(')');
|
||||||
|
return Ok(Value::Ident(string, QuoteKind::None));
|
||||||
|
}
|
||||||
|
v => return Err(format!("$blue: {} is not a number.", v).into()),
|
||||||
|
};
|
||||||
|
let alpha = match arg!(
|
||||||
|
args,
|
||||||
|
scope,
|
||||||
|
super_selector,
|
||||||
|
3,
|
||||||
|
"alpha" = Value::Dimension(Number::one(), Unit::None)
|
||||||
|
) {
|
||||||
|
Value::Dimension(n, Unit::None) => n,
|
||||||
|
Value::Dimension(n, Unit::Percent) => n / Number::from(100),
|
||||||
|
v @ Value::Dimension(..) => {
|
||||||
|
return Err(
|
||||||
|
format!("$alpha: Expected {} to have no units or \"%\".", v).into()
|
||||||
|
)
|
||||||
|
}
|
||||||
|
v if v.is_special_function() => {
|
||||||
|
let string = format!("rgba({}, {}, {}, {})", red, green, blue, v);
|
||||||
|
return Ok(Value::Ident(string, QuoteKind::None));
|
||||||
|
}
|
||||||
|
v => return Err(format!("$alpha: {} is not a number.", v).into()),
|
||||||
|
};
|
||||||
|
Ok(Value::Color(Color::from_rgba(red, green, blue, alpha)))
|
||||||
|
}
|
||||||
|
}),
|
||||||
|
);
|
||||||
f.insert(
|
f.insert(
|
||||||
"red".to_owned(),
|
"red".to_owned(),
|
||||||
Builtin::new(|mut args, scope, super_selector| {
|
Builtin::new(|mut args, scope, super_selector| {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user