improve code coverage
also identified dead code and a bug in saturation and hue fns
This commit is contained in:
parent
dceaea6d42
commit
be9eb8e0b0
@ -237,25 +237,23 @@ impl Color {
|
||||
let red = self.red() / Number::from(255);
|
||||
let green = self.green() / Number::from(255);
|
||||
let blue = self.blue() / Number::from(255);
|
||||
|
||||
let min = min(&red, min(&green, &blue)).clone();
|
||||
let max = max(&red, max(&green, &blue)).clone();
|
||||
if min == max {
|
||||
return Number::zero();
|
||||
}
|
||||
|
||||
let mut hue = if blue == max {
|
||||
Number::from(4) + (red - green) / (max - min)
|
||||
} else if green == max {
|
||||
Number::from(2) + (blue - red) / (max - min)
|
||||
let delta = max.clone() - min.clone();
|
||||
|
||||
let hue = if min == max {
|
||||
Number::zero()
|
||||
} else if max == red {
|
||||
Number::from(60_u8) * (green - blue) / delta
|
||||
} else if max == green {
|
||||
Number::from(120_u8) + Number::from(60_u8) * (blue - red) / delta
|
||||
} else {
|
||||
(green - blue) / (max - min)
|
||||
Number::from(240_u8) + Number::from(60_u8) * (red - green) / delta
|
||||
};
|
||||
|
||||
if hue.is_negative() {
|
||||
hue += Number::from(360);
|
||||
}
|
||||
|
||||
(hue * Number::from(60)).round()
|
||||
hue % Number::from(360)
|
||||
}
|
||||
|
||||
/// Calculate saturation from RGBA values
|
||||
@ -275,14 +273,18 @@ impl Color {
|
||||
return Number::zero();
|
||||
}
|
||||
|
||||
let d = max.clone() - min.clone();
|
||||
let mm = max + min;
|
||||
let s = d / if mm > Number::one() {
|
||||
Number::from(2) - mm
|
||||
let delta = max.clone() - min.clone();
|
||||
|
||||
let sum = max + min;
|
||||
|
||||
let s = delta
|
||||
/ if sum > Number::one() {
|
||||
Number::from(2) - sum
|
||||
} else {
|
||||
mm
|
||||
sum
|
||||
};
|
||||
(s * Number::from(100)).round()
|
||||
|
||||
s * Number::from(100)
|
||||
}
|
||||
|
||||
/// Calculate luminance from RGBA values
|
||||
@ -392,11 +394,6 @@ impl Color {
|
||||
);
|
||||
|
||||
if saturation.is_zero() {
|
||||
let luminance = if luminance > Number::from(100) {
|
||||
Number::from(100)
|
||||
} else {
|
||||
luminance
|
||||
};
|
||||
let val = luminance * Number::from(255);
|
||||
let repr = repr(&val, &val, &val, &alpha);
|
||||
return Color::new_hsla(val.clone(), val.clone(), val, alpha, hsla, repr);
|
||||
@ -455,11 +452,14 @@ impl Color {
|
||||
if weight.is_zero() {
|
||||
return self.clone();
|
||||
}
|
||||
|
||||
let red = Number::from(u8::max_value()) - self.red();
|
||||
let green = Number::from(u8::max_value()) - self.green();
|
||||
let blue = Number::from(u8::max_value()) - self.blue();
|
||||
let repr = repr(&red, &green, &blue, &self.alpha());
|
||||
|
||||
let inverse = Color::new_rgba(red, green, blue, self.alpha(), repr);
|
||||
|
||||
inverse.mix(self, weight)
|
||||
}
|
||||
|
||||
|
@ -293,18 +293,10 @@ impl Formatter for CompressedFormatter {
|
||||
|
||||
let mut selectors = selectors.iter();
|
||||
if let Some(selector) = selectors.next() {
|
||||
match selector {
|
||||
KeyframesSelector::To => write!(buf, "to")?,
|
||||
KeyframesSelector::From => write!(buf, "from")?,
|
||||
KeyframesSelector::Percent(p) => write!(buf, "{}%", p)?,
|
||||
}
|
||||
write!(buf, "{}", selector)?;
|
||||
}
|
||||
for selector in selectors {
|
||||
match selector {
|
||||
KeyframesSelector::To => write!(buf, ",to")?,
|
||||
KeyframesSelector::From => write!(buf, ",from")?,
|
||||
KeyframesSelector::Percent(p) => write!(buf, ",{}%", p)?,
|
||||
}
|
||||
write!(buf, ",{}", selector)?;
|
||||
}
|
||||
|
||||
write!(buf, "{{")?;
|
||||
|
@ -181,6 +181,7 @@ impl<'a> Parser<'a> {
|
||||
c => string.push(c),
|
||||
}
|
||||
}
|
||||
|
||||
Err(("expected \"{\".", span).into())
|
||||
}
|
||||
|
||||
|
@ -60,6 +60,7 @@ impl<'a> Parser<'a> {
|
||||
return Err(("expected \"(\".", self.span_before).into());
|
||||
}
|
||||
};
|
||||
|
||||
while let Some(tok) = self.toks.next() {
|
||||
span = span.merge(tok.pos());
|
||||
match tok.kind {
|
||||
@ -73,6 +74,7 @@ impl<'a> Parser<'a> {
|
||||
_ => return Err(("expected \"(\".", span).into()),
|
||||
}
|
||||
}
|
||||
|
||||
Ok(string)
|
||||
}
|
||||
|
||||
@ -84,20 +86,10 @@ impl<'a> Parser<'a> {
|
||||
self.whitespace();
|
||||
|
||||
while let Some(tok) = self.toks.next() {
|
||||
let kind = tok.kind;
|
||||
|
||||
if kind == '!'
|
||||
|| kind == '%'
|
||||
|| kind == '&'
|
||||
|| ('*'..='~').contains(&kind)
|
||||
|| kind as u32 >= 0x0080
|
||||
{
|
||||
buf.push(kind);
|
||||
} else if kind == '\\' {
|
||||
buf.push_str(&self.parse_escape(false)?);
|
||||
} else if kind == '#' {
|
||||
if let Some(Token { kind: '{', .. }) = self.toks.peek() {
|
||||
self.toks.next();
|
||||
match tok.kind {
|
||||
'!' | '%' | '&' | '*'..='~' | '\u{80}'..=char::MAX => buf.push(tok.kind),
|
||||
'#' => {
|
||||
if self.consume_char_if_exists('{') {
|
||||
let interpolation = self.parse_interpolation()?;
|
||||
match interpolation.node {
|
||||
Value::String(ref s, ..) => buf.push_str(s),
|
||||
@ -106,23 +98,24 @@ impl<'a> Parser<'a> {
|
||||
} else {
|
||||
buf.push('#');
|
||||
}
|
||||
} else if kind == ')' {
|
||||
}
|
||||
')' => {
|
||||
buf.push(')');
|
||||
|
||||
return Ok(Some(buf));
|
||||
} else if kind.is_whitespace() {
|
||||
}
|
||||
' ' | '\t' | '\n' | '\r' => {
|
||||
self.whitespace();
|
||||
|
||||
if let Some(Token { kind: ')', .. }) = self.toks.peek() {
|
||||
self.toks.next();
|
||||
if self.consume_char_if_exists(')') {
|
||||
buf.push(')');
|
||||
|
||||
return Ok(Some(buf));
|
||||
}
|
||||
|
||||
break;
|
||||
} else {
|
||||
break;
|
||||
}
|
||||
_ => break,
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -52,8 +52,6 @@ impl<'a> Parser<'a> {
|
||||
return Ok(());
|
||||
}
|
||||
|
||||
var_value?.node
|
||||
} else if self.at_root {
|
||||
var_value?.node
|
||||
} else {
|
||||
if self.scopes.default_var_exists(ident) {
|
||||
|
@ -743,3 +743,48 @@ test!(
|
||||
"a {\n color: type-of(r#{e}d);\n}\n",
|
||||
"a {\n color: string;\n}\n"
|
||||
);
|
||||
test!(
|
||||
color_equality_differ_in_green_channel,
|
||||
"a {\n color: rgb(1, 1, 1) == rgb(1, 2, 1);\n}\n",
|
||||
"a {\n color: false;\n}\n"
|
||||
);
|
||||
test!(
|
||||
color_equality_differ_in_blue_channel,
|
||||
"a {\n color: rgb(1, 1, 1) == rgb(1, 1, 2);\n}\n",
|
||||
"a {\n color: false;\n}\n"
|
||||
);
|
||||
test!(
|
||||
color_equality_differ_in_alpha_channel,
|
||||
"a {\n color: rgba(1, 1, 1, 1.0) == rgba(1, 1, 1, 0.5);\n}\n",
|
||||
"a {\n color: false;\n}\n"
|
||||
);
|
||||
test!(
|
||||
hue_of_rgb_is_negative,
|
||||
"a {\n color: hue(rgb(255, 0, 1));\n}\n",
|
||||
"a {\n color: 359.7647058824deg;\n}\n"
|
||||
);
|
||||
test!(
|
||||
saturation_of_rgb_all_channels_equal,
|
||||
"a {\n color: saturation(rgb(125, 125, 125));\n}\n",
|
||||
"a {\n color: 0%;\n}\n"
|
||||
);
|
||||
test!(
|
||||
saturation_of_rgb_min_and_max_above_1,
|
||||
"a {\n color: saturation(rgb(255, 248, 248));\n}\n",
|
||||
"a {\n color: 100%;\n}\n"
|
||||
);
|
||||
test!(
|
||||
saturation_of_rgb_min_and_max_below_1,
|
||||
"a {\n color: saturation(rgb(88, 88, 87));\n}\n",
|
||||
"a {\n color: 0.5714285714%;\n}\n"
|
||||
);
|
||||
test!(
|
||||
invert_weight_zero_is_nop,
|
||||
"a {\n color: invert(#0f0f0f, 0);\n}\n",
|
||||
"a {\n color: #0f0f0f;\n}\n"
|
||||
);
|
||||
test!(
|
||||
mix_combined_weight_is_normalized_weight,
|
||||
"a {\n color: mix(rgba(255, 20, 0, 0), rgba(0, 20, 255, 1), 100);\n}\n",
|
||||
"a {\n color: rgba(255, 20, 0, 0);\n}\n"
|
||||
);
|
||||
|
@ -1,7 +1,6 @@
|
||||
#[macro_use]
|
||||
mod macros;
|
||||
|
||||
// @content inside keyframes
|
||||
test!(
|
||||
content_inside_keyframes,
|
||||
"@mixin foo {
|
||||
@ -137,3 +136,42 @@ test!(
|
||||
}",
|
||||
"@keyframes foo {\n 12.5% {\n color: red;\n }\n}\n"
|
||||
);
|
||||
test!(
|
||||
keyframes_hash_in_name,
|
||||
"@keyframes #identifier {
|
||||
to {
|
||||
color: red;
|
||||
}
|
||||
}",
|
||||
"@keyframes #identifier {\n to {\n color: red;\n }\n}\n"
|
||||
);
|
||||
test!(
|
||||
keyframes_interpolated_selector,
|
||||
"@keyframes foo {
|
||||
#{t}o {
|
||||
color: red;
|
||||
}
|
||||
}",
|
||||
"@keyframes foo {\n to {\n color: red;\n }\n}\n"
|
||||
);
|
||||
error!(
|
||||
keyframes_denies_selector_with_hash,
|
||||
"@keyframes foo {
|
||||
#to {
|
||||
color: red;
|
||||
}
|
||||
}",
|
||||
"Error: Expected \"to\" or \"from\"."
|
||||
);
|
||||
error!(
|
||||
keyframes_nothing_after_forward_slash_in_selector,
|
||||
"@keyframes foo { a/", "Error: Expected selector."
|
||||
);
|
||||
error!(
|
||||
keyframes_no_ident_after_forward_slash_in_selector,
|
||||
"@keyframes foo { a/ {} }", "Error: expected selector."
|
||||
);
|
||||
error!(
|
||||
keyframes_nothing_after_selector,
|
||||
"@keyframes foo { a", "Error: expected \"{\"."
|
||||
);
|
||||
|
@ -286,6 +286,14 @@ test!(
|
||||
"@function a(){} a {\n color: function-exists($name: a)\n}\n",
|
||||
"a {\n color: true;\n}\n"
|
||||
);
|
||||
test!(
|
||||
function_exists_not_global,
|
||||
"a {
|
||||
@function foo () {}
|
||||
color: function-exists($name: 'foo');
|
||||
}",
|
||||
"a {\n color: true;\n}\n"
|
||||
);
|
||||
error!(
|
||||
function_exists_non_string,
|
||||
"a {color: function-exists(12px)}", "Error: $name: 12px is not a string."
|
||||
|
@ -186,3 +186,64 @@ test!(
|
||||
"a {\n color: min(calc(1 /* #{5} */ 2));\n}\n",
|
||||
"a {\n color: min(calc(1 /* #{5} */ 2));\n}\n"
|
||||
);
|
||||
test!(
|
||||
min_uppercase,
|
||||
"a {\n color: MIN(1);\n}\n",
|
||||
"a {\n color: min(1);\n}\n"
|
||||
);
|
||||
test!(
|
||||
max_uppercase,
|
||||
"a {\n color: MAX(1);\n}\n",
|
||||
"a {\n color: max(1);\n}\n"
|
||||
);
|
||||
|
||||
test!(
|
||||
min_parenthesis_around_arg,
|
||||
"a {\n color: min((1));\n}\n",
|
||||
"a {\n color: min((1));\n}\n"
|
||||
);
|
||||
error!(
|
||||
min_parenthesis_around_arg_with_comma,
|
||||
"a {\n color: min((1, 1));\n}\n", "Error: 1, 1 is not a number."
|
||||
);
|
||||
error!(
|
||||
min_hash_without_interpolation,
|
||||
"a {\n color: min(#a);\n}\n", "Error: #a is not a number."
|
||||
);
|
||||
error!(
|
||||
min_calc_no_parens,
|
||||
"a {\n color: min(calc);\n}\n", "Error: calc is not a number."
|
||||
);
|
||||
error!(
|
||||
min_env_no_parens,
|
||||
"a {\n color: min(env);\n}\n", "Error: env is not a number."
|
||||
);
|
||||
error!(
|
||||
min_var_no_parens,
|
||||
"a {\n color: min(var);\n}\n", "Error: var is not a number."
|
||||
);
|
||||
error!(
|
||||
min_min_unfinished,
|
||||
"a {\n color: min(mi);\n}\n", "Error: mi is not a number."
|
||||
);
|
||||
error!(
|
||||
min_max_unfinished,
|
||||
"a {\n color: min(ma);\n}\n", "Error: ma is not a number."
|
||||
);
|
||||
error!(
|
||||
min_min_no_parens,
|
||||
"a {\n color: min(min);\n}\n", "Error: min is not a number."
|
||||
);
|
||||
error!(
|
||||
min_max_no_parens,
|
||||
"a {\n color: min(max);\n}\n", "Error: max is not a number."
|
||||
);
|
||||
error!(
|
||||
min_min_invalid,
|
||||
"a {\n color: min(min(#));\n}\n", "Error: Expected identifier."
|
||||
);
|
||||
test!(
|
||||
min_calc_parens_no_args,
|
||||
"a {\n color: min(calc());\n}\n",
|
||||
"a {\n color: min(calc());\n}\n"
|
||||
);
|
||||
|
@ -185,3 +185,7 @@ test!(
|
||||
"a {\n color: \"foo\\\nbar\";\n}\n",
|
||||
"a {\n color: \"foobar\";\n}\n"
|
||||
);
|
||||
error!(
|
||||
newline_after_escape,
|
||||
"a {\n color: \\\n", "Error: Expected escape sequence."
|
||||
);
|
||||
|
@ -176,6 +176,16 @@ test!(
|
||||
"a {\n color: unit(1-\\65);\n}\n",
|
||||
"a {\n color: \"-e\";\n}\n"
|
||||
);
|
||||
test!(
|
||||
viewport_relative_comparable_same,
|
||||
"a {\n color: comparable(1vw, 2vw);\n}\n",
|
||||
"a {\n color: true;\n}\n"
|
||||
);
|
||||
test!(
|
||||
viewport_relative_comparable_different,
|
||||
"a {\n color: comparable(1vw, 1vh);\n}\n",
|
||||
"a {\n color: false;\n}\n"
|
||||
);
|
||||
error!(
|
||||
display_single_div_with_none_numerator,
|
||||
"a {\n color: (1 / 1em);\n}\n", "Error: 1em^-1 isn't a valid CSS value."
|
||||
|
@ -405,3 +405,22 @@ error!(
|
||||
only_semicolon_after_hash_in_variable_decl,
|
||||
"$color: #;", "Error: Expected identifier."
|
||||
);
|
||||
|
||||
test!(
|
||||
variable_name_begins_with_escape,
|
||||
"$\\69: red;
|
||||
|
||||
a {
|
||||
color: $\\69;
|
||||
}",
|
||||
"a {\n color: red;\n}\n"
|
||||
);
|
||||
test!(
|
||||
variable_name_contains_escape,
|
||||
"$a\\69: red;
|
||||
|
||||
a {
|
||||
color: $a\\69;
|
||||
}",
|
||||
"a {\n color: red;\n}\n"
|
||||
);
|
||||
|
Loading…
x
Reference in New Issue
Block a user