/* Any copyright is dedicated to the Public Domain.
http://creativecommons.org/publicdomain/zero/1.0/ */
"use strict" ;
add_task(async function () {
await pushPref("layout.css.backdrop-filter.enabled" , true );
await addTab("about:blank" );
await performTest();
gBrowser.removeCurrentTab();
});
async function performTest() {
await SpecialPowers.pushPrefEnv({
set: [["security.allow_unsafe_parent_loads" , true ]],
});
await pushPref("layout.css.relative-color-syntax.enabled" , true );
const OutputParser = require("resource://devtools/client/shared/output-parser.js");
const { host, doc } = await createHost(
"bottom" ,
"data:text/html," + "browser_outputParser.js
"
);
const cssProperties = getClientCssProperties();
const parser = new OutputParser(doc, cssProperties);
testParseCssProperty(doc, parser);
testParseCssVar(doc, parser);
testParseURL(doc, parser);
testParseFilter(doc, parser);
testParseBackdropFilter(doc, parser);
testParseAngle(doc, parser);
testParseShape(doc, parser);
testParseVariable(doc, parser);
testParseColorVariable(doc, parser);
testParseFontFamily(doc, parser);
testParseLightDark(doc, parser);
host.destroy();
}
// Class name used in color swatch.
var COLOR_TEST_CLASS = "test-class" ;
// Create a new CSS color-parsing test. |name| is the name of the CSS
// property. |value| is the CSS text to use. |segments| is an array
// describing the expected result. If an element of |segments| is a
// string, it is simply appended to the expected string. Otherwise,
// it must be an object with a |name| property, which is the color
// name as it appears in the input.
//
// This approach is taken to reduce boilerplate and to make it simpler
// to modify the test when the parseCssProperty output changes.
function makeColorTest(name, value, segments) {
const result = {
name,
value,
expected: "" ,
};
for (const segment of segments) {
if (typeof segment === "string" ) {
result.expected += segment;
} else {
const buttonAttributes = {
class : COLOR_TEST_CLASS,
style: `background-color:${segment.name}`,
tabindex: 0,
role: "button" ,
};
if (segment.colorFunction) {
buttonAttributes["data-color-function" ] = segment.colorFunction;
}
const buttonAttrString = Object.entries(buttonAttributes)
.map(([attr, v]) => `${attr}="${v}" `)
.join(" " );
// prettier-ignore
result.expected +=
`<span data-color="${segment.name}" class ="color-swatch-container" >` +
`<span ${buttonAttrString}></span>`+
`<span>${segment.name}</span>` +
`</span>`;
}
}
result.desc = "Testing " + name + ": " + value;
return result;
}
function testParseCssProperty(doc, parser) {
const tests = [
makeColorTest("border" , "1px solid red" , ["1px solid " , { name: "red" }]),
makeColorTest(
"background-image" ,
"linear-gradient(to right, #F60 10%, rgba(0,0,0,1))" ,
[
"linear-gradient(to right, " ,
{ name: "#F60" , colorFunction: "linear-gradient" },
" 10%, " ,
{ name: "rgba(0,0,0,1)" , colorFunction: "linear-gradient" },
")" ,
]
),
// In "arial black", "black" is a font, not a color.
// (The font-family parser creates a span)
makeColorTest("font-family" , "arial black" , ["arial black " ]),
makeColorTest("box-shadow" , "0 0 1em red" , ["0 0 1em " , { name: "red" }]),
makeColorTest("box-shadow" , "0 0 1em red, 2px 2px 0 0 rgba(0,0,0,.5)" , [
"0 0 1em " ,
{ name: "red" },
", 2px 2px 0 0 " ,
{ name: "rgba(0,0,0,.5)" },
]),
makeColorTest("content" , '"red"' , ['"red"' ]),
// Invalid property names should not cause exceptions.
makeColorTest("hellothere" , "'red'" , ["'red'" ]),
makeColorTest(
"filter" ,
"blur(1px) drop-shadow(0 0 0 blue) url(red.svg#blue)" ,
[
'' ,
"blur(1px) drop-shadow(0 0 0 " ,
{ name: "blue" , colorFunction: "drop-shadow" },
") url(red.svg#blue) " ,
]
),
makeColorTest("color" , "currentColor" , ["currentColor" ]),
// Test a very long property.
makeColorTest(
"background-image" ,
"linear-gradient(to left, transparent 0, transparent 5%,#F00 0, #F00 10%,#FF0 0, #FF0 15%,#0F0 0, #0F0 20%,#0FF 0, #0FF 25%,#00F 0, #00F 30%,#800 0, #800 35%,#880 0, #880 40%,#080 0, #080 45%,#088 0, #088 50%,#008 0, #008 55%,#FFF 0, #FFF 60%,#EEE 0, #EEE 65%,#CCC 0, #CCC 70%,#999 0, #999 75%,#666 0, #666 80%,#333 0, #333 85%,#111 0, #111 90%,#000 0, #000 95%,transparent 0, transparent 100%)" ,
[
"linear-gradient(to left, " ,
{ name: "transparent" , colorFunction: "linear-gradient" },
" 0, " ,
{ name: "transparent" , colorFunction: "linear-gradient" },
" 5%," ,
{ name: "#F00" , colorFunction: "linear-gradient" },
" 0, " ,
{ name: "#F00" , colorFunction: "linear-gradient" },
" 10%," ,
{ name: "#FF0" , colorFunction: "linear-gradient" },
" 0, " ,
{ name: "#FF0" , colorFunction: "linear-gradient" },
" 15%," ,
{ name: "#0F0" , colorFunction: "linear-gradient" },
" 0, " ,
{ name: "#0F0" , colorFunction: "linear-gradient" },
" 20%," ,
{ name: "#0FF" , colorFunction: "linear-gradient" },
" 0, " ,
{ name: "#0FF" , colorFunction: "linear-gradient" },
" 25%," ,
{ name: "#00F" , colorFunction: "linear-gradient" },
" 0, " ,
{ name: "#00F" , colorFunction: "linear-gradient" },
" 30%," ,
{ name: "#800" , colorFunction: "linear-gradient" },
" 0, " ,
{ name: "#800" , colorFunction: "linear-gradient" },
" 35%," ,
{ name: "#880" , colorFunction: "linear-gradient" },
" 0, " ,
{ name: "#880" , colorFunction: "linear-gradient" },
" 40%," ,
{ name: "#080" , colorFunction: "linear-gradient" },
" 0, " ,
{ name: "#080" , colorFunction: "linear-gradient" },
" 45%," ,
{ name: "#088" , colorFunction: "linear-gradient" },
" 0, " ,
{ name: "#088" , colorFunction: "linear-gradient" },
" 50%," ,
{ name: "#008" , colorFunction: "linear-gradient" },
" 0, " ,
{ name: "#008" , colorFunction: "linear-gradient" },
" 55%," ,
{ name: "#FFF" , colorFunction: "linear-gradient" },
" 0, " ,
{ name: "#FFF" , colorFunction: "linear-gradient" },
" 60%," ,
{ name: "#EEE" , colorFunction: "linear-gradient" },
" 0, " ,
{ name: "#EEE" , colorFunction: "linear-gradient" },
" 65%," ,
{ name: "#CCC" , colorFunction: "linear-gradient" },
" 0, " ,
{ name: "#CCC" , colorFunction: "linear-gradient" },
" 70%," ,
{ name: "#999" , colorFunction: "linear-gradient" },
" 0, " ,
{ name: "#999" , colorFunction: "linear-gradient" },
" 75%," ,
{ name: "#666" , colorFunction: "linear-gradient" },
" 0, " ,
{ name: "#666" , colorFunction: "linear-gradient" },
" 80%," ,
{ name: "#333" , colorFunction: "linear-gradient" },
" 0, " ,
{ name: "#333" , colorFunction: "linear-gradient" },
" 85%," ,
{ name: "#111" , colorFunction: "linear-gradient" },
" 0, " ,
{ name: "#111" , colorFunction: "linear-gradient" },
" 90%," ,
{ name: "#000" , colorFunction: "linear-gradient" },
" 0, " ,
{ name: "#000" , colorFunction: "linear-gradient" },
" 95%," ,
{ name: "transparent" , colorFunction: "linear-gradient" },
" 0, " ,
{ name: "transparent" , colorFunction: "linear-gradient" },
" 100%)" ,
]
),
// Note the lack of a space before the color here.
makeColorTest("border" , "1px dotted#f06" , [
"1px dotted " ,
{ name: "#f06" },
]),
makeColorTest("color" , "color-mix(in srgb, red, blue)" , [
"color-mix(in srgb, " ,
{ name: "red" , colorFunction: "color-mix" },
", " ,
{ name: "blue" , colorFunction: "color-mix" },
")" ,
]),
makeColorTest(
"background-image" ,
"linear-gradient(to top, color-mix(in srgb, #008000, rgba(255, 255, 0, 0.9)), blue)" ,
[
"linear-gradient(to top, " ,
"color-mix(in srgb, " ,
{ name: "#008000" , colorFunction: "color-mix" },
", " ,
{ name: "rgba(255, 255, 0, 0.9)" , colorFunction: "color-mix" },
"), " ,
{ name: "blue" , colorFunction: "linear-gradient" },
")" ,
]
),
makeColorTest("color" , "light-dark(red, blue)" , [
"light-dark(" ,
{ name: "red" , colorFunction: "light-dark" },
", " ,
{ name: "blue" , colorFunction: "light-dark" },
")" ,
]),
makeColorTest(
"background-image" ,
"linear-gradient(to top, light-dark(#008000, rgba(255, 255, 0, 0.9)), blue)" ,
[
"linear-gradient(to top, " ,
"light-dark(" ,
{ name: "#008000" , colorFunction: "light-dark" },
", " ,
{ name: "rgba(255, 255, 0, 0.9)" , colorFunction: "light-dark" },
"), " ,
{ name: "blue" , colorFunction: "linear-gradient" },
")" ,
]
),
makeColorTest("color" , "rgb(from gold r g b)" , [
{ name: "rgb(from gold r g b)" },
]),
makeColorTest("color" , "color(from hsl(0 100% 50%) xyz x y 0.5)" , [
{ name: "color(from hsl(0 100% 50%) xyz x y 0.5)" },
]),
makeColorTest(
"color" ,
"oklab(from red calc(l - 1) calc(a * 2) calc(b + 3) / alpha)" ,
[{ name: "oklab(from red calc(l - 1) calc(a * 2) calc(b + 3) / alpha)" }]
),
makeColorTest(
"color" ,
"rgb(from color-mix(in lch, plum 40%, pink) r g b)" ,
[{ name: "rgb(from color-mix(in lch, plum 40%, pink) r g b)" }]
),
makeColorTest("color" , "rgb(from rgb(from gold r g b) r g b)" , [
{ name: "rgb(from rgb(from gold r g b) r g b)" },
]),
makeColorTest(
"background-image" ,
"linear-gradient(to right, #F60 10%, rgb(from gold r g b))" ,
[
"linear-gradient(to right, " ,
{ name: "#F60" , colorFunction: "linear-gradient" },
" 10%, " ,
{ name: "rgb(from gold r g b)" , colorFunction: "linear-gradient" },
")" ,
]
),
{
desc: "--a: (min-width:680px)" ,
name: "--a" ,
value: "(min-width:680px)" ,
expected: "(min-width:680px)" ,
},
{
desc: "Interactive color swatch" ,
name: "color" ,
value: "gold" ,
expected:
// prettier-ignore
`<span data-color="gold" class ="color-swatch-container" >` +
`<span class ="test-class" style="background-color:gold" tabindex="0" role="button" ></span>` +
`<span>gold</span>` +
`</span>`,
parserExtraOptions: {
colorSwatchReadOnly: false ,
},
},
{
desc: "Read-only color swatch" ,
name: "color" ,
value: "gold" ,
expected:
// prettier-ignore
`<span data-color="gold" class ="color-swatch-container" >` +
`<span class ="test-class" style="background-color:gold" ></span>` +
`<span>gold</span>` +
`</span>`,
parserExtraOptions: {
colorSwatchReadOnly: true ,
},
},
];
const target = doc.querySelector("div" );
ok(target, "captain, we have the div" );
for (const test of tests) {
info(test.desc);
const frag = parser.parseCssProperty(test.name, test.value, {
colorSwatchClass: COLOR_TEST_CLASS,
...(test.parserExtraOptions || {}),
});
target.appendChild(frag);
is(
target.innerHTML,
test.expected,
"CSS property correctly parsed for " + test.name + ": " + test.value
);
target.innerHTML = "" ;
}
}
function testParseCssVar(doc, parser) {
const frag = parser.parseCssProperty("color" , "var(--some-kind-of-green)" , {
colorSwatchClass: "test-colorswatch" ,
});
const target = doc.querySelector("div" );
ok(target, "captain, we have the div" );
target.appendChild(frag);
is(
target.innerHTML,
"var(--some-kind-of-green)" ,
"CSS property correctly parsed"
);
target.innerHTML = "" ;
}
function testParseURL(doc, parser) {
info("Test that URL parsing preserves quoting style" );
const tests = [
{
desc: "simple test without quotes" ,
leader: "url(" ,
trailer: ")" ,
},
{
desc: "simple test with single quotes" ,
leader: "url('" ,
trailer: "')" ,
},
{
desc: "simple test with double quotes" ,
leader: 'url("' ,
trailer: '")' ,
},
{
desc: "test with single quotes and whitespace" ,
leader: "url( \t'" ,
trailer: "'\r\n\f)" ,
},
{
desc: "simple test with uppercase" ,
leader: "URL(" ,
trailer: ")" ,
},
{
desc: "bad url, missing paren" ,
leader: "url(" ,
trailer: "" ,
expectedTrailer: ")" ,
},
{
desc: "bad url, missing paren, with baseURI" ,
baseURI: "data:text/html," ,
leader: "url(" ,
trailer: "" ,
expectedTrailer: ")" ,
},
{
desc: "bad url, double quote, missing paren" ,
leader: 'url("' ,
trailer: '"' ,
expectedTrailer: '")' ,
},
{
desc: "bad url, single quote, missing paren and quote" ,
leader: "url('" ,
trailer: "" ,
expectedTrailer: "')" ,
},
];
for (const test of tests) {
const url = test.leader + "something.jpg" + test.trailer;
const frag = parser.parseCssProperty("background" , url, {
urlClass: "test-urlclass" ,
baseURI: test.baseURI,
});
const target = doc.querySelector("div" );
target.appendChild(frag);
const expectedTrailer = test.expectedTrailer || test.trailer;
const expected =
test.leader +
' +
'href="something.jpg">something.jpg ' +
expectedTrailer;
is(target.innerHTML, expected, test.desc);
target.innerHTML = "" ;
}
}
function testParseFilter(doc, parser) {
const frag = parser.parseCssProperty("filter" , "something invalid" , {
filterSwatchClass: "test-filterswatch" ,
});
const swatchCount = frag.querySelectorAll(".test-filterswatch" ).length;
is(swatchCount, 1, "filter swatch was created" );
}
function testParseBackdropFilter(doc, parser) {
const frag = parser.parseCssProperty("backdrop-filter" , "something invalid" , {
filterSwatchClass: "test-filterswatch" ,
});
const swatchCount = frag.querySelectorAll(".test-filterswatch" ).length;
is(swatchCount, 1, "filter swatch was created for backdrop-filter" );
}
function testParseAngle(doc, parser) {
let frag = parser.parseCssProperty("rotate" , "90deg" , {
angleSwatchClass: "test-angleswatch" ,
});
let swatchCount = frag.querySelectorAll(".test-angleswatch" ).length;
is(swatchCount, 1, "angle swatch was created" );
frag = parser.parseCssProperty(
"background-image" ,
"linear-gradient(90deg, red, blue" ,
{
angleSwatchClass: "test-angleswatch" ,
}
);
swatchCount = frag.querySelectorAll(".test-angleswatch" ).length;
is(swatchCount, 1, "angle swatch was created" );
}
function testParseShape(doc, parser) {
info("Test shape parsing" );
const tests = [
{
desc: "Polygon shape" ,
definition:
"polygon(evenodd, 0px 0px, 10%200px,30%30% , calc(250px - 10px) 0 ,\n " +
"12em var(--variable), 100% 100%) margin-box" ,
spanCount: 18,
},
{
desc: "POLYGON()" ,
definition:
"POLYGON(evenodd, 0px 0px, 10%200px,30%30% , calc(250px - 10px) 0 ,\n " +
"12em var(--variable), 100% 100%) margin-box" ,
spanCount: 18,
},
{
desc: "Invalid polygon shape" ,
definition: "polygon(0px 0px 100px 20px, 20% 20%)" ,
spanCount: 0,
},
{
desc: "Circle shape with all arguments" ,
definition: "circle(25% at\n 30% 200px) border-box" ,
spanCount: 4,
},
{
desc: "Circle shape with only one center" ,
definition: "circle(25em at 40%)" ,
spanCount: 3,
},
{
desc: "Circle shape with no radius" ,
definition: "circle(at 30% 40%)" ,
spanCount: 3,
},
{
desc: "Circle shape with no center" ,
definition: "circle(12em)" ,
spanCount: 1,
},
{
desc: "Circle shape with no arguments" ,
definition: "circle()" ,
spanCount: 0,
},
{
desc: "Circle shape with no space before at" ,
definition: "circle(25%at 30% 30%)" ,
spanCount: 4,
},
{
desc: "CIRCLE" ,
definition: "CIRCLE(12em)" ,
spanCount: 1,
},
{
desc: "Invalid circle shape" ,
definition: "circle(25%at30%30%)" ,
spanCount: 0,
},
{
desc: "Ellipse shape with all arguments" ,
definition: "ellipse(200px 10em at 25% 120px) content-box" ,
spanCount: 5,
},
{
desc: "Ellipse shape with only one center" ,
definition: "ellipse(200px 10% at 120px)" ,
spanCount: 4,
},
{
desc: "Ellipse shape with no radius" ,
definition: "ellipse(at 25% 120px)" ,
spanCount: 3,
},
{
desc: "Ellipse shape with no center" ,
definition: "ellipse(200px\n10em)" ,
spanCount: 2,
},
{
desc: "Ellipse shape with no arguments" ,
definition: "ellipse()" ,
spanCount: 0,
},
{
desc: "ELLIPSE()" ,
definition: "ELLIPSE(200px 10em)" ,
spanCount: 2,
},
{
desc: "Invalid ellipse shape" ,
definition: "ellipse(200px100px at 30$ 20%)" ,
spanCount: 0,
},
{
desc: "Inset shape with 4 arguments" ,
definition: "inset(200px 100px\n 30%15%)" ,
spanCount: 4,
},
{
desc: "Inset shape with 3 arguments" ,
definition: "inset(200px 100px 15%)" ,
spanCount: 3,
},
{
desc: "Inset shape with 2 arguments" ,
definition: "inset(200px 100px)" ,
spanCount: 2,
},
{
desc: "Inset shape with 1 argument" ,
definition: "inset(200px)" ,
spanCount: 1,
},
{
desc: "Inset shape with 0 arguments" ,
definition: "inset()" ,
spanCount: 0,
},
{
desc: "INSET()" ,
definition: "INSET(200px)" ,
spanCount: 1,
},
{
desc: "offset-path property with inset shape value" ,
property: "offset-path" ,
definition: "inset(200px)" ,
spanCount: 1,
},
];
for (const { desc, definition, property = "clip-path" , spanCount } of tests) {
info(desc);
const frag = parser.parseCssProperty(property, definition, {
shapeClass: "inspector-shape" ,
});
const spans = frag.querySelectorAll(".inspector-shape-point" );
is(spans.length, spanCount, desc + " span count" );
is(frag.textContent, definition, desc + " text content" );
}
}
function testParseVariable(doc, parser) {
const TESTS = [
{
text: "var(--seen)" ,
variables: { "--seen" : "chartreuse" },
expected:
// prettier-ignore
'' +
"var(" +
'--seen )' +
" " +
" " ,
},
{
text: "var(--seen)" ,
variables: {
"--seen" : { value: "var(--base)" , computedValue: "1em" },
},
expected:
// prettier-ignore
"var(" +
'--seen )' +
" " ,
},
{
text: "var(--not-seen)" ,
variables: {},
expected:
// prettier-ignore
"var(" +
'--not-seen ' +
") " ,
},
{
text: "var(--seen, seagreen)" ,
variables: { "--seen" : "chartreuse" },
expected:
// prettier-ignore
'' +
"var(" +
'--seen ,' +
' ' +
'' +
"seagreen " +
" " +
" )" +
" " +
" " ,
},
{
text: "var(--not-seen, var(--seen))" ,
variables: { "--seen" : "chartreuse" },
expected:
// prettier-ignore
"var(" +
'--not-seen ,' +
" " +
'' +
"var(" +
'--seen )' +
" " +
" " +
" )" +
" " ,
},
{
text: "color-mix(in sgrb, var(--x), purple)" ,
variables: { "--x" : "yellow" },
expected:
// prettier-ignore
`color-mix(in sgrb, ` +
`<span data-color="yellow" class ="color-swatch-container" >` +
`<span class ="test-class" style="background-color:yellow" tabindex="0" role="button" data-color-function="color-mix" >` +
`</span>` +
`<span>var (<span data-variable="yellow" >--x</span>)</span>` +
`</span>` +
`, ` +
`<span data-color="purple" class ="color-swatch-container" >` +
`<span class ="test-class" style="background-color:purple" tabindex="0" role="button" data-color-function="color-mix" >` +
`</span>` +
`<span>purple</span>` +
`</span>` +
`)`,
parserExtraOptions: {
colorSwatchClass: COLOR_TEST_CLASS,
},
},
{
text: "light-dark(var(--light), var(--dark))" ,
variables: { "--light" : "yellow" , "--dark" : "gold" },
expected:
// prettier-ignore
`light-dark(` +
`<span data-color="yellow" class ="color-swatch-container" >` +
`<span class ="test-class" style="background-color:yellow" tabindex="0" role="button" data-color-function="light-dark" >` +
`</span>` +
`<span>var (<span data-variable="yellow" >--light</span>)</span>` +
`</span>` +
`, ` +
`<span data-color="gold" class ="color-swatch-container" >` +
`<span class ="test-class" style="background-color:gold" tabindex="0" role="button" data-color-function="light-dark" >` +
`</span>` +
`<span>var (<span data-variable="gold" >--dark</span>)</span>` +
`</span>` +
`)`,
parserExtraOptions: {
colorSwatchClass: COLOR_TEST_CLASS,
},
},
{
text: "1px solid var(--seen, seagreen)" ,
// See Bug 1911974
skipVariableDeclarationTest: true ,
variables: { "--seen" : "chartreuse" },
expected:
// prettier-ignore
'1px solid ' +
'' +
"var(" +
'--seen ,' +
' ' +
'' +
"seagreen " +
" " +
" )" +
" " +
" " ,
},
{
text: "1px solid var(--not-seen, seagreen)" ,
// See Bug 1911975
skipVariableDeclarationTest: true ,
variables: {},
expected:
// prettier-ignore
`1px solid ` +
`<span>var (` +
`<span class ="unmatched-class" data-variable="--not-seen is not set" >--not-seen</span>,` +
`<span> ` +
`<span data-color="seagreen" >` +
`<span>seagreen</span>` +
`</span>` +
`</span>)` +
`</span>`,
},
{
text: "rgba(var(--r), 0, 0, var(--a))" ,
variables: { "--r" : "255" , "--a" : "0.5" },
expected:
// prettier-ignore
'' +
"rgba(" +
"" +
'var(--r )' +
" , 0, 0, " +
"" +
'var(--a )' +
" " +
") " +
" " ,
},
{
text: "rgba(from var(--base) r g 0 / calc(var(--a) * 0.5))" ,
variables: { "--base" : "red" , "--a" : "0.8" },
expected:
// prettier-ignore
'' +
"rgba(" +
"from " +
"" +
'var(--base )' +
" r g 0 / " +
"calc(" +
"" +
'var(--a )' +
" " +
" * 0.5)" +
") " +
" " ,
},
{
text: "rgb(var(--not-seen, 255), 0, 0)" ,
variables: {},
expected:
// prettier-ignore
'' +
"rgb(" +
"var(" +
`<span class ="unmatched-class" data-variable="--not-seen is not set" >--not-seen</span>,` +
`<span> 255</span>` +
") , 0, 0" +
") " +
" " ,
},
{
text: "rgb(var(--not-seen), 0, 0)" ,
variables: {},
expected:
// prettier-ignore
`rgb(` +
`<span>` +
`var (` +
`<span class ="unmatched-class" data-variable="--not-seen is not set" >` +
`--not-seen` +
`</span>` +
`)` +
`</span>` +
`, 0, 0` +
`)`,
},
{
text: "var(--registered)" ,
variables: {
"--registered" : {
value: "chartreuse" ,
registeredProperty: {
syntax: "" ,
inherits: true ,
initialValue: "hotpink" ,
},
},
},
expected:
// prettier-ignore
'' +
"var(" +
' +
'data-variable="chartreuse" ' +
'data-registered-property-initial-value="hotpink" ' +
'data-registered-property-syntax="" ' +
'data-registered-property-inherits="true"' +
'>--registered )' +
" " +
" " ,
},
{
text: "var(--registered-universal)" ,
variables: {
"--registered-universal" : {
value: "chartreuse" ,
registeredProperty: {
syntax: "*" ,
inherits: false ,
},
},
},
expected:
// prettier-ignore
'' +
"var(" +
' +
'data-variable="chartreuse" ' +
'data-registered-property-syntax="*" ' +
'data-registered-property-inherits="false"' +
'>--registered-universal )' +
" " +
" " ,
},
{
text: "var(--x)" ,
variables: {
"--x" : "light-dark(red, blue)" ,
},
parserExtraOptions: {
isDarkColorScheme: false ,
},
expected:
'var(--x ) ' ,
},
{
text: "var(--x)" ,
variables: {
"--x" : "color-mix(in srgb, red 50%, blue)" ,
},
parserExtraOptions: {
isDarkColorScheme: false ,
},
expected:
// prettier-ignore
'' +
'var(' +
'--x ' +
') ' +
' ' ,
},
{
text: "var(--refers-empty)" ,
variables: {
"--refers-empty" : { value: "var(--empty)" , computedValue: "" },
},
expected:
// prettier-ignore
"var(" +
'--refers-empty )' +
" " ,
},
{
text: "hsl(50, 70%, var(--foo))" ,
variables: {
"--foo" : "40%" ,
},
expected:
// prettier-ignore
`<span data-color="hsl(50, 70%, 40%)" >` +
`<span>`+
`hsl(50, 70%, ` +
`<span>` +
`var (` +
`<span data-variable="40%" >--foo</span>` +
`)` +
`</span>)` +
`</span>` +
`</span>`,
},
{
text: "var(--bar)" ,
variables: {
"--foo" : "40%" ,
"--bar" : "hsl(50, 70%, var(--foo))" ,
},
expected:
// prettier-ignore
`<span data-color="hsl(50, 70%, 40%)" >` +
`<span>` +
`var (` +
`<span data-variable="hsl(50, 70%, var(--foo))" data-variable-computed="hsl(50, 70%, 40%)" >--bar</span>` +
`)` +
`</span>` +
`</span>`,
},
{
text: "var(--primary)" ,
variables: {
"--primary" : "hsl(10, 100%, var(--fur))" ,
"--fur" : "var(--bar)" ,
"--bar" : "var(--foo)" ,
"--foo" : "50%" ,
},
expected:
// prettier-ignore
`<span data-color="hsl(10, 100%, 50%)" >` +
`<span>` +
`var (` +
`<span data-variable="hsl(10, 100%, var(--fur))" data-variable-computed="hsl(10, 100%, 50%)" >--primary</span>` +
`)` +
`</span>` +
`</span>`,
},
{
text: "oklch(var(--fur) 20 var(--boo))" ,
variables: {
"--fur" : "var(--baz)" ,
"--baz" : "var(--foo)" ,
"--foo" : "10" ,
"--boo" : "30" ,
},
expected:
// prettier-ignore
`<span data-color="oklch(10 20 30)" >` +
`<span>oklch(` +
`<span>` +
`var (` +
`<span data-variable="var(--baz)" data-variable-computed="10" >--fur</span>` +
`)` +
`</span>` +
` 20 ` +
`<span>` +
`var (` +
`<span data-variable="30" >--boo</span>` +
`)` +
`</span>` +
`)</span>` +
`</span>`,
},
];
const target = doc.querySelector("div" );
const VAR_NAME_TO_DEFINE = "--test-parse-variable" ;
for (const test of TESTS) {
// VAR_NAME_TO_DEFINE is used to test parsing the test.text if it's set for a
// variable declaration, so it shouldn't be set in test.variables to avoid
// messing with the test results.
if (VAR_NAME_TO_DEFINE in test.variables) {
throw new Error(`${VAR_NAME_TO_DEFINE} shouldn't be set in variables`);
}
// Also set the variable we're going to define, so its value can be computed as well
const variables = {
...(test.variables || {}),
[VAR_NAME_TO_DEFINE]: test.text,
};
// Set the variables to an element so we can get their computed values
for (const [varName, varData] of Object.entries(variables)) {
doc.body.style.setProperty(
varName,
typeof varData === "string" ? varData : varData.value
);
}
const getVariableData = function (varName) {
if (typeof variables[varName] === "string" ) {
const value = variables[varName];
const computedValue = getComputedStyle(doc.body).getPropertyValue(
varName
);
return { value, computedValue };
}
return variables[varName] || {};
};
const frag = parser.parseCssProperty("color" , test.text, {
getVariableData,
unmatchedClass: "unmatched-class" ,
...(test.parserExtraOptions || {}),
});
target.appendChild(frag);
is(
target.innerHTML,
test.expected,
`"color: ${test.text}" is parsed as expected`
);
target.innerHTML = "" ;
if (test.skipVariableDeclarationTest) {
continue ;
}
const varFrag = parser.parseCssProperty(
"--test-parse-variable" ,
test.text,
{
getVariableData,
unmatchedClass: "unmatched-class" ,
...(test.parserExtraOptions || {}),
}
);
target.appendChild(varFrag);
is(
target.innerHTML,
test.expected,
`"--test-parse-variable: ${test.text}" is parsed as expected`
);
target.innerHTML = "" ;
// Remove the variables to an element so we can get their computed values
for (const varName in variables || {}) {
doc.body.style.removeProperty(varName);
}
}
}
function testParseColorVariable(doc, parser) {
const testCategories = [
{
desc: "Test for CSS variable defining color" ,
tests: [
makeColorTest("--test-var" , "lime" , [{ name: "lime" }]),
makeColorTest("--test-var" , "#000" , [{ name: "#000" }]),
],
},
{
desc: "Test for CSS variable not defining color" ,
tests: [
makeColorTest("--foo" , "something" , ["something" ]),
makeColorTest("--bar" , "Arial Black" , ["Arial Black" ]),
makeColorTest("--baz" , "10vmin" , ["10vmin" ]),
],
},
{
desc: "Test for non CSS variable defining color" ,
tests: [
makeColorTest("non-css-variable" , "lime" , ["lime" ]),
makeColorTest("-non-css-variable" , "#000" , ["#000" ]),
],
},
];
for (const category of testCategories) {
info(category.desc);
for (const test of category.tests) {
info(test.desc);
const target = doc.querySelector("div" );
const frag = parser.parseCssProperty(test.name, test.value, {
colorSwatchClass: COLOR_TEST_CLASS,
});
target.appendChild(frag);
is(
target.innerHTML,
test.expected,
`The parsed result for '${test.name}: ${test.value}' is correct`
);
target.innerHTML = "" ;
}
}
}
function testParseFontFamily(doc, parser) {
info("Test font-family parsing" );
const tests = [
{
desc: "No fonts" ,
definition: "" ,
families: [],
},
{
desc: "List of fonts" ,
definition: "Arial,Helvetica,sans-serif" ,
families: ["Arial" , "Helvetica" , "sans-serif" ],
},
{
desc: "Fonts with spaces" ,
definition: "Open Sans" ,
families: ["Open Sans" ],
},
{
desc: "Quoted fonts" ,
definition: "\" Arial\",'Open Sans'" ,
families: ["Arial" , "Open Sans" ],
},
{
desc: "Fonts with extra whitespace" ,
definition: " Open Sans " ,
families: ["Open Sans" ],
},
];
const textContentTests = [
{
desc: "No whitespace between fonts" ,
definition: "Arial,Helvetica,sans-serif" ,
output: "Arial,Helvetica,sans-serif" ,
},
{
desc: "Whitespace between fonts" ,
definition: "Arial , Helvetica, sans-serif" ,
output: "Arial , Helvetica, sans-serif" ,
},
{
desc: "Whitespace before first font trimmed" ,
definition: " Arial,Helvetica,sans-serif" ,
output: "Arial,Helvetica,sans-serif" ,
},
{
desc: "Whitespace after last font trimmed" ,
definition: "Arial,Helvetica,sans-serif " ,
output: "Arial,Helvetica,sans-serif" ,
},
{
desc: "Whitespace between quoted fonts" ,
definition: "'Arial' , \" Helvetica\" " ,
output: "'Arial' , \" Helvetica\"" ,
},
{
desc: "Whitespace within font preserved" ,
definition: "' Ari al '" ,
output: "' Ari al '" ,
},
];
for (const { desc, definition, families } of tests) {
info(desc);
const frag = parser.parseCssProperty("font-family" , definition, {
fontFamilyClass: "ruleview-font-family" ,
});
const spans = frag.querySelectorAll(".ruleview-font-family" );
is(spans.length, families.length, desc + " span count" );
for (let i = 0; i < spans.length; i++) {
is(spans[i].textContent, families[i], desc + " span contents" );
}
}
info("Test font-family text content" );
for (const { desc, definition, output } of textContentTests) {
info(desc);
const frag = parser.parseCssProperty("font-family" , definition, {});
is(frag.textContent, output, desc + " text content matches" );
}
info("Test font-family with custom properties" );
const frag = parser.parseCssProperty(
"font-family" ,
"var(--family, Georgia, serif)" ,
{
getVariableData: () => ({}),
unmatchedClass: "unmatched-class" ,
fontFamilyClass: "ruleview-font-family" ,
}
);
const target = doc.createElement("div" );
target.appendChild(frag);
is(
target.innerHTML,
// prettier-ignore
`<span>var (` +
`<span class ="unmatched-class" data-variable="--family is not set" >` +
`--family` +
`</span>` +
`,` +
`<span> ` +
`<span class ="ruleview-font-family" >Georgia</span>` +
`, ` +
`<span class ="ruleview-font-family" >serif</span>` +
`</span>)` +
`</span>`,
"Got expected output for font-family with custom properties"
);
}
function testParseLightDark(doc, parser) {
const TESTS = [
{
message:
"Not passing isDarkColorScheme doesn't add unmatched classes to parameters" ,
propertyName: "color" ,
propertyValue: "light-dark(red, blue)" ,
expected:
// prettier-ignore
`light-dark(` +
`<span data-color="red" class ="color-swatch-container" >` +
`<span class ="test-class" style="background-color:red" tabindex="0" role="button" data-color-function="light-dark" ></span>` +
`<span>red</span>` +
`</span>, ` +
`<span data-color="blue" class ="color-swatch-container" >` +
`<span class ="test-class" style="background-color:blue" tabindex="0" role="button" data-color-function="light-dark" ></span>` +
`<span>blue</span>` +
`</span>` +
`)`,
},
{
message: "in light mode, the second parameter gets the unmatched class" ,
propertyName: "color" ,
propertyValue: "light-dark(red, blue)" ,
isDarkColorScheme: false ,
expected:
// prettier-ignore
`light-dark(` +
`<span data-color="red" class ="color-swatch-container" >` +
`<span class ="test-class" style="background-color:red" tabindex="0" role="button" data-color-function="light-dark" ></span>` +
`<span>red</span>` +
`</span>, ` +
`<span data-color="blue" class ="color-swatch-container unmatched-class" >` +
`<span class ="test-class" style="background-color:blue" tabindex="0" role="button" data-color-function="light-dark" ></span>` +
`<span>blue</span>` +
`</span>` +
`)`,
},
{
message: "in dark mode, the first parameter gets the unmatched class" ,
propertyName: "color" ,
propertyValue: "light-dark(red, blue)" ,
isDarkColorScheme: true ,
expected:
// prettier-ignore
`light-dark(` +
`<span data-color="red" class ="color-swatch-container unmatched-class" >` +
`<span class ="test-class" style="background-color:red" tabindex="0" role="button" data-color-function="light-dark" ></span>` +
`<span>red</span>` +
`</span>, ` +
`<span data-color="blue" class ="color-swatch-container" >` +
`<span class ="test-class" style="background-color:blue" tabindex="0" role="button" data-color-function="light-dark" ></span>` +
`<span>blue</span>` +
`</span>` +
`)`,
},
{
message: "light-dark gets parsed as expected in shorthands in light mode" ,
propertyName: "border" ,
propertyValue: "1px solid light-dark(red, blue)" ,
isDarkColorScheme: false ,
expected:
// prettier-ignore
`1px solid light-dark(` +
`<span data-color="red" class ="color-swatch-container" >` +
`<span class ="test-class" style="background-color:red" tabindex="0" role="button" data-color-function="light-dark" ></span>` +
`<span>red</span>` +
`</span>, ` +
`<span data-color="blue" class ="color-swatch-container unmatched-class" >` +
`<span class ="test-class" style="background-color:blue" tabindex="0" role="button" data-color-function="light-dark" ></span>` +
`<span>blue</span>` +
`</span>` +
`)`,
},
{
message: "light-dark gets parsed as expected in shorthands in dark mode" ,
propertyName: "border" ,
propertyValue: "1px solid light-dark(red, blue)" ,
isDarkColorScheme: true ,
expected:
// prettier-ignore
`1px solid light-dark(` +
`<span data-color="red" class ="color-swatch-container unmatched-class" >` +
`<span class ="test-class" style="background-color:red" tabindex="0" role="button" data-color-function="light-dark" ></span>` +
`<span>red</span>` +
`</span>, ` +
`<span data-color="blue" class ="color-swatch-container" >` +
`<span class ="test-class" style="background-color:blue" tabindex="0" role="button" data-color-function="light-dark" ></span>` +
`<span>blue</span>` +
`</span>` +
`)`,
},
{
message: "Nested light-dark gets parsed as expected in light mode" ,
propertyName: "background" ,
propertyValue:
"linear-gradient(45deg, light-dark(red, blue), light-dark(pink, cyan))" ,
isDarkColorScheme: false ,
expected:
// prettier-ignore
`linear-gradient(` +
`<span data-angle="45deg" ><span>45deg</span></span>, ` +
`light-dark(` +
`<span data-color="red" class ="color-swatch-container" >` +
`<span class ="test-class" style="background-color:red" tabindex="0" role="button" data-color-function="light-dark" ></span>`+
`<span>red</span>`+
`</span>, `+
`<span data-color="blue" class ="color-swatch-container unmatched-class" >` +
`<span class ="test-class" style="background-color:blue" tabindex="0" role="button" data-color-function="light-dark" ></span>` +
`<span>blue</span>` +
`</span>` +
`), ` +
`light-dark(` +
`<span data-color="pink" class ="color-swatch-container" >` +
`<span class ="test-class" style="background-color:pink" tabindex="0" role="button" data-color-function="light-dark" ></span>` +
`<span>pink</span>` +
`</span>, ` +
`<span data-color="cyan" class ="color-swatch-container unmatched-class" >` +
`<span class ="test-class" style="background-color:cyan" tabindex="0" role="button" data-color-function="light-dark" ></span>` +
`<span>cyan</span>` +
`</span>` +
`)` +
`)`,
},
{
message: "Nested light-dark gets parsed as expected in dark mode" ,
propertyName: "background" ,
propertyValue:
"linear-gradient(33deg, light-dark(red, blue), light-dark(pink, cyan))" ,
isDarkColorScheme: true ,
expected:
// prettier-ignore
`linear-gradient(` +
`<span data-angle="33deg" ><span>33deg</span></span>, ` +
`light-dark(` +
`<span data-color="red" class ="color-swatch-container unmatched-class" >` +
`<span class ="test-class" style="background-color:red" tabindex="0" role="button" data-color-function="light-dark" ></span>`+
`<span>red</span>`+
`</span>, `+
`<span data-color="blue" class ="color-swatch-container" >` +
`<span class ="test-class" style="background-color:blue" tabindex="0" role="button" data-color-function="light-dark" ></span>` +
`<span>blue</span>` +
`</span>` +
`), ` +
`light-dark(` +
`<span data-color="pink" class ="color-swatch-container unmatched-class" >` +
`<span class ="test-class" style="background-color:pink" tabindex="0" role="button" data-color-function="light-dark" ></span>` +
`<span>pink</span>` +
`</span>, ` +
`<span data-color="cyan" class ="color-swatch-container" >` +
`<span class ="test-class" style="background-color:cyan" tabindex="0" role="button" data-color-function="light-dark" ></span>` +
`<span>cyan</span>` +
`</span>` +
`)` +
`)`,
},
{
message:
"in light mode, the second parameter gets the unmatched class when it's a variable" ,
propertyName: "color" ,
propertyValue: "light-dark(var(--x), var(--y))" ,
isDarkColorScheme: false ,
variables: { "--x" : "red" , "--y" : "blue" },
expected:
// prettier-ignore
`light-dark(` +
`<span data-color="red" class ="color-swatch-container" >` +
`<span class ="test-class" style="background-color:red" tabindex="0" role="button" data-color-function="light-dark" ></span>` +
`<span>var (` +
`<span data-variable="red" >--x</span>` +
`)</span>` +
`</span>, ` +
`<span data-color="blue" class ="color-swatch-container unmatched-class" >` +
`<span class ="test-class" style="background-color:blue" tabindex="0" role="button" data-color-function="light-dark" ></span>` +
`<span>var (` +
`<span data-variable="blue" >--y</span>` +
`)</span>` +
`</span>` +
`)`,
},
{
message:
"in light mode, the second parameter gets the unmatched class when some param are not parsed" ,
propertyName: "color" ,
// Using `notacolor` so we don't get a wrapping Node for it (contrary to colors).
// The value is still valid at parse time since we're using a variable,
// so the OutputParser will actually parse the different parts
propertyValue: "light-dark(var(--x),notacolor)" ,
isDarkColorScheme: false ,
variables: { "--x" : "red" },
expected:
// prettier-ignore
`light-dark(` +
`<span data-color="red" class ="color-swatch-container" >` +
`<span class ="test-class" style="background-color:red" tabindex="0" role="button" data-color-function="light-dark" ></span>` +
`<span>` +
`var (<span data-variable="red" >--x</span>)` +
`</span>` +
`</span>,` +
`<span class ="unmatched-class" >notacolor</span>` +
`)`,
},
{
message:
"in dark mode, the first parameter gets the unmatched class when some param are not parsed" ,
propertyName: "color" ,
// Using `notacolor` so we don't get a wrapping Node for it (contrary to colors).
// The value is still valid at parse time since we're using a variable,
// so the OutputParser will actually parse the different parts
propertyValue: "light-dark(notacolor,var(--x))" ,
isDarkColorScheme: true ,
variables: { "--x" : "red" },
expected:
// prettier-ignore
`light-dark(` +
`<span class ="unmatched-class" >notacolor</span>,` +
`<span data-color="red" class ="color-swatch-container" >` +
`<span class ="test-class" style="background-color:red" tabindex="0" role="button" data-color-function="light-dark" ></span>` +
`<span>` +
`var (<span data-variable="red" >--x</span>)` +
`</span>` +
`</span>` +
`)`,
},
{
message:
"in light mode, the second parameter gets the unmatched class, comments are stripped out and whitespace are preserved" ,
propertyName: "color" ,
propertyValue:
"light-dark( /* 1st param */ var(--x) /* delim */ , /* 2nd param */ notacolor /* delim */ )",
isDarkColorScheme: false ,
variables: { "--x" : "red" },
expected:
// prettier-ignore
`light-dark( ` +
`<span data-color="red" class ="color-swatch-container" >` +
`<span class ="test-class" style="background-color:red" tabindex="0" role="button" data-color-function="light-dark" ></span>` +
`<span>` +
`var (<span data-variable="red" >--x</span>)` +
`</span>` +
`</span> , ` +
`<span class ="unmatched-class" >notacolor</span> ` +
`)`,
},
{
message:
"in dark mode, the first parameter gets the unmatched class, comments are stripped out and whitespace are preserved" ,
propertyName: "color" ,
propertyValue:
"light-dark( /* 1st param */ notacolor /* delim */ , /* 2nd param */ var(--x) /* delim */ )",
isDarkColorScheme: true ,
variables: { "--x" : "red" },
expected:
// prettier-ignore
`light-dark( ` +
`<span class ="unmatched-class" >notacolor</span> , ` +
`<span data-color="red" class ="color-swatch-container" >` +
`<span class ="test-class" style="background-color:red" tabindex="0" role="button" data-color-function="light-dark" ></span>` +
`<span>` +
`var (<span data-variable="red" >--x</span>)` +
`</span>` +
`</span> ` +
`)`,
},
{
message:
"in light mode with a single parameter, we don't strike through any parameter (TODO wrap with IACVT - Bug 1910845)" ,
propertyName: "color" ,
propertyValue: "light-dark(var(--x))" ,
isDarkColorScheme: false ,
variables: { "--x" : "red" },
expected:
// prettier-ignore
`light-dark(` +
`<span data-color="red" class ="color-swatch-container" >` +
`<span class ="test-class" style="background-color:red" tabindex="0" role="button" data-color-function="light-dark" ></span>` +
`<span>` +
`var (<span data-variable="red" >--x</span>)` +
`</span>` +
`</span>` +
`)`,
},
{
message:
"in dark mode with a single parameter, we don't strike through any parameter (TODO wrap with IACVT - Bug 1910845)" ,
propertyName: "color" ,
propertyValue: "light-dark(var(--x))" ,
isDarkColorScheme: true ,
variables: { "--x" : "red" },
expected:
// prettier-ignore
`light-dark(` +
`<span data-color="red" class ="color-swatch-container" >` +
`<span class ="test-class" style="background-color:red" tabindex="0" role="button" data-color-function="light-dark" ></span>` +
`<span>` +
`var (<span data-variable="red" >--x</span>)` +
`</span>` +
`</span>` +
`)`,
},
{
message:
"in light mode with 3 parameters, we don't strike through any parameter (TODO wrap with IACVT - Bug 1910845)" ,
propertyName: "color" ,
propertyValue: "light-dark(var(--x),a,b)" ,
isDarkColorScheme: false ,
variables: { "--x" : "red" },
expected:
// prettier-ignore
`light-dark(` +
`<span data-color="red" class ="color-swatch-container" >` +
`<span class ="test-class" style="background-color:red" tabindex="0" role="button" data-color-function="light-dark" ></span>` +
`<span>` +
`var (<span data-variable="red" >--x</span>)` +
`</span>` +
`</span>,a,b` +
`)`,
},
{
message:
"in dark mode with 3 parameters, we don't strike through any parameter (TODO wrap with IACVT - Bug 1910845)" ,
propertyName: "color" ,
propertyValue: "light-dark(var(--x),a,b)" ,
isDarkColorScheme: true ,
variables: { "--x" : "red" },
expected:
// prettier-ignore
`light-dark(` +
`<span data-color="red" class ="color-swatch-container" >` +
`<span class ="test-class" style="background-color:red" tabindex="0" role="button" data-color-function="light-dark" ></span>` +
`<span>` +
`var (<span data-variable="red" >--x</span>)` +
`</span>` +
`</span>,a,b` +
`)`,
},
];
for (const test of TESTS) {
const frag = parser.parseCssProperty(
test.propertyName,
test.propertyValue,
{
isDarkColorScheme: test.isDarkColorScheme,
unmatchedClass: "unmatched-class" ,
colorSwatchClass: COLOR_TEST_CLASS,
getVariableData: varName => {
if (typeof test.variables[varName] === "string" ) {
return { value: test.variables[varName] };
}
return test.variables[varName] || {};
},
}
);
const target = doc.querySelector("div" );
target.appendChild(frag);
is(target.innerHTML, test.expected, test.message);
target.innerHTML = "" ;
}
}
quality 93%
¤ Dauer der Verarbeitung: 0.27 Sekunden
¤
*© Formatika GbR, Deutschland