/* Any copyright is dedicated to the Public Domain. http://creativecommons.org/publicdomain/zero/1.0/ */ "use strict"; const { getClientCssProperties, } = require("resource://devtools/client/fronts/css-properties.js"); add_task(async function () { await pushPref("layout.css.backdrop-filter.enabled", true); await pushPref("layout.css.individual-transform.enabled", true); await pushPref("layout.css.color-mix.enabled", true); await addTab("about:blank"); await performTest(); gBrowser.removeCurrentTab(); }); async function performTest() { await SpecialPowers.pushPrefEnv({ set: [["security.allow_unsafe_parent_loads", 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); 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 += `` + ``+ `${segment.name}` + ``; } } 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" }, ")", ] ), ]; 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, }); 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 + '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: "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: "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: "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, }, ]; for (const { desc, definition, spanCount } of tests) { info(desc); const frag = parser.parseCssProperty("clip-path", definition, { shapeClass: "ruleview-shape", }); const spans = frag.querySelectorAll(".ruleview-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(--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, ` + `` + `` + `` + `var(--x)` + `` + `, ` + `` + `` + `` + `purple` + `` + `)`, parserExtraOptions: { colorSwatchClass: COLOR_TEST_CLASS, }, }, ]; for (const test of TESTS) { const getValue = function (varName) { return test.variables[varName]; }; const frag = parser.parseCssProperty("color", test.text, { getVariableValue: getValue, unmatchedVariableClass: "unmatched-class", ...(test.parserExtraOptions || {}), }); const target = doc.querySelector("div"); target.appendChild(frag); is(target.innerHTML, test.expected, test.text); target.innerHTML = ""; } } 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"); } }