@projectwallace/css-analyzer 5.2.1 → 5.5.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/analyzer.cjs +1 -1
- package/dist/analyzer.cjs.map +1 -1
- package/dist/analyzer.modern.js +1 -1
- package/dist/analyzer.modern.js.map +1 -1
- package/dist/analyzer.module.js +1 -1
- package/dist/analyzer.module.js.map +1 -1
- package/dist/analyzer.umd.js +1 -1
- package/dist/analyzer.umd.js.map +1 -1
- package/dist/index.d.ts +57 -22
- package/package.json +6 -7
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"analyzer.module.js","sources":["../src/string-utils.js","../src/selectors/specificity.js","../src/values/colors.js","../src/countable-collection.js","../src/values/font-families.js","../src/values/font-sizes.js","../src/values/values.js","../src/values/animations.js","../src/vendor-prefix.js","../src/values/vendor-prefix.js","../src/atrules/atrules.js","../src/context-collection.js","../src/aggregate-collection.js","../src/properties/property-utils.js","../src/index.js","../src/rules/rules.js"],"sourcesContent":["/**\n * Case-insensitive compare two character codes\n * @param {string} charA\n * @param {string} charB\n * @see https://github.com/csstree/csstree/blob/41f276e8862d8223eeaa01a3d113ab70bb13d2d9/lib/tokenizer/utils.js#L22\n */\nfunction compareChar(referenceCode, testCode) {\n // if uppercase\n if (testCode >= 0x0041 && testCode <= 0x005A) {\n // shifting the 6th bit makes a letter lowercase\n testCode = testCode | 32\n }\n return referenceCode === testCode\n}\n\n/**\n * Case-insensitive string-comparison\n * @param {string} base\n * @param {string} test\n * @returns {boolean} true if the two strings are the same, false otherwise\n */\nexport function strEquals(base, test) {\n if (base.length !== test.length) return false\n\n for (let i = 0; i < base.length; i++) {\n if (compareChar(base.charCodeAt(i), test.charCodeAt(i)) === false) {\n return false\n }\n }\n\n return true\n}\n\n/**\n * Case-insensitive testing whether a string ends with a given substring\n * @param {string} base e.g. '-webkit-transform'\n * @param {string} cmp e.g. 'transform'\n * @returns {boolean} true if `test` ends with `base`, false otherwise\n */\nexport function endsWith(base, test) {\n const offset = test.length - base.length\n\n if (offset < 0) {\n return false\n }\n\n for (let i = test.length - 1; i >= offset; i--) {\n if (compareChar(base.charCodeAt(i - offset), test.charCodeAt(i)) === false) {\n return false\n }\n }\n\n return true\n}\n\n/**\n * Case-insensitive testing whether a string starts with a given substring\n * @param {string} base\n * @param {test} test\n * @returns {boolean} true if `test` starts with `base`, false otherwise\n */\nexport function startsWith(base, test) {\n if (test.length < base.length) return false\n\n for (let i = 0; i < base.length; i++) {\n if (compareChar(base.charCodeAt(i), test.charCodeAt(i)) === false) {\n return false\n }\n }\n\n return true\n}\n","import walk from 'css-tree/walker'\nimport { startsWith } from '../string-utils.js'\n\n/**\n * Compare specificity A to Specificity B\n * @param {[number,number,number]} a - Specificity A\n * @param {[number,number,number]} b - Specificity B\n * @returns {number} sortIndex - 0 when a==b, 1 when a<b, -1 when a>b\n */\nfunction compareSpecificity(a, b) {\n if (a[0] === b[0]) {\n if (a[1] === b[1]) {\n return b[2] - a[2]\n }\n\n return b[1] - a[1]\n }\n\n return b[0] - a[0]\n}\n\n/**\n *\n * @param {import('css-tree').SelectorList} selectorListAst\n * @returns {Selector} topSpecificitySelector\n */\nfunction selectorListSpecificities(selectorListAst) {\n const childSelectors = []\n walk(selectorListAst, {\n visit: 'Selector',\n enter(node) {\n childSelectors.push(analyzeSpecificity(node))\n }\n })\n\n return childSelectors.sort((a, b) => compareSpecificity(a.specificity, b.specificity))\n}\n\n/**\n * Get the Specificity for the AST of a Selector Node\n * @param {import('css-tree').Selector} ast - AST Node for a Selector\n * @return {Object}\n * @property {[number,number,number]} specificity\n * @property {number} complexity\n * @property {Boolean} isId\n * @property {Boolean} isA11y\n */\nconst analyzeSpecificity = (node) => {\n let A = 0\n let B = 0\n let C = 0\n let complexity = 0\n let isA11y = false\n\n walk(node, function (selector) {\n switch (selector.type) {\n case 'IdSelector': {\n A++\n complexity++\n break\n }\n case 'ClassSelector': {\n B++\n complexity++\n break\n }\n case 'AttributeSelector': {\n B++\n complexity++\n\n // Add 1 for [attr=value] (or any variation using *= $= ^= |= )\n if (Boolean(selector.value)) {\n complexity++\n }\n\n isA11y = selector.name.name === 'role' || startsWith('aria-', selector.name.name)\n break\n }\n case 'PseudoElementSelector':\n case 'TypeSelector': {\n complexity++\n\n // 42 === '*'.charCodeAt(0)\n if (selector.name.charCodeAt(0) === 42 && selector.name.length === 1) {\n break\n }\n\n C++\n break\n }\n case 'PseudoClassSelector': {\n switch (selector.name) {\n case 'before':\n case 'after':\n case 'first-letter':\n case 'first-line': {\n C++\n complexity++\n return this.skip\n }\n\n // The specificity of an :is(), :not(), or :has() pseudo-class is\n // replaced by the specificity of the most specific complex\n // selector in its selector list argument.\n case 'where':\n case 'is':\n case 'has':\n case 'matches':\n case '-webkit-any':\n case '-moz-any':\n case 'not':\n case 'nth-child':\n case 'nth-last-child': {\n // The specificity of an :nth-child() or :nth-last-child() selector\n // is the specificity of the pseudo class itself (counting as one\n // pseudo-class selector) plus the specificity of the most\n // specific complex selector in its selector list argument (if any).\n if (selector.name === 'nth-child' || selector.name === 'nth-last-child') {\n // +1 for the pseudo class itself\n B++\n }\n\n const selectorList = selectorListSpecificities(selector)\n\n // Bail out for empty/non-existent :nth-child() params\n if (selectorList.length === 0) return\n\n // The specificity of a :where() pseudo-class is replaced by zero,\n // but it does count towards complexity.\n if (selector.name !== 'where') {\n const [topA, topB, topC] = selectorList[0].specificity\n A += topA\n B += topB\n C += topC\n }\n\n for (let i = 0; i < selectorList.length; i++) {\n const listItem = selectorList[i]\n if (listItem.isA11y) {\n isA11y = true\n }\n complexity += listItem.complexity\n }\n\n complexity++\n return this.skip\n }\n\n default: {\n // Regular pseudo classes have specificity [0,1,0]\n complexity++\n B++\n return this.skip\n }\n }\n }\n case 'Combinator': {\n complexity++\n break\n }\n }\n })\n\n return {\n /** @type {[number,number,number]} */\n specificity: [A, B, C],\n complexity,\n isId: A > 0,\n isA11y\n }\n}\n\nexport {\n analyzeSpecificity,\n compareSpecificity,\n}","export const colorNames = {\n // CSS Named Colors\n // Spec: https://drafts.csswg.org/css-color/#named-colors\n aliceblue: 1,\n antiquewhite: 1,\n aqua: 1,\n aquamarine: 1,\n azure: 1,\n beige: 1,\n bisque: 1,\n black: 1,\n blanchedalmond: 1,\n blue: 1,\n blueviolet: 1,\n brown: 1,\n burlywood: 1,\n cadetblue: 1,\n chartreuse: 1,\n chocolate: 1,\n coral: 1,\n cornflowerblue: 1,\n cornsilk: 1,\n crimson: 1,\n cyan: 1,\n darkblue: 1,\n darkcyan: 1,\n darkgoldenrod: 1,\n darkgray: 1,\n darkgreen: 1,\n darkgrey: 1,\n darkkhaki: 1,\n darkmagenta: 1,\n darkolivegreen: 1,\n darkorange: 1,\n darkorchid: 1,\n darkred: 1,\n darksalmon: 1,\n darkseagreen: 1,\n darkslateblue: 1,\n darkslategray: 1,\n darkslategrey: 1,\n darkturquoise: 1,\n darkviolet: 1,\n deeppink: 1,\n deepskyblue: 1,\n dimgray: 1,\n dimgrey: 1,\n dodgerblue: 1,\n firebrick: 1,\n floralwhite: 1,\n forestgreen: 1,\n fuchsia: 1,\n gainsboro: 1,\n ghostwhite: 1,\n gold: 1,\n goldenrod: 1,\n gray: 1,\n green: 1,\n greenyellow: 1,\n grey: 1,\n honeydew: 1,\n hotpink: 1,\n indianred: 1,\n indigo: 1,\n ivory: 1,\n khaki: 1,\n lavender: 1,\n lavenderblush: 1,\n lawngreen: 1,\n lemonchiffon: 1,\n lightblue: 1,\n lightcoral: 1,\n lightcyan: 1,\n lightgoldenrodyellow: 1,\n lightgray: 1,\n lightgreen: 1,\n lightgrey: 1,\n lightpink: 1,\n lightsalmon: 1,\n lightseagreen: 1,\n lightskyblue: 1,\n lightslategray: 1,\n lightslategrey: 1,\n lightsteelblue: 1,\n lightyellow: 1,\n lime: 1,\n limegreen: 1,\n linen: 1,\n magenta: 1,\n maroon: 1,\n mediumaquamarine: 1,\n mediumblue: 1,\n mediumorchid: 1,\n mediumpurple: 1,\n mediumseagreen: 1,\n mediumslateblue: 1,\n mediumspringgreen: 1,\n mediumturquoise: 1,\n mediumvioletred: 1,\n midnightblue: 1,\n mintcream: 1,\n mistyrose: 1,\n moccasin: 1,\n navajowhite: 1,\n navy: 1,\n oldlace: 1,\n olive: 1,\n olivedrab: 1,\n orange: 1,\n orangered: 1,\n orchid: 1,\n palegoldenrod: 1,\n palegreen: 1,\n paleturquoise: 1,\n palevioletred: 1,\n papayawhip: 1,\n peachpuff: 1,\n peru: 1,\n pink: 1,\n plum: 1,\n powderblue: 1,\n purple: 1,\n rebeccapurple: 1,\n red: 1,\n rosybrown: 1,\n royalblue: 1,\n saddlebrown: 1,\n salmon: 1,\n sandybrown: 1,\n seagreen: 1,\n seashell: 1,\n sienna: 1,\n silver: 1,\n skyblue: 1,\n slateblue: 1,\n slategray: 1,\n slategrey: 1,\n snow: 1,\n springgreen: 1,\n steelblue: 1,\n tan: 1,\n teal: 1,\n thistle: 1,\n tomato: 1,\n turquoise: 1,\n violet: 1,\n wheat: 1,\n white: 1,\n whitesmoke: 1,\n yellow: 1,\n yellowgreen: 1,\n\n // CSS System Colors\n // Spec: https://drafts.csswg.org/css-color/#css-system-colors\n canvas: 1,\n canvastext: 1,\n linktext: 1,\n visitedtext: 1,\n activetext: 1,\n buttonface: 1,\n buttontext: 1,\n buttonborder: 1,\n field: 1,\n fieldtext: 1,\n highlight: 1,\n highlighttext: 1,\n selecteditem: 1,\n selecteditemtext: 1,\n mark: 1,\n marktext: 1,\n graytext: 1,\n\n // TODO: Deprecated CSS System colors\n // Spec: https://drafts.csswg.org/css-color/#deprecated-system-colors\n}\n\nexport const colorFunctions = {\n rgb: 1,\n rgba: 1,\n hsl: 1,\n hsla: 1,\n hwb: 1,\n lab: 1,\n lch: 1,\n oklab: 1,\n oklch: 1,\n color: 1,\n}\n","class CountableCollection {\n /**\n * @param {string[]} initial\n */\n constructor(initial) {\n /** @type [index: string]: string */\n this.items = {}\n /** @type number */\n this.total = 0\n /** @type number */\n this.totalUnique = 0\n\n if (initial) {\n for (let index = 0; index < initial.length; index++) {\n this.push(initial[index])\n }\n }\n }\n\n /**\n * Push an item to the end of this collection\n * @param {string} item\n * @returns {void}\n */\n push(item) {\n this.total++\n\n if (this.items[item]) {\n this.items[item]++\n return\n }\n\n this.items[item] = 1\n this.totalUnique++\n }\n\n /**\n * Get the size of this collection\n * @returns {number} the size of this collection\n */\n size() {\n return this.total\n }\n\n /**\n * Get the counts of this collection, like total, uniques, etc.\n */\n count() {\n return {\n total: this.total,\n totalUnique: this.totalUnique,\n unique: this.items,\n uniquenessRatio: this.total === 0 ? 0 : this.totalUnique / this.total,\n }\n }\n}\n\nexport {\n CountableCollection\n}","import walk from 'css-tree/walker'\nimport { CountableCollection } from '../countable-collection.js'\n\nconst systemKeywords = {\n // Global CSS keywords\n 'inherit': 1,\n 'initial': 1,\n 'unset': 1,\n 'revert': 1,\n\n // System font keywords\n 'caption': 1,\n 'icon': 1,\n 'menu': 1,\n 'message-box': 1,\n 'small-caption': 1,\n 'status-bar': 1,\n}\n\nconst keywordDisallowList = {\n // font-weight, font-stretch, font-style\n 'normal': 1,\n\n // font-size keywords\n 'xx-small': 1,\n 'x-small': 1,\n 'small': 1,\n 'medium': 1,\n 'large': 1,\n 'x-large': 1,\n 'xx-large': 1,\n 'larger': 1,\n 'smaller': 1,\n\n // font-weight keywords\n 'bold': 1,\n 'bolder': 1,\n 'lighter': 1,\n\n // font-stretch keywords\n 'ultra-condensed': 1,\n 'extra-condensed': 1,\n 'condensed': 1,\n 'semi-condensed': 1,\n 'semi-expanded': 1,\n 'expanded': 1,\n 'extra-expanded': 1,\n 'ultra-expanded': 1,\n\n // font-style keywords\n 'italic': 1,\n 'oblique': 1,\n}\n\nconst COMMA = 44 // ','.charCodeAt(0) === 44\n\nconst analyzeFontFamilies = ({ fontValues, fontFamilyValues, stringifyNode }) => {\n const all = new CountableCollection(fontFamilyValues)\n\n for (let index = 0; index < fontValues.length; index++) {\n const value = fontValues[index]\n\n // Avoid tree traversal as soon as possible\n const firstChild = value.children.first\n\n if (firstChild.type === 'Identifier' && systemKeywords[firstChild.name]) {\n continue\n }\n\n let parts = ''\n\n walk(value, {\n reverse: true,\n enter: function (fontNode) {\n if (fontNode.type === 'String') {\n const loc = fontNode.loc.start\n // Stringify the first character to get the correct quote character\n const quote = stringifyNode({\n loc: {\n start: {\n line: loc.line,\n column: loc.column\n },\n end: {\n line: loc.line,\n column: loc.column + 1\n }\n }\n })\n return parts = quote + fontNode.value + quote + parts\n }\n if (fontNode.type === 'Operator' && fontNode.value.charCodeAt(0) === COMMA) {\n return parts = fontNode.value + parts\n }\n if (fontNode.type === 'Identifier') {\n if (keywordDisallowList[fontNode.name]) {\n return this.skip\n }\n return parts = fontNode.name + parts\n }\n }\n })\n\n all.push(parts)\n }\n\n return all.count()\n}\n\nexport {\n analyzeFontFamilies\n}","import walk from 'css-tree/walker'\nimport { CountableCollection } from '../countable-collection.js'\n\nconst sizeKeywords = {\n 'xx-small': 1,\n 'x-small': 1,\n 'small': 1,\n 'medium': 1,\n 'large': 1,\n 'x-large': 1,\n 'xx-large': 1,\n 'larger': 1,\n 'smaller': 1,\n}\n\nconst keywords = {\n // Global CSS keywords\n 'inherit': 1,\n 'initial': 1,\n 'unset': 1,\n 'revert': 1,\n\n // System font keywords\n 'caption': 1,\n 'icon': 1,\n 'menu': 1,\n 'message-box': 1,\n 'small-caption': 1,\n 'status-bar': 1,\n}\n\nconst ZERO = 48 // '0'.charCodeAt(0) === 48\nconst SLASH = 47 // '/'.charCodeAt(0) === 47\n\nconst analyzeFontSizes = ({ stringifyNode, fontSizeValues, fontValues }) => {\n const all = new CountableCollection(fontSizeValues)\n\n for (let index = 0; index < fontValues.length; index++) {\n const fontNode = fontValues[index];\n // Try to eliminate a keyword before we continue\n const firstChild = fontNode.children.first\n\n if (firstChild.type === 'Identifier' && keywords[firstChild.name]) {\n continue\n }\n\n let operator = false\n let size\n\n walk(fontNode, function (fontNode) {\n switch (fontNode.type) {\n case 'Number': {\n // Special case for `font: 0/0 a`\n if (fontNode.value.charCodeAt(0) === ZERO) {\n size = '0'\n return this.break\n }\n }\n case 'Operator': {\n if (fontNode.value.charCodeAt(0) === SLASH) {\n operator = true\n }\n break\n }\n case 'Dimension': {\n if (!operator) {\n size = stringifyNode(fontNode)\n return this.break\n }\n }\n case 'Identifier': {\n if (sizeKeywords[fontNode.name]) {\n size = fontNode.name\n return this.break\n }\n }\n }\n })\n\n if (size) {\n all.push(size)\n }\n }\n\n return all.count()\n}\n\nexport {\n analyzeFontSizes\n}","import { CountableCollection } from '../countable-collection.js'\n\nconst keywords = {\n 'auto': 1,\n 'inherit': 1,\n 'initial': 1,\n 'unset': 1,\n 'revert': 1,\n 'none': 1, // for `text-shadow` and `box-shadow`\n}\n\nconst analyzeValues = ({ values, stringifyNode }) => {\n const all = new CountableCollection()\n\n for (let i = 0; i < values.length; i++) {\n const node = values[i]\n const firstChild = node.children.first\n\n if (!firstChild) continue\n if (firstChild.type === 'Identifier' && keywords[firstChild.name]) continue\n\n all.push(stringifyNode(node))\n }\n\n return all.count()\n}\n\nexport {\n analyzeValues\n}","import { CountableCollection } from '../countable-collection.js'\n\nconst timingKeywords = {\n 'linear': 1,\n 'ease': 1,\n 'ease-in': 1,\n 'ease-out': 1,\n 'ease-in-out': 1,\n 'step-start': 1,\n 'step-end': 1,\n}\n\nconst analyzeAnimations = ({ animations, durations, timingFunctions, stringifyNode }) => {\n const allDurations = new CountableCollection(durations)\n const allTimingFunctions = new CountableCollection(timingFunctions)\n\n for (let index = 0; index < animations.length; index++) {\n // Flag to know if we've grabbed the first Duration\n // yet (the first Dimension in a shorthand)\n let durationFound = false\n\n animations[index].forEach(child => {\n // Right after a ',' we start over again\n if (child.type === 'Operator') {\n return durationFound = false\n }\n if (child.type === 'Dimension' && durationFound === false) {\n durationFound = true\n return allDurations.push(stringifyNode(child))\n }\n if (child.type === 'Identifier' && timingKeywords[child.name]) {\n return allTimingFunctions.push(stringifyNode(child))\n }\n if (child.type === 'Function'\n && (\n child.name === 'cubic-bezier' || child.name === 'steps'\n )\n ) {\n return allTimingFunctions.push(stringifyNode(child))\n }\n })\n }\n\n return {\n durations: allDurations.count(),\n timingFunctions: allTimingFunctions.count(),\n }\n}\n\nexport {\n analyzeAnimations\n}","const HYPHENMINUS = 45; // '-'.charCodeAt()\n\n/**\n * @param {string} keyword\n * @returns {boolean}\n */\nfunction hasVendorPrefix(keyword) {\n if (keyword.charCodeAt(0) === HYPHENMINUS && keyword.charCodeAt(1) !== HYPHENMINUS) {\n // String must have a 2nd occurrence of '-', at least at position 3 (offset=2)\n if (keyword.indexOf('-', 2) !== -1) {\n return true\n }\n }\n\n return false\n}\n\nexport {\n hasVendorPrefix\n}","import { CountableCollection } from '../countable-collection.js'\nimport { hasVendorPrefix } from '../vendor-prefix.js'\n\nfunction isAstVendorPrefixed(children) {\n children = children.toArray()\n\n for (let index = 0; index < children.length; index++) {\n const child = children[index];\n\n if (child.type === 'Identifier' && child.name.length >= 3) {\n if (hasVendorPrefix(child.name)) {\n return true\n }\n }\n\n if (child.type === 'Function') {\n if (hasVendorPrefix(child.name)) {\n return true\n }\n\n if (child.children && isAstVendorPrefixed(child.children)) {\n return true\n }\n }\n }\n return false\n}\n\nconst analyzeVendorPrefixes = ({ values, stringifyNode }) => {\n const all = new CountableCollection()\n\n for (let i = 0; i < values.length; i++) {\n /** @type {import('css-tree').Value} */\n const value = values[i]\n\n if (value.children && isAstVendorPrefixed(value.children)) {\n all.push(stringifyNode(value))\n }\n }\n\n return all.count()\n}\n\nexport {\n analyzeVendorPrefixes\n}","import { CountableCollection } from '../countable-collection.js'\nimport { hasVendorPrefix } from '../vendor-prefix.js'\nimport { endsWith } from '../string-utils.js'\n\nconst analyzeAtRules = ({ atrules, stringifyNode }) => {\n /** @type {{[index: string]: string}[]} */\n const fontfaces = []\n const layers = new CountableCollection()\n const imports = new CountableCollection()\n const medias = new CountableCollection()\n const charsets = new CountableCollection()\n const supports = new CountableCollection()\n const keyframes = new CountableCollection()\n const prefixedKeyframes = new CountableCollection()\n const containers = new CountableCollection()\n\n const machine = {\n 'font-face': (node) => {\n /** @type {[index: string]: string} */\n const descriptors = {}\n\n node.block.children.forEach(\n /** @param {import('css-tree').Declaration} descriptor */\n descriptor => (descriptors[descriptor.property] = stringifyNode(descriptor.value))\n )\n\n fontfaces.push(descriptors)\n },\n 'media': node => medias.push(node.prelude),\n 'supports': node => supports.push(node.prelude),\n 'keyframes': node => keyframes.push(`@${node.name} ${node.prelude}`),\n 'import': node => imports.push(node.prelude),\n 'charset': node => charsets.push(node.prelude),\n 'container': node => containers.push(node.prelude),\n 'layer': node => {\n node.prelude.trim()\n .split(',')\n .map(name => name.trim())\n .forEach(name => layers.push(name))\n },\n }\n\n for (let i = 0; i < atrules.length; i++) {\n /** @type {import('css-tree').Atrule} */\n const node = atrules[i]\n const atRuleName = node.name\n const action = machine[atRuleName]\n if (action) {\n action(node)\n continue\n }\n\n if (endsWith('keyframes', atRuleName)) {\n const name = `@${atRuleName} ${node.prelude}`\n keyframes.push(name)\n\n if (hasVendorPrefix(atRuleName)) {\n prefixedKeyframes.push(name)\n }\n continue\n }\n }\n\n return {\n fontface: {\n total: fontfaces.length,\n totalUnique: fontfaces.length,\n unique: fontfaces,\n uniquenessRatio: fontfaces.length === 0 ? 0 : 1\n },\n import: imports.count(),\n media: medias.count(),\n charset: charsets.count(),\n supports: supports.count(),\n keyframes: {\n ...keyframes.count(),\n prefixed: {\n ...prefixedKeyframes.count(),\n ratio: keyframes.size() === 0 ? 0 : prefixedKeyframes.size() / keyframes.size()\n }\n },\n container: containers.count(),\n layer: layers.count(),\n }\n}\n\nexport {\n analyzeAtRules\n}","import { CountableCollection } from './countable-collection.js'\n\nclass ContextCollection {\n constructor() {\n this.list = new CountableCollection()\n /** @type {[index; string]: CountableCollection} */\n this.contexts = {}\n }\n\n /**\n * Add an item to this list's context\n * @param {string} item Item to push\n * @param {string} context Context to push Item to\n */\n push(item, context) {\n this.list.push(item)\n\n if (!this.contexts[context]) {\n this.contexts[context] = new CountableCollection()\n }\n\n this.contexts[context].push(item)\n }\n\n count() {\n /** @type {[index: string]: string} */\n const itemsPerContext = {}\n\n for (let context in this.contexts) {\n itemsPerContext[context] = this.contexts[context].count()\n }\n\n return Object.assign(this.list.count(), {\n itemsPerContext\n })\n }\n}\n\nexport {\n ContextCollection\n}","/**\n * Find the mode (most occurring value) in an array of Numbers\n * Takes the mean/average of multiple values if multiple values occur the same amount of times.\n *\n * @see https://github.com/angus-c/just/blob/684af9ca0c7808bc78543ec89379b1fdfce502b1/packages/array-mode/index.js\n * @param {Array} arr - Array to find the mode value for\n * @returns {Number} mode - The `mode` value of `arr`\n */\nfunction Mode(arr) {\n const frequencies = Object.create(null)\n let maxOccurrences = -1\n let maxOccurenceCount = 0\n let sum = 0\n\n for (let i = 0; i < arr.length; i++) {\n const element = arr[i]\n const updatedCount = (frequencies[element] || 0) + 1\n frequencies[element] = updatedCount\n\n if (updatedCount > maxOccurrences) {\n maxOccurrences = updatedCount\n maxOccurenceCount = 0\n sum = 0\n }\n\n if (updatedCount >= maxOccurrences) {\n maxOccurenceCount++\n sum += element\n }\n }\n\n return sum / maxOccurenceCount\n}\n\n/**\n * Find the middle number in an Array of Numbers\n * Returns the average of 2 numbers if the Array length is an even number\n * @see https://github.com/angus-c/just/blob/684af9ca0c7808bc78543ec89379b1fdfce502b1/packages/array-median/index.js\n * @param {Array} arr - A sorted Array\n * @returns {Number} - The array's Median\n */\nfunction Median(arr) {\n const middle = arr.length / 2\n const lowerMiddleRank = Math.floor(middle)\n\n if (middle !== lowerMiddleRank) {\n return arr[lowerMiddleRank]\n }\n return (arr[lowerMiddleRank] + arr[lowerMiddleRank - 1]) / 2\n}\n\nclass AggregateCollection {\n constructor() {\n /** @type number[] */\n this.items = []\n }\n\n /**\n * Add a new Integer at the end of this AggregateCollection\n * @param {number} item - The item to add\n */\n add(item) {\n this.items.push(item)\n }\n\n aggregate() {\n if (this.items.length === 0) {\n return {\n min: 0,\n max: 0,\n mean: 0,\n mode: 0,\n median: 0,\n range: 0,\n sum: 0,\n }\n }\n\n /** @type Number[] */\n const sorted = this.items.slice().sort((a, b) => a - b)\n const min = sorted[0]\n const max = sorted[sorted.length - 1]\n\n const sum = this.items.reduce((total, num) => (total += num))\n const mode = Mode(this.items)\n const median = Median(sorted)\n\n return {\n min,\n max,\n mean: sum / this.items.length,\n mode,\n median,\n range: max - min,\n sum,\n }\n }\n\n /**\n * @returns {number[]} All items in this collection\n */\n toArray() {\n return this.items\n }\n}\n\nexport {\n AggregateCollection\n}","import { hasVendorPrefix } from '../vendor-prefix.js'\nimport { endsWith } from '../string-utils.js'\n\n/**\n * @param {string} property\n * @see https://github.com/csstree/csstree/blob/master/lib/utils/names.js#L69\n */\nexport function isHack(property) {\n if (isCustom(property) || hasVendorPrefix(property)) return false\n\n let code = property.charCodeAt(0)\n\n return code === 47 // /\n || code === 95 // _\n || code === 43 // +\n || code === 42 // *\n || code === 38 // &\n || code === 36 // $\n || code === 35 // #\n}\n\nexport function isCustom(property) {\n if (property.length < 3) return false\n // 45 === '-'.charCodeAt(0)\n return property.charCodeAt(0) === 45 && property.charCodeAt(1) === 45\n}\n\nexport function isProperty(basename, property) {\n if (isCustom(property)) return false\n return endsWith(basename, property)\n}","import parse from 'css-tree/parser'\nimport walk from 'css-tree/walker'\nimport { analyzeRule } from './rules/rules.js'\nimport { analyzeSpecificity, compareSpecificity } from './selectors/specificity.js'\nimport { colorFunctions, colorNames } from './values/colors.js'\nimport { analyzeFontFamilies } from './values/font-families.js'\nimport { analyzeFontSizes } from './values/font-sizes.js'\nimport { analyzeValues } from './values/values.js'\nimport { analyzeAnimations } from './values/animations.js'\nimport { analyzeVendorPrefixes } from './values/vendor-prefix.js'\nimport { analyzeAtRules } from './atrules/atrules.js'\nimport { ContextCollection } from './context-collection.js'\nimport { CountableCollection } from './countable-collection.js'\nimport { AggregateCollection } from './aggregate-collection.js'\nimport { strEquals, startsWith, endsWith } from './string-utils.js'\nimport { hasVendorPrefix } from './vendor-prefix.js'\nimport { isCustom, isHack, isProperty } from './properties/property-utils.js'\n\n/**\n * Analyze CSS\n * @param {string} css\n */\nconst analyze = (css) => {\n const start = new Date()\n\n // We need all lines later on when we need to stringify the AST again\n // e.g. for Selectors\n const lines = css.split(/\\r?\\n/)\n\n /**\n * Recreate the authored CSS from a CSSTree node\n * @param {import('css-tree').CssNode} node - Node from CSSTree AST to stringify\n * @returns {string} str - The stringified node\n */\n function stringifyNode(node) {\n const start = node.loc.start\n const end = node.loc.end\n const lineCount = end.line - start.line\n\n // Single-line nodes\n if (lineCount === 0) {\n return lines[start.line - 1].substring(start.column - 1, end.column - 1)\n }\n\n // Multi-line nodes\n let value = ''\n\n for (let i = start.line; i <= end.line; i++) {\n const line = lines[i - 1]\n // First line\n if (i === start.line) {\n value += line.substring(start.column - 1) + '\\n'\n continue\n }\n // Last line\n if (i === end.line) {\n value += line.substring(0, end.column - 1)\n continue\n }\n // All lines in between first and last\n value += line + '\\n'\n }\n\n return value\n }\n\n const startParse = new Date()\n let totalComments = 0\n let commentsSize = 0\n\n const ast = parse(css, {\n parseAtrulePrelude: false,\n parseCustomProperty: true, // To find font-families, colors, etc.\n positions: true, // So we can use stringifyNode()\n onComment: function (comment) {\n totalComments++\n commentsSize += comment.length\n },\n })\n\n const startAnalysis = new Date()\n const embeds = new CountableCollection()\n const atrules = []\n\n let totalRules = 0\n let emptyRules = 0\n const selectorsPerRule = new AggregateCollection()\n const declarationsPerRule = new AggregateCollection()\n\n const keyframeSelectors = new CountableCollection()\n const declarationsCache = Object.create(null)\n let totalDeclarations = 0\n let importantDeclarations = 0\n let importantsInKeyframes = 0\n\n const properties = new CountableCollection()\n const propertyHacks = new CountableCollection()\n const propertyVendorPrefixes = new CountableCollection()\n const customProperties = new CountableCollection()\n\n const values = []\n const zindex = []\n const textShadows = []\n const boxShadows = []\n const fontValues = []\n const fontFamilyValues = []\n const fontSizeValues = []\n const animations = []\n const timingFunctions = []\n const durations = []\n const colors = new ContextCollection()\n const units = new ContextCollection()\n\n // SELECTORS\n /** @type number */\n const selectorCounts = Object.create(null)\n /** @type [number,number,number] */\n let maxSpecificity\n /** @type [number,number,number] */\n let minSpecificity\n let specificityA = new AggregateCollection()\n let specificityB = new AggregateCollection()\n let specificityC = new AggregateCollection()\n const complexityAggregator = new AggregateCollection()\n /** @type [number,number,number][] */\n const specificities = []\n /** @type number[] */\n const complexities = []\n const ids = new CountableCollection()\n const a11y = new CountableCollection()\n\n walk(ast, function (node) {\n switch (node.type) {\n case 'Atrule': {\n atrules.push({\n name: node.name,\n prelude: node.prelude && node.prelude.value,\n block: strEquals('font-face', node.name) && node.block,\n })\n break\n }\n case 'Rule': {\n const [numSelectors, numDeclarations] = analyzeRule(node)\n\n totalRules++\n\n if (numDeclarations === 0) {\n emptyRules++\n }\n\n selectorsPerRule.add(numSelectors)\n declarationsPerRule.add(numDeclarations)\n break\n }\n case 'Selector': {\n const selector = stringifyNode(node)\n\n if (this.atrule && endsWith('keyframes', this.atrule.name)) {\n keyframeSelectors.push(selector)\n return this.skip\n }\n\n const { specificity, complexity, isId, isA11y } = analyzeSpecificity(node)\n\n if (isId) {\n ids.push(selector)\n }\n\n if (isA11y) {\n a11y.push(selector)\n }\n\n if (selectorCounts[selector]) {\n selectorCounts[selector]++\n } else {\n selectorCounts[selector] = 1\n }\n\n complexityAggregator.add(complexity)\n\n if (maxSpecificity === undefined) {\n maxSpecificity = specificity\n }\n\n if (minSpecificity === undefined) {\n minSpecificity = specificity\n }\n\n specificityA.add(specificity[0])\n specificityB.add(specificity[1])\n specificityC.add(specificity[2])\n\n if (minSpecificity !== undefined && compareSpecificity(minSpecificity, specificity) < 0) {\n minSpecificity = specificity\n }\n\n if (maxSpecificity !== undefined && compareSpecificity(maxSpecificity, specificity) > 0) {\n maxSpecificity = specificity\n }\n\n specificities.push(specificity)\n complexities.push(complexity)\n\n // Avoid deeper walking of selectors to not mess with\n // our specificity calculations in case of a selector\n // with :where() or :is() that contain SelectorLists\n // as children\n return this.skip\n }\n case 'Dimension': {\n if (!this.declaration) {\n break\n }\n\n units.push(node.unit, this.declaration.property)\n\n return this.skip\n }\n case 'Url': {\n if (startsWith('data:', node.value)) {\n embeds.push(node.value)\n }\n break\n }\n case 'Declaration': {\n totalDeclarations++\n\n const declaration = stringifyNode(node)\n if (declarationsCache[declaration]) {\n declarationsCache[declaration]++\n } else {\n declarationsCache[declaration] = 1\n }\n\n if (node.important) {\n importantDeclarations++\n\n if (this.atrule && endsWith('keyframes', this.atrule.name)) {\n importantsInKeyframes++\n }\n }\n\n const { value, property } = node\n\n properties.push(property)\n values.push(value)\n\n if (hasVendorPrefix(property)) {\n propertyVendorPrefixes.push(property)\n } else if (isHack(property)) {\n propertyHacks.push(property)\n } else if (isCustom(property)) {\n customProperties.push(property)\n }\n\n // Process properties first that don't have colors,\n // so we can avoid further walking them;\n if (isProperty('z-index', property)) {\n zindex.push(value)\n return this.skip\n } else if (isProperty('font', property)) {\n fontValues.push(value)\n break\n } else if (isProperty('font-size', property)) {\n fontSizeValues.push(stringifyNode(value))\n break\n } else if (isProperty('font-family', property)) {\n fontFamilyValues.push(stringifyNode(value))\n break\n } else if (isProperty('transition', property) || isProperty('animation', property)) {\n animations.push(value.children)\n break\n } else if (isProperty('animation-duration', property) || isProperty('transition-duration', property)) {\n durations.push(stringifyNode(value))\n break\n } else if (isProperty('transition-timing-function', property) || isProperty('animation-timing-function', property)) {\n timingFunctions.push(stringifyNode(value))\n break\n } else if (isProperty('text-shadow', property)) {\n textShadows.push(value)\n // no break here: potentially contains colors\n } else if (isProperty('box-shadow', property)) {\n boxShadows.push(value)\n // no break here: potentially contains colors\n }\n\n walk(value, function (valueNode) {\n switch (valueNode.type) {\n case 'Hash': {\n colors.push('#' + valueNode.value, property)\n\n return this.skip\n }\n case 'Identifier': {\n const { name } = valueNode\n // Bail out if it can't be a color name\n // 20 === 'lightgoldenrodyellow'.length\n // 3 === 'red'.length\n if (name.length > 20 || name.length < 3) {\n return this.skip\n }\n if (colorNames[name.toLowerCase()]) {\n colors.push(stringifyNode(valueNode), property)\n }\n return this.skip\n }\n case 'Function': {\n if (colorFunctions[valueNode.name.toLowerCase()]) {\n colors.push(stringifyNode(valueNode), property)\n }\n // No this.skip here intentionally,\n // otherwise we'll miss colors in linear-gradient() etc.\n }\n }\n })\n }\n }\n })\n\n const embeddedContent = embeds.count()\n const embedSize = Object.keys(embeddedContent.unique).join('').length\n\n const totalUniqueDeclarations = Object.keys(declarationsCache).length\n\n const totalSelectors = complexities.length\n const aggregatesA = specificityA.aggregate()\n const aggregatesB = specificityB.aggregate()\n const aggregatesC = specificityC.aggregate()\n const complexityCount = new CountableCollection(complexities).count()\n const totalUniqueSelectors = Object.values(selectorCounts).length\n\n return {\n stylesheet: {\n sourceLinesOfCode: atrules.length + totalSelectors + totalDeclarations + keyframeSelectors.size(),\n linesOfCode: lines.length,\n size: css.length,\n comments: {\n total: totalComments,\n size: commentsSize,\n },\n embeddedContent: Object.assign(embeddedContent, {\n size: {\n total: embedSize,\n ratio: css.length === 0 ? 0 : embedSize / css.length,\n },\n }),\n },\n atrules: analyzeAtRules({ atrules, stringifyNode }),\n rules: {\n total: totalRules,\n empty: {\n total: emptyRules,\n ratio: totalRules === 0 ? 0 : emptyRules / totalRules\n },\n selectors: {\n ...selectorsPerRule.aggregate(),\n items: selectorsPerRule.toArray(),\n },\n declarations: {\n ...declarationsPerRule.aggregate(),\n items: declarationsPerRule.toArray()\n },\n },\n selectors: {\n total: totalSelectors,\n totalUnique: totalUniqueSelectors,\n uniquenessRatio: totalSelectors === 0 ? 0 : totalUniqueSelectors / totalSelectors,\n specificity: {\n min: minSpecificity === undefined ? [0, 0, 0] : minSpecificity,\n max: maxSpecificity === undefined ? [0, 0, 0] : maxSpecificity,\n sum: [aggregatesA.sum, aggregatesB.sum, aggregatesC.sum],\n mean: [aggregatesA.mean, aggregatesB.mean, aggregatesC.mean],\n mode: [aggregatesA.mode, aggregatesB.mode, aggregatesC.mode],\n median: [aggregatesA.median, aggregatesB.median, aggregatesC.median],\n items: specificities\n },\n complexity: {\n ...complexityAggregator.aggregate(),\n ...complexityCount,\n items: complexities,\n },\n id: {\n ...ids.count(),\n ratio: totalSelectors === 0 ? 0 : ids.size() / totalSelectors,\n },\n accessibility: {\n ...a11y.count(),\n ratio: totalSelectors === 0 ? 0 : a11y.size() / totalSelectors,\n },\n keyframes: keyframeSelectors.count(),\n },\n declarations: {\n total: totalDeclarations,\n unique: {\n total: totalUniqueDeclarations,\n ratio: totalDeclarations === 0 ? 0 : totalUniqueDeclarations / totalDeclarations,\n },\n importants: {\n total: importantDeclarations,\n ratio: totalDeclarations === 0 ? 0 : importantDeclarations / totalDeclarations,\n inKeyframes: {\n total: importantsInKeyframes,\n ratio: importantDeclarations === 0 ? 0 : importantsInKeyframes / importantDeclarations,\n },\n },\n },\n properties: {\n ...properties.count(),\n prefixed: {\n ...propertyVendorPrefixes.count(),\n ratio: properties.size() === 0 ? 0 : propertyVendorPrefixes.size() / properties.size(),\n },\n custom: {\n ...customProperties.count(),\n ratio: properties.size() === 0 ? 0 : customProperties.size() / properties.size(),\n },\n browserhacks: {\n ...propertyHacks.count(),\n ratio: properties.size() === 0 ? 0 : propertyHacks.size() / properties.size(),\n }\n },\n values: {\n colors: colors.count(),\n fontFamilies: analyzeFontFamilies({ stringifyNode, fontValues, fontFamilyValues }),\n fontSizes: analyzeFontSizes({ stringifyNode, fontValues, fontSizeValues }),\n zindexes: analyzeValues({ values: zindex, stringifyNode }),\n textShadows: analyzeValues({ values: textShadows, stringifyNode }),\n boxShadows: analyzeValues({ values: boxShadows, stringifyNode }),\n animations: analyzeAnimations({ animations, timingFunctions, durations, stringifyNode }),\n prefixes: analyzeVendorPrefixes({ values, stringifyNode }),\n units: units.count(),\n },\n __meta__: {\n parseTime: startAnalysis - startParse,\n analyzeTime: new Date() - startAnalysis,\n total: new Date() - start\n }\n }\n}\n\nexport {\n analyze,\n compareSpecificity,\n}\n","import walk from 'css-tree/walker'\n\nconst analyzeRule = function (ruleNode) {\n let numSelectors = 0\n let numDeclarations = 0\n\n walk(ruleNode, function (childNode) {\n if (childNode.type === 'Selector') {\n numSelectors++\n return this.skip\n }\n\n if (childNode.type === 'Declaration') {\n numDeclarations++\n return this.skip\n }\n })\n\n return [numSelectors, numDeclarations]\n}\n\nexport {\n analyzeRule,\n}"],"names":["compareChar","referenceCode","testCode","strEquals","base","test","length","i","charCodeAt","endsWith","offset","startsWith","compareSpecificity","a","b","analyzeSpecificity","node","A","B","C","complexity","isA11y","walk","selector","type","Boolean","value","name","skip","selectorList","childSelectors","visit","enter","push","sort","specificity","listItem","isId","colorNames","aliceblue","antiquewhite","aqua","aquamarine","azure","beige","bisque","black","blanchedalmond","blue","blueviolet","brown","burlywood","cadetblue","chartreuse","chocolate","coral","cornflowerblue","cornsilk","crimson","cyan","darkblue","darkcyan","darkgoldenrod","darkgray","darkgreen","darkgrey","darkkhaki","darkmagenta","darkolivegreen","darkorange","darkorchid","darkred","darksalmon","darkseagreen","darkslateblue","darkslategray","darkslategrey","darkturquoise","darkviolet","deeppink","deepskyblue","dimgray","dimgrey","dodgerblue","firebrick","floralwhite","forestgreen","fuchsia","gainsboro","ghostwhite","gold","goldenrod","gray","green","greenyellow","grey","honeydew","hotpink","indianred","indigo","ivory","khaki","lavender","lavenderblush","lawngreen","lemonchiffon","lightblue","lightcoral","lightcyan","lightgoldenrodyellow","lightgray","lightgreen","lightgrey","lightpink","lightsalmon","lightseagreen","lightskyblue","lightslategray","lightslategrey","lightsteelblue","lightyellow","lime","limegreen","linen","magenta","maroon","mediumaquamarine","mediumblue","mediumorchid","mediumpurple","mediumseagreen","mediumslateblue","mediumspringgreen","mediumturquoise","mediumvioletred","midnightblue","mintcream","mistyrose","moccasin","navajowhite","navy","oldlace","olive","olivedrab","orange","orangered","orchid","palegoldenrod","palegreen","paleturquoise","palevioletred","papayawhip","peachpuff","peru","pink","plum","powderblue","purple","rebeccapurple","red","rosybrown","royalblue","saddlebrown","salmon","sandybrown","seagreen","seashell","sienna","silver","skyblue","slateblue","slategray","slategrey","snow","springgreen","steelblue","tan","teal","thistle","tomato","turquoise","violet","wheat","white","whitesmoke","yellow","yellowgreen","canvas","canvastext","linktext","visitedtext","activetext","buttonface","buttontext","buttonborder","field","fieldtext","highlight","highlighttext","selecteditem","selecteditemtext","mark","marktext","graytext","colorFunctions","rgb","rgba","hsl","hsla","hwb","lab","lch","oklab","oklch","color","CountableCollection","initial","this","items","total","totalUnique","index","item","size","count","unique","uniquenessRatio","systemKeywords","inherit","unset","revert","caption","icon","menu","keywordDisallowList","normal","small","medium","large","larger","smaller","bold","bolder","lighter","condensed","expanded","italic","oblique","analyzeFontFamilies","fontValues","stringifyNode","all","fontFamilyValues","firstChild","children","first","parts","reverse","fontNode","loc","start","quote","line","column","end","sizeKeywords","keywords","analyzeFontSizes","fontSizeValues","operator","auto","none","analyzeValues","values","timingKeywords","linear","ease","analyzeAnimations","animations","timingFunctions","allDurations","durations","allTimingFunctions","durationFound","forEach","child","hasVendorPrefix","keyword","indexOf","isAstVendorPrefixed","toArray","analyzeVendorPrefixes","analyzeAtRules","atrules","fontfaces","layers","imports","medias","charsets","supports","keyframes","prefixedKeyframes","containers","machine","descriptors","block","descriptor","property","media","prelude","import","charset","container","layer","trim","split","map","atRuleName","action","fontface","prefixed","ratio","ContextCollection","list","contexts","context","itemsPerContext","Object","assign","AggregateCollection","add","aggregate","min","max","mean","mode","median","range","sum","arr","middle","lowerMiddleRank","sorted","slice","reduce","num","frequencies","create","maxOccurrences","maxOccurenceCount","element","updatedCount","Mode","Math","floor","isCustom","isProperty","basename","analyze","css","Date","lines","substring","maxSpecificity","minSpecificity","startParse","totalComments","commentsSize","ast","parse","parseAtrulePrelude","parseCustomProperty","positions","onComment","comment","startAnalysis","embeds","totalRules","emptyRules","selectorsPerRule","declarationsPerRule","keyframeSelectors","declarationsCache","totalDeclarations","importantDeclarations","importantsInKeyframes","properties","propertyHacks","propertyVendorPrefixes","customProperties","zindex","textShadows","boxShadows","colors","units","selectorCounts","specificityA","specificityB","specificityC","complexityAggregator","specificities","complexities","ids","a11y","ruleNode","numSelectors","numDeclarations","childNode","analyzeRule","atrule","undefined","declaration","unit","important","code","isHack","valueNode","toLowerCase","embeddedContent","embedSize","keys","join","totalUniqueDeclarations","totalSelectors","aggregatesA","aggregatesB","aggregatesC","complexityCount","totalUniqueSelectors","stylesheet","sourceLinesOfCode","linesOfCode","comments","rules","empty","selectors","declarations","id","accessibility","importants","inKeyframes","custom","browserhacks","fontFamilies","fontSizes","zindexes","prefixes","__meta__","parseTime","analyzeTime"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"analyzer.module.js","sources":["../src/string-utils.js","../src/selectors/specificity.js","../src/values/colors.js","../src/values/font-families.js","../src/values/font-sizes.js","../src/values/values.js","../src/values/animations.js","../src/vendor-prefix.js","../src/values/vendor-prefix.js","../src/countable-collection.js","../src/context-collection.js","../src/aggregate-collection.js","../src/properties/property-utils.js","../src/occurrence-counter.js","../src/index.js"],"sourcesContent":["/**\n * Case-insensitive compare two character codes\n * @param {string} charA\n * @param {string} charB\n * @see https://github.com/csstree/csstree/blob/41f276e8862d8223eeaa01a3d113ab70bb13d2d9/lib/tokenizer/utils.js#L22\n */\nfunction compareChar(referenceCode, testCode) {\n // if uppercase\n if (testCode >= 0x0041 && testCode <= 0x005A) {\n // shifting the 6th bit makes a letter lowercase\n testCode = testCode | 32\n }\n return referenceCode === testCode\n}\n\n/**\n * Case-insensitive string-comparison\n * @param {string} base\n * @param {string} test\n * @returns {boolean} true if the two strings are the same, false otherwise\n */\nexport function strEquals(base, test) {\n if (base.length !== test.length) return false\n\n for (let i = 0; i < base.length; i++) {\n if (compareChar(base.charCodeAt(i), test.charCodeAt(i)) === false) {\n return false\n }\n }\n\n return true\n}\n\n/**\n * Case-insensitive testing whether a string ends with a given substring\n * @param {string} base e.g. '-webkit-transform'\n * @param {string} cmp e.g. 'transform'\n * @returns {boolean} true if `test` ends with `base`, false otherwise\n */\nexport function endsWith(base, test) {\n const offset = test.length - base.length\n\n if (offset < 0) {\n return false\n }\n\n for (let i = test.length - 1; i >= offset; i--) {\n if (compareChar(base.charCodeAt(i - offset), test.charCodeAt(i)) === false) {\n return false\n }\n }\n\n return true\n}\n\n/**\n * Case-insensitive testing whether a string starts with a given substring\n * @param {string} base\n * @param {test} test\n * @returns {boolean} true if `test` starts with `base`, false otherwise\n */\nexport function startsWith(base, test) {\n if (test.length < base.length) return false\n\n for (let i = 0; i < base.length; i++) {\n if (compareChar(base.charCodeAt(i), test.charCodeAt(i)) === false) {\n return false\n }\n }\n\n return true\n}\n","import walk from 'css-tree/walker'\nimport { startsWith } from '../string-utils.js'\n\n/**\n * Compare specificity A to Specificity B\n * @param {[number,number,number]} a - Specificity A\n * @param {[number,number,number]} b - Specificity B\n * @returns {number} sortIndex - 0 when a==b, 1 when a<b, -1 when a>b\n */\nfunction compareSpecificity(a, b) {\n if (a[0] === b[0]) {\n if (a[1] === b[1]) {\n return b[2] - a[2]\n }\n\n return b[1] - a[1]\n }\n\n return b[0] - a[0]\n}\n\n/**\n *\n * @param {import('css-tree').SelectorList} selectorListAst\n * @returns {Selector} topSpecificitySelector\n */\nfunction selectorListSpecificities(selectorListAst) {\n const childSelectors = []\n walk(selectorListAst, {\n visit: 'Selector',\n enter(node) {\n childSelectors.push(analyzeSpecificity(node))\n }\n })\n\n return childSelectors.sort((a, b) => compareSpecificity(a.specificity, b.specificity))\n}\n\n/**\n * Get the Specificity for the AST of a Selector Node\n * @param {import('css-tree').Selector} ast - AST Node for a Selector\n * @return {Object}\n * @property {[number,number,number]} specificity\n * @property {number} complexity\n * @property {Boolean} isId\n * @property {Boolean} isA11y\n */\nconst analyzeSpecificity = (node) => {\n let A = 0\n let B = 0\n let C = 0\n let complexity = 0\n let isA11y = false\n\n walk(node, function (selector) {\n switch (selector.type) {\n case 'IdSelector': {\n A++\n complexity++\n break\n }\n case 'ClassSelector': {\n B++\n complexity++\n break\n }\n case 'AttributeSelector': {\n B++\n complexity++\n\n // Add 1 for [attr=value] (or any variation using *= $= ^= |= )\n if (Boolean(selector.value)) {\n complexity++\n }\n\n isA11y = selector.name.name === 'role' || startsWith('aria-', selector.name.name)\n break\n }\n case 'PseudoElementSelector':\n case 'TypeSelector': {\n complexity++\n\n // 42 === '*'.charCodeAt(0)\n if (selector.name.charCodeAt(0) === 42 && selector.name.length === 1) {\n break\n }\n\n C++\n break\n }\n case 'PseudoClassSelector': {\n switch (selector.name) {\n case 'before':\n case 'after':\n case 'first-letter':\n case 'first-line': {\n C++\n complexity++\n return this.skip\n }\n\n // The specificity of an :is(), :not(), or :has() pseudo-class is\n // replaced by the specificity of the most specific complex\n // selector in its selector list argument.\n case 'where':\n case 'is':\n case 'has':\n case 'matches':\n case '-webkit-any':\n case '-moz-any':\n case 'not':\n case 'nth-child':\n case 'nth-last-child': {\n // The specificity of an :nth-child() or :nth-last-child() selector\n // is the specificity of the pseudo class itself (counting as one\n // pseudo-class selector) plus the specificity of the most\n // specific complex selector in its selector list argument (if any).\n if (selector.name === 'nth-child' || selector.name === 'nth-last-child') {\n // +1 for the pseudo class itself\n B++\n }\n\n const selectorList = selectorListSpecificities(selector)\n\n // Bail out for empty/non-existent :nth-child() params\n if (selectorList.length === 0) return\n\n // The specificity of a :where() pseudo-class is replaced by zero,\n // but it does count towards complexity.\n if (selector.name !== 'where') {\n const [topA, topB, topC] = selectorList[0].specificity\n A += topA\n B += topB\n C += topC\n }\n\n for (let i = 0; i < selectorList.length; i++) {\n const listItem = selectorList[i]\n if (listItem.isA11y) {\n isA11y = true\n }\n complexity += listItem.complexity\n }\n\n complexity++\n return this.skip\n }\n\n default: {\n // Regular pseudo classes have specificity [0,1,0]\n complexity++\n B++\n return this.skip\n }\n }\n }\n case 'Combinator': {\n complexity++\n break\n }\n }\n })\n\n return {\n /** @type {[number,number,number]} */\n specificity: [A, B, C],\n complexity,\n isId: A > 0,\n isA11y\n }\n}\n\nexport {\n analyzeSpecificity,\n compareSpecificity,\n}","export const colorNames = {\n // CSS Named Colors\n // Spec: https://drafts.csswg.org/css-color/#named-colors\n aliceblue: 1,\n antiquewhite: 1,\n aqua: 1,\n aquamarine: 1,\n azure: 1,\n beige: 1,\n bisque: 1,\n black: 1,\n blanchedalmond: 1,\n blue: 1,\n blueviolet: 1,\n brown: 1,\n burlywood: 1,\n cadetblue: 1,\n chartreuse: 1,\n chocolate: 1,\n coral: 1,\n cornflowerblue: 1,\n cornsilk: 1,\n crimson: 1,\n cyan: 1,\n darkblue: 1,\n darkcyan: 1,\n darkgoldenrod: 1,\n darkgray: 1,\n darkgreen: 1,\n darkgrey: 1,\n darkkhaki: 1,\n darkmagenta: 1,\n darkolivegreen: 1,\n darkorange: 1,\n darkorchid: 1,\n darkred: 1,\n darksalmon: 1,\n darkseagreen: 1,\n darkslateblue: 1,\n darkslategray: 1,\n darkslategrey: 1,\n darkturquoise: 1,\n darkviolet: 1,\n deeppink: 1,\n deepskyblue: 1,\n dimgray: 1,\n dimgrey: 1,\n dodgerblue: 1,\n firebrick: 1,\n floralwhite: 1,\n forestgreen: 1,\n fuchsia: 1,\n gainsboro: 1,\n ghostwhite: 1,\n gold: 1,\n goldenrod: 1,\n gray: 1,\n green: 1,\n greenyellow: 1,\n grey: 1,\n honeydew: 1,\n hotpink: 1,\n indianred: 1,\n indigo: 1,\n ivory: 1,\n khaki: 1,\n lavender: 1,\n lavenderblush: 1,\n lawngreen: 1,\n lemonchiffon: 1,\n lightblue: 1,\n lightcoral: 1,\n lightcyan: 1,\n lightgoldenrodyellow: 1,\n lightgray: 1,\n lightgreen: 1,\n lightgrey: 1,\n lightpink: 1,\n lightsalmon: 1,\n lightseagreen: 1,\n lightskyblue: 1,\n lightslategray: 1,\n lightslategrey: 1,\n lightsteelblue: 1,\n lightyellow: 1,\n lime: 1,\n limegreen: 1,\n linen: 1,\n magenta: 1,\n maroon: 1,\n mediumaquamarine: 1,\n mediumblue: 1,\n mediumorchid: 1,\n mediumpurple: 1,\n mediumseagreen: 1,\n mediumslateblue: 1,\n mediumspringgreen: 1,\n mediumturquoise: 1,\n mediumvioletred: 1,\n midnightblue: 1,\n mintcream: 1,\n mistyrose: 1,\n moccasin: 1,\n navajowhite: 1,\n navy: 1,\n oldlace: 1,\n olive: 1,\n olivedrab: 1,\n orange: 1,\n orangered: 1,\n orchid: 1,\n palegoldenrod: 1,\n palegreen: 1,\n paleturquoise: 1,\n palevioletred: 1,\n papayawhip: 1,\n peachpuff: 1,\n peru: 1,\n pink: 1,\n plum: 1,\n powderblue: 1,\n purple: 1,\n rebeccapurple: 1,\n red: 1,\n rosybrown: 1,\n royalblue: 1,\n saddlebrown: 1,\n salmon: 1,\n sandybrown: 1,\n seagreen: 1,\n seashell: 1,\n sienna: 1,\n silver: 1,\n skyblue: 1,\n slateblue: 1,\n slategray: 1,\n slategrey: 1,\n snow: 1,\n springgreen: 1,\n steelblue: 1,\n tan: 1,\n teal: 1,\n thistle: 1,\n tomato: 1,\n turquoise: 1,\n violet: 1,\n wheat: 1,\n white: 1,\n whitesmoke: 1,\n yellow: 1,\n yellowgreen: 1,\n\n // CSS System Colors\n // Spec: https://drafts.csswg.org/css-color/#css-system-colors\n canvas: 1,\n canvastext: 1,\n linktext: 1,\n visitedtext: 1,\n activetext: 1,\n buttonface: 1,\n buttontext: 1,\n buttonborder: 1,\n field: 1,\n fieldtext: 1,\n highlight: 1,\n highlighttext: 1,\n selecteditem: 1,\n selecteditemtext: 1,\n mark: 1,\n marktext: 1,\n graytext: 1,\n\n // TODO: Deprecated CSS System colors\n // Spec: https://drafts.csswg.org/css-color/#deprecated-system-colors\n}\n\nexport const colorFunctions = {\n rgb: 1,\n rgba: 1,\n hsl: 1,\n hsla: 1,\n hwb: 1,\n lab: 1,\n lch: 1,\n oklab: 1,\n oklch: 1,\n color: 1,\n}\n","import walk from 'css-tree/walker'\n\nconst systemKeywords = {\n // Global CSS keywords\n 'inherit': 1,\n 'initial': 1,\n 'unset': 1,\n 'revert': 1,\n\n // System font keywords\n 'caption': 1,\n 'icon': 1,\n 'menu': 1,\n 'message-box': 1,\n 'small-caption': 1,\n 'status-bar': 1,\n}\n\nconst keywordDisallowList = {\n // font-weight, font-stretch, font-style\n 'normal': 1,\n\n // font-size keywords\n 'xx-small': 1,\n 'x-small': 1,\n 'small': 1,\n 'medium': 1,\n 'large': 1,\n 'x-large': 1,\n 'xx-large': 1,\n 'larger': 1,\n 'smaller': 1,\n\n // font-weight keywords\n 'bold': 1,\n 'bolder': 1,\n 'lighter': 1,\n\n // font-stretch keywords\n 'ultra-condensed': 1,\n 'extra-condensed': 1,\n 'condensed': 1,\n 'semi-condensed': 1,\n 'semi-expanded': 1,\n 'expanded': 1,\n 'extra-expanded': 1,\n 'ultra-expanded': 1,\n\n // font-style keywords\n 'italic': 1,\n 'oblique': 1,\n}\n\nconst COMMA = 44 // ','.charCodeAt(0) === 44\n\nexport function isFontFamilyKeyword(node) {\n const firstChild = node.children.first\n return firstChild.type === 'Identifier' && systemKeywords[firstChild.name]\n}\n\nexport function getFamilyFromFont(node, stringifyNode) {\n let parts = ''\n\n walk(node, {\n reverse: true,\n enter: function (fontNode) {\n if (fontNode.type === 'String') {\n const loc = fontNode.loc.start\n // Stringify the first character to get the correct quote character\n const quote = stringifyNode({\n loc: {\n start: {\n line: loc.line,\n column: loc.column\n },\n end: {\n line: loc.line,\n column: loc.column + 1\n }\n }\n })\n return parts = quote + fontNode.value + quote + parts\n }\n if (fontNode.type === 'Operator' && fontNode.value.charCodeAt(0) === COMMA) {\n return parts = fontNode.value + parts\n }\n if (fontNode.type === 'Identifier') {\n if (keywordDisallowList[fontNode.name]) {\n return this.skip\n }\n return parts = fontNode.name + parts\n }\n }\n })\n\n return parts\n}\n","import walk from 'css-tree/walker'\n\nconst sizeKeywords = {\n 'xx-small': 1,\n 'x-small': 1,\n 'small': 1,\n 'medium': 1,\n 'large': 1,\n 'x-large': 1,\n 'xx-large': 1,\n 'larger': 1,\n 'smaller': 1,\n}\n\nconst keywords = {\n // Global CSS keywords\n 'inherit': 1,\n 'initial': 1,\n 'unset': 1,\n 'revert': 1,\n\n // System font keywords\n 'caption': 1,\n 'icon': 1,\n 'menu': 1,\n 'message-box': 1,\n 'small-caption': 1,\n 'status-bar': 1,\n}\n\nconst ZERO = 48 // '0'.charCodeAt(0) === 48\nconst SLASH = 47 // '/'.charCodeAt(0) === 47\n\nexport function isFontSizeKeyword(node) {\n const firstChild = node.children.first\n return firstChild.type === 'Identifier' && keywords[firstChild.name]\n}\n\nexport function getSizeFromFont(node) {\n let operator = false\n let size\n\n walk(node, function (fontNode) {\n switch (fontNode.type) {\n case 'Number': {\n // Special case for `font: 0/0 a`\n if (fontNode.value.charCodeAt(0) === ZERO) {\n size = '0'\n return this.break\n }\n }\n case 'Operator': {\n if (fontNode.value.charCodeAt(0) === SLASH) {\n operator = true\n }\n break\n }\n case 'Dimension': {\n if (!operator) {\n size = fontNode.value + fontNode.unit\n return this.break\n }\n }\n case 'Identifier': {\n if (sizeKeywords[fontNode.name]) {\n size = fontNode.name\n return this.break\n }\n }\n }\n })\n\n return size\n}\n","const keywords = {\n 'auto': 1,\n 'inherit': 1,\n 'initial': 1,\n 'unset': 1,\n 'revert': 1,\n 'none': 1, // for `text-shadow`, `box-shadow` and `background`\n}\n\nexport function isValueKeyword(node) {\n if (!node.children) return false\n const firstChild = node.children.first\n\n if (!firstChild) return false\n if (node.children.size > 1) return false\n return firstChild.type === 'Identifier' && keywords[firstChild.name]\n}\n","const timingKeywords = {\n 'linear': 1,\n 'ease': 1,\n 'ease-in': 1,\n 'ease-out': 1,\n 'ease-in-out': 1,\n 'step-start': 1,\n 'step-end': 1,\n}\n\nexport function analyzeAnimation(children, stringifyNode) {\n let durationFound = false\n const durations = []\n const timingFunctions = []\n\n children.forEach(child => {\n // Right after a ',' we start over again\n if (child.type === 'Operator') {\n return durationFound = false\n }\n if (child.type === 'Dimension' && durationFound === false) {\n durationFound = true\n return durations.push(stringifyNode(child))\n }\n if (child.type === 'Identifier' && timingKeywords[child.name]) {\n return timingFunctions.push(stringifyNode(child))\n }\n if (child.type === 'Function'\n && (\n child.name === 'cubic-bezier' || child.name === 'steps'\n )\n ) {\n return timingFunctions.push(stringifyNode(child))\n }\n })\n\n return [durations, timingFunctions]\n}\n","const HYPHENMINUS = 45; // '-'.charCodeAt()\n\n/**\n * @param {string} keyword\n * @returns {boolean}\n */\nfunction hasVendorPrefix(keyword) {\n if (keyword.charCodeAt(0) === HYPHENMINUS && keyword.charCodeAt(1) !== HYPHENMINUS) {\n // String must have a 2nd occurrence of '-', at least at position 3 (offset=2)\n if (keyword.indexOf('-', 2) !== -1) {\n return true\n }\n }\n\n return false\n}\n\nexport {\n hasVendorPrefix\n}","import { hasVendorPrefix } from '../vendor-prefix.js'\n\nexport function isAstVendorPrefixed(node) {\n if (!node.children) {\n return false\n }\n\n const children = node.children.toArray()\n\n for (let index = 0; index < children.length; index++) {\n const node = children[index]\n const { type, name } = node;\n\n if (type === 'Identifier' && hasVendorPrefix(name)) {\n return true\n }\n\n if (type === 'Function') {\n if (hasVendorPrefix(name)) {\n return true\n }\n\n if (isAstVendorPrefixed(node)) {\n return true\n }\n }\n }\n\n return false\n}\n","class CountableCollection {\n /**\n * @param {string[]} initial\n */\n constructor(initial) {\n /** @type [index: string]: string */\n this._items = {}\n /** @type number */\n this._total = 0\n /** @type number */\n this._totalUnique = 0\n\n if (initial) {\n for (let i = 0; i < initial.length; i++) {\n this.push(initial[i])\n }\n }\n }\n\n /**\n * Push an item to the end of this collection\n * @param {string} item\n * @returns {void}\n */\n push(item) {\n this._total++\n\n if (this._items[item]) {\n this._items[item]++\n return\n }\n\n this._items[item] = 1\n this._totalUnique++\n }\n\n /**\n * Get the size of this collection\n * @returns {number} the size of this collection\n */\n size() {\n return this._total\n }\n\n /**\n * Get the counts of this collection, like total, uniques, etc.\n */\n count() {\n return {\n total: this._total,\n totalUnique: this._totalUnique,\n unique: this._items,\n uniquenessRatio: this._total === 0 ? 0 : this._totalUnique / this._total,\n }\n }\n}\n\nexport {\n CountableCollection\n}","import { CountableCollection } from './countable-collection.js'\n\nclass ContextCollection {\n constructor() {\n this._list = new CountableCollection()\n /** @type {[index; string]: CountableCollection} */\n this._contexts = {}\n }\n\n /**\n * Add an item to this _list's context\n * @param {string} item Item to push\n * @param {string} context Context to push Item to\n */\n push(item, context) {\n this._list.push(item)\n\n if (!this._contexts[context]) {\n this._contexts[context] = new CountableCollection()\n }\n\n this._contexts[context].push(item)\n }\n\n count() {\n /** @type {[index: string]: string} */\n const itemsPerContext = {}\n\n for (let context in this._contexts) {\n itemsPerContext[context] = this._contexts[context].count()\n }\n\n return Object.assign(this._list.count(), {\n itemsPerContext\n })\n }\n}\n\nexport {\n ContextCollection\n}","/**\n * Find the mode (most occurring value) in an array of Numbers\n * Takes the mean/average of multiple values if multiple values occur the same amount of times.\n *\n * @see https://github.com/angus-c/just/blob/684af9ca0c7808bc78543ec89379b1fdfce502b1/packages/array-mode/index.js\n * @param {Array} arr - Array to find the mode value for\n * @returns {Number} mode - The `mode` value of `arr`\n */\nfunction Mode(arr) {\n const frequencies = Object.create(null)\n let maxOccurrences = -1\n let maxOccurenceCount = 0\n let sum = 0\n\n for (let i = 0; i < arr.length; i++) {\n const element = arr[i]\n const updatedCount = (frequencies[element] || 0) + 1\n frequencies[element] = updatedCount\n\n if (updatedCount > maxOccurrences) {\n maxOccurrences = updatedCount\n maxOccurenceCount = 0\n sum = 0\n }\n\n if (updatedCount >= maxOccurrences) {\n maxOccurenceCount++\n sum += element\n }\n }\n\n return sum / maxOccurenceCount\n}\n\n/**\n * Find the middle number in an Array of Numbers\n * Returns the average of 2 numbers if the Array length is an even number\n * @see https://github.com/angus-c/just/blob/684af9ca0c7808bc78543ec89379b1fdfce502b1/packages/array-median/index.js\n * @param {Array} arr - A sorted Array\n * @returns {Number} - The array's Median\n */\nfunction Median(arr) {\n const middle = arr.length / 2\n const lowerMiddleRank = Math.floor(middle)\n\n if (middle !== lowerMiddleRank) {\n return arr[lowerMiddleRank]\n }\n return (arr[lowerMiddleRank] + arr[lowerMiddleRank - 1]) / 2\n}\n\nclass AggregateCollection {\n constructor() {\n /** @type number[] */\n this._items = []\n }\n\n /**\n * Add a new Integer at the end of this AggregateCollection\n * @param {number} item - The item to add\n */\n add(item) {\n this._items.push(item)\n }\n\n size() {\n return this._items.length\n }\n\n aggregate() {\n if (this._items.length === 0) {\n return {\n min: 0,\n max: 0,\n mean: 0,\n mode: 0,\n median: 0,\n range: 0,\n sum: 0,\n }\n }\n\n // TODO: can we avoid this sort()? It's slow\n /** @type Number[] */\n const sorted = this._items.slice().sort((a, b) => a - b)\n const min = sorted[0]\n const max = sorted[sorted.length - 1]\n\n const sum = this._items.reduce((total, num) => (total += num))\n const mode = Mode(this._items)\n const median = Median(sorted)\n\n return {\n min,\n max,\n mean: sum / this._items.length,\n mode,\n median,\n range: max - min,\n sum,\n }\n }\n\n /**\n * @returns {number[]} All _items in this collection\n */\n toArray() {\n return this._items\n }\n}\n\nexport {\n AggregateCollection\n}","import { hasVendorPrefix } from '../vendor-prefix.js'\nimport { endsWith } from '../string-utils.js'\n\n/**\n * @param {string} property\n * @see https://github.com/csstree/csstree/blob/master/lib/utils/names.js#L69\n */\nexport function isHack(property) {\n if (isCustom(property) || hasVendorPrefix(property)) return false\n\n let code = property.charCodeAt(0)\n\n return code === 47 // /\n || code === 95 // _\n || code === 43 // +\n || code === 42 // *\n || code === 38 // &\n || code === 36 // $\n || code === 35 // #\n}\n\nexport function isCustom(property) {\n if (property.length < 3) return false\n // 45 === '-'.charCodeAt(0)\n return property.charCodeAt(0) === 45 && property.charCodeAt(1) === 45\n}\n\nexport function isProperty(basename, property) {\n if (isCustom(property)) return false\n return endsWith(basename, property)\n}","export class OccurrenceCounter {\n constructor() {\n this._items = Object.create(null)\n }\n\n push(item) {\n if (this._items[item]) {\n return this._items[item]++\n }\n\n return this._items[item] = 1\n }\n\n count() {\n return Object.keys(this._items).length\n }\n}\n","import parse from 'css-tree/parser'\nimport walk from 'css-tree/walker'\nimport { analyzeSpecificity, compareSpecificity } from './selectors/specificity.js'\nimport { colorFunctions, colorNames } from './values/colors.js'\nimport { isFontFamilyKeyword, getFamilyFromFont } from './values/font-families.js'\nimport { isFontSizeKeyword, getSizeFromFont } from './values/font-sizes.js'\nimport { isValueKeyword } from './values/values.js'\nimport { analyzeAnimation } from './values/animations.js'\nimport { isAstVendorPrefixed } from './values/vendor-prefix.js'\nimport { ContextCollection } from './context-collection.js'\nimport { CountableCollection } from './countable-collection.js'\nimport { AggregateCollection } from './aggregate-collection.js'\nimport { strEquals, startsWith, endsWith } from './string-utils.js'\nimport { hasVendorPrefix } from './vendor-prefix.js'\nimport { isCustom, isHack, isProperty } from './properties/property-utils.js'\nimport { OccurrenceCounter } from './occurrence-counter.js'\n\n/**\n * Analyze CSS\n * @param {string} css\n */\nconst analyze = (css) => {\n const start = new Date()\n\n // We need all lines later on when we need to stringify the AST again\n // e.g. for Selectors\n const lines = css.split(/\\r?\\n/)\n\n /**\n * Recreate the authored CSS from a CSSTree node\n * @param {import('css-tree').CssNode} node - Node from CSSTree AST to stringify\n * @returns {string} str - The stringified node\n */\n function stringifyNode(node) {\n const start = node.loc.start\n const end = node.loc.end\n const lineCount = end.line - start.line\n\n // Single-line nodes\n if (lineCount === 0) {\n return lines[start.line - 1].substring(start.column - 1, end.column - 1)\n }\n\n // Multi-line nodes\n let value = ''\n\n for (let i = start.line; i <= end.line; i++) {\n const line = lines[i - 1]\n // First line\n if (i === start.line) {\n value += line.substring(start.column - 1) + '\\n'\n continue\n }\n // Last line\n if (i === end.line) {\n value += line.substring(0, end.column - 1)\n continue\n }\n // All lines in between first and last\n value += line + '\\n'\n }\n\n return value\n }\n\n // Stylesheet\n let totalComments = 0\n let commentsSize = 0\n const embeds = new CountableCollection()\n\n const startParse = new Date()\n\n const ast = parse(css, {\n parseAtrulePrelude: false,\n parseCustomProperty: true, // To find font-families, colors, etc.\n positions: true, // So we can use stringifyNode()\n onComment: function (comment) {\n totalComments++\n commentsSize += comment.length\n },\n })\n\n const startAnalysis = new Date()\n\n // Atrules\n let totalAtRules = 0\n /** @type {{[index: string]: string}[]} */\n const fontfaces = []\n const layers = new CountableCollection()\n const imports = new CountableCollection()\n const medias = new CountableCollection()\n const charsets = new CountableCollection()\n const supports = new CountableCollection()\n const keyframes = new CountableCollection()\n const prefixedKeyframes = new CountableCollection()\n const containers = new CountableCollection()\n\n // Rules\n let totalRules = 0\n let emptyRules = 0\n const ruleSizes = new AggregateCollection()\n const selectorsPerRule = new AggregateCollection()\n const declarationsPerRule = new AggregateCollection()\n\n // Selectors\n const keyframeSelectors = new CountableCollection()\n /** @type number */\n const uniqueSelectors = new OccurrenceCounter()\n /** @type [number,number,number] */\n let maxSpecificity\n /** @type [number,number,number] */\n let minSpecificity\n const specificityA = new AggregateCollection()\n const specificityB = new AggregateCollection()\n const specificityC = new AggregateCollection()\n const uniqueSpecificities = new CountableCollection()\n const selectorComplexities = new AggregateCollection()\n /** @type [number,number,number][] */\n const specificities = []\n const ids = new CountableCollection()\n const a11y = new CountableCollection()\n\n // Declarations\n const uniqueDeclarations = new OccurrenceCounter()\n let totalDeclarations = 0\n let importantDeclarations = 0\n let importantsInKeyframes = 0\n\n // Properties\n const properties = new CountableCollection()\n const propertyHacks = new CountableCollection()\n const propertyVendorPrefixes = new CountableCollection()\n const customProperties = new CountableCollection()\n\n // Values\n const vendorPrefixedValues = new CountableCollection()\n const zindex = new CountableCollection()\n const textShadows = new CountableCollection()\n const boxShadows = new CountableCollection()\n const fontFamilies = new CountableCollection()\n const fontSizes = new CountableCollection()\n const timingFunctions = new CountableCollection()\n const durations = new CountableCollection()\n const colors = new ContextCollection()\n const units = new ContextCollection()\n\n walk(ast, function (node) {\n switch (node.type) {\n case 'Atrule': {\n totalAtRules++\n const atRuleName = node.name\n\n if (atRuleName === 'font-face') {\n /** @type {[index: string]: string} */\n const descriptors = {}\n\n node.block.children.forEach(\n /** @param {import('css-tree').Declaration} descriptor */\n descriptor => (descriptors[descriptor.property] = stringifyNode(descriptor.value))\n )\n\n fontfaces.push(descriptors)\n break\n }\n if (atRuleName === 'media') {\n medias.push(node.prelude.value)\n break\n }\n if (atRuleName === 'supports') {\n supports.push(node.prelude.value)\n break\n }\n if (endsWith('keyframes', atRuleName)) {\n const name = '@' + atRuleName + ' ' + node.prelude.value\n if (hasVendorPrefix(atRuleName)) {\n prefixedKeyframes.push(name)\n }\n keyframes.push(name)\n break\n }\n if (atRuleName === 'import') {\n imports.push(node.prelude.value)\n break\n }\n if (atRuleName === 'charset') {\n charsets.push(node.prelude.value)\n break\n }\n if (atRuleName === 'container') {\n containers.push(node.prelude.value)\n break\n }\n if (atRuleName === 'layer') {\n containers.push(\n node.prelude.value.trim()\n .split(',')\n .map(name => name.trim())\n .forEach(name => layers.push(name))\n )\n }\n break\n }\n case 'Rule': {\n const numSelectors = node.prelude.children ? node.prelude.children.size : 0\n const numDeclarations = node.block.children ? node.block.children.size : 0\n\n ruleSizes.add(numSelectors + numDeclarations)\n selectorsPerRule.add(numSelectors)\n declarationsPerRule.add(numDeclarations)\n\n totalRules++\n\n if (numDeclarations === 0) {\n emptyRules++\n }\n break\n }\n case 'Selector': {\n const selector = stringifyNode(node)\n\n if (this.atrule && endsWith('keyframes', this.atrule.name)) {\n keyframeSelectors.push(selector)\n return this.skip\n }\n\n const { specificity, complexity, isId, isA11y } = analyzeSpecificity(node)\n\n if (isId) {\n ids.push(selector)\n }\n\n if (isA11y) {\n a11y.push(selector)\n }\n\n uniqueSelectors.push(selector)\n selectorComplexities.add(complexity)\n uniqueSpecificities.push(specificity)\n\n if (maxSpecificity === undefined) {\n maxSpecificity = specificity\n }\n\n if (minSpecificity === undefined) {\n minSpecificity = specificity\n }\n\n specificityA.add(specificity[0])\n specificityB.add(specificity[1])\n specificityC.add(specificity[2])\n\n if (minSpecificity !== undefined && compareSpecificity(minSpecificity, specificity) < 0) {\n minSpecificity = specificity\n }\n\n if (maxSpecificity !== undefined && compareSpecificity(maxSpecificity, specificity) > 0) {\n maxSpecificity = specificity\n }\n\n specificities.push(specificity)\n\n // Avoid deeper walking of selectors to not mess with\n // our specificity calculations in case of a selector\n // with :where() or :is() that contain SelectorLists\n // as children\n return this.skip\n }\n case 'Dimension': {\n if (!this.declaration) {\n break\n }\n\n units.push(node.unit, this.declaration.property)\n\n return this.skip\n }\n case 'Url': {\n if (startsWith('data:', node.value)) {\n embeds.push(node.value)\n }\n break\n }\n case 'Value': {\n if (isValueKeyword(node)) {\n break\n }\n\n const property = this.declaration.property\n\n if (isAstVendorPrefixed(node)) {\n vendorPrefixedValues.push(stringifyNode(node))\n }\n\n // Process properties first that don't have colors,\n // so we can avoid further walking them;\n if (isProperty('z-index', property)) {\n if (!isValueKeyword(node)) {\n zindex.push(stringifyNode(node))\n }\n return this.skip\n } else if (isProperty('font', property)) {\n if (!isFontFamilyKeyword(node)) {\n fontFamilies.push(getFamilyFromFont(node, stringifyNode))\n }\n if (!isFontSizeKeyword(node)) {\n const size = getSizeFromFont(node)\n if (size) {\n fontSizes.push(size)\n }\n }\n break\n } else if (isProperty('font-size', property)) {\n if (!isFontSizeKeyword(node)) {\n fontSizes.push(stringifyNode(node))\n }\n break\n } else if (isProperty('font-family', property)) {\n if (!isFontFamilyKeyword(node)) {\n fontFamilies.push(stringifyNode(node))\n }\n break\n } else if (isProperty('transition', property) || isProperty('animation', property)) {\n const [times, fns] = analyzeAnimation(node.children, stringifyNode)\n for (let i = 0; i < times.length; i++) {\n durations.push(times[i])\n }\n for (let i = 0; i < fns.length; i++) {\n timingFunctions.push(fns[i])\n }\n break\n } else if (isProperty('animation-duration', property) || isProperty('transition-duration', property)) {\n durations.push(stringifyNode(node))\n break\n } else if (isProperty('transition-timing-function', property) || isProperty('animation-timing-function', property)) {\n timingFunctions.push(stringifyNode(node))\n break\n } else if (isProperty('text-shadow', property)) {\n if (!isValueKeyword(node)) {\n textShadows.push(stringifyNode(node))\n }\n // no break here: potentially contains colors\n } else if (isProperty('box-shadow', property)) {\n if (!isValueKeyword(node)) {\n boxShadows.push(stringifyNode(node))\n }\n // no break here: potentially contains colors\n }\n\n walk(node, function (valueNode) {\n switch (valueNode.type) {\n case 'Hash': {\n colors.push('#' + valueNode.value, property)\n\n return this.skip\n }\n case 'Identifier': {\n const { name } = valueNode\n // Bail out if it can't be a color name\n // 20 === 'lightgoldenrodyellow'.length\n // 3 === 'red'.length\n if (name.length > 20 || name.length < 3) {\n return this.skip\n }\n if (colorNames[name.toLowerCase()]) {\n colors.push(stringifyNode(valueNode), property)\n }\n return this.skip\n }\n case 'Function': {\n // Don't walk var() multiple times\n if (strEquals('var', valueNode.name)) {\n return this.skip\n }\n if (colorFunctions[valueNode.name.toLowerCase()]) {\n colors.push(stringifyNode(valueNode), property)\n }\n // No this.skip here intentionally,\n // otherwise we'll miss colors in linear-gradient() etc.\n }\n }\n })\n break\n }\n case 'Declaration': {\n totalDeclarations++\n\n const declaration = stringifyNode(node)\n uniqueDeclarations.push(declaration)\n\n if (node.important) {\n importantDeclarations++\n\n if (this.atrule && endsWith('keyframes', this.atrule.name)) {\n importantsInKeyframes++\n }\n }\n\n const { property } = node\n\n properties.push(property)\n\n if (hasVendorPrefix(property)) {\n propertyVendorPrefixes.push(property)\n } else if (isHack(property)) {\n propertyHacks.push(property)\n } else if (isCustom(property)) {\n customProperties.push(property)\n }\n break\n }\n }\n })\n\n const embeddedContent = embeds.count()\n const embedSize = Object.keys(embeddedContent.unique).join('').length\n\n const totalUniqueDeclarations = uniqueDeclarations.count()\n\n const totalSelectors = selectorComplexities.size()\n const specificitiesA = specificityA.aggregate()\n const specificitiesB = specificityB.aggregate()\n const specificitiesC = specificityC.aggregate()\n const uniqueSpecificitiesCount = uniqueSpecificities.count()\n const complexityCount = new CountableCollection(selectorComplexities.toArray()).count()\n const totalUniqueSelectors = uniqueSelectors.count()\n const uniqueRuleSize = new CountableCollection(ruleSizes.toArray()).count()\n const uniqueSelectorsPerRule = new CountableCollection(selectorsPerRule.toArray()).count()\n const uniqueDeclarationsPerRule = new CountableCollection(declarationsPerRule.toArray()).count()\n const assign = Object.assign\n\n return {\n stylesheet: {\n sourceLinesOfCode: totalAtRules + totalSelectors + totalDeclarations + keyframeSelectors.size(),\n linesOfCode: lines.length,\n size: css.length,\n comments: {\n total: totalComments,\n size: commentsSize,\n },\n embeddedContent: assign(embeddedContent, {\n size: {\n total: embedSize,\n ratio: css.length === 0 ? 0 : embedSize / css.length,\n },\n }),\n },\n atrules: {\n fontface: {\n total: fontfaces.length,\n totalUnique: fontfaces.length,\n unique: fontfaces,\n uniquenessRatio: fontfaces.length === 0 ? 0 : 1\n },\n import: imports.count(),\n media: medias.count(),\n charset: charsets.count(),\n supports: supports.count(),\n keyframes: assign(\n keyframes.count(), {\n prefixed: assign(\n prefixedKeyframes.count(), {\n ratio: keyframes.size() === 0 ? 0 : prefixedKeyframes.size() / keyframes.size()\n }),\n }),\n container: containers.count(),\n layer: layers.count(),\n },\n rules: {\n total: totalRules,\n empty: {\n total: emptyRules,\n ratio: totalRules === 0 ? 0 : emptyRules / totalRules\n },\n sizes: assign(\n ruleSizes.aggregate(),\n {\n items: ruleSizes.toArray(),\n unique: uniqueRuleSize.unique,\n totalUnique: uniqueRuleSize.totalUnique,\n uniquenessRatio: uniqueRuleSize.uniquenessRatio\n },\n ),\n selectors: assign(\n selectorsPerRule.aggregate(),\n {\n items: selectorsPerRule.toArray(),\n unique: uniqueSelectorsPerRule.unique,\n totalUnique: uniqueSelectorsPerRule.totalUnique,\n uniquenessRatio: uniqueSelectorsPerRule.uniquenessRatio\n },\n ),\n declarations: assign(\n declarationsPerRule.aggregate(),\n {\n items: declarationsPerRule.toArray(),\n unique: uniqueDeclarationsPerRule.unique,\n totalUnique: uniqueDeclarationsPerRule.totalUnique,\n uniquenessRatio: uniqueDeclarationsPerRule.uniquenessRatio,\n },\n ),\n },\n selectors: {\n total: totalSelectors,\n totalUnique: totalUniqueSelectors,\n uniquenessRatio: totalSelectors === 0 ? 0 : totalUniqueSelectors / totalSelectors,\n specificity: {\n min: minSpecificity === undefined ? [0, 0, 0] : minSpecificity,\n max: maxSpecificity === undefined ? [0, 0, 0] : maxSpecificity,\n sum: [specificitiesA.sum, specificitiesB.sum, specificitiesC.sum],\n mean: [specificitiesA.mean, specificitiesB.mean, specificitiesC.mean],\n mode: [specificitiesA.mode, specificitiesB.mode, specificitiesC.mode],\n median: [specificitiesA.median, specificitiesB.median, specificitiesC.median],\n items: specificities,\n unique: uniqueSpecificitiesCount.unique,\n totalUnique: uniqueSpecificitiesCount.totalUnique,\n uniquenessRatio: uniqueSpecificitiesCount.uniquenessRatio,\n },\n complexity: assign(\n selectorComplexities.aggregate(),\n complexityCount, {\n items: selectorComplexities.toArray(),\n }),\n id: assign(\n ids.count(), {\n ratio: totalSelectors === 0 ? 0 : ids.size() / totalSelectors,\n }),\n accessibility: assign(\n a11y.count(), {\n ratio: totalSelectors === 0 ? 0 : a11y.size() / totalSelectors,\n }),\n keyframes: keyframeSelectors.count(),\n },\n declarations: {\n total: totalDeclarations,\n unique: {\n total: totalUniqueDeclarations,\n ratio: totalDeclarations === 0 ? 0 : totalUniqueDeclarations / totalDeclarations,\n },\n importants: {\n total: importantDeclarations,\n ratio: totalDeclarations === 0 ? 0 : importantDeclarations / totalDeclarations,\n inKeyframes: {\n total: importantsInKeyframes,\n ratio: importantDeclarations === 0 ? 0 : importantsInKeyframes / importantDeclarations,\n },\n },\n },\n properties: assign(\n properties.count(), {\n prefixed: assign(\n propertyVendorPrefixes.count(), {\n ratio: properties.size() === 0 ? 0 : propertyVendorPrefixes.size() / properties.size(),\n }),\n custom: assign(\n customProperties.count(), {\n ratio: properties.size() === 0 ? 0 : customProperties.size() / properties.size(),\n }),\n browserhacks: assign(\n propertyHacks.count(), {\n ratio: properties.size() === 0 ? 0 : propertyHacks.size() / properties.size(),\n }),\n }),\n values: {\n colors: colors.count(),\n fontFamilies: fontFamilies.count(),\n fontSizes: fontSizes.count(),\n zindexes: zindex.count(),\n textShadows: textShadows.count(),\n boxShadows: boxShadows.count(),\n animations: {\n durations: durations.count(),\n timingFunctions: timingFunctions.count(),\n },\n prefixes: vendorPrefixedValues.count(),\n units: units.count(),\n },\n __meta__: {\n parseTime: startAnalysis - startParse,\n analyzeTime: new Date() - startAnalysis,\n total: new Date() - start\n }\n }\n}\n\nexport {\n analyze,\n compareSpecificity,\n}\n"],"names":["compareChar","referenceCode","testCode","endsWith","base","test","offset","length","i","charCodeAt","startsWith","compareSpecificity","a","b","analyzeSpecificity","node","A","B","C","complexity","isA11y","walk","selector","type","Boolean","value","name","skip","selectorList","childSelectors","visit","enter","push","sort","specificity","listItem","isId","colorNames","aliceblue","antiquewhite","aqua","aquamarine","azure","beige","bisque","black","blanchedalmond","blue","blueviolet","brown","burlywood","cadetblue","chartreuse","chocolate","coral","cornflowerblue","cornsilk","crimson","cyan","darkblue","darkcyan","darkgoldenrod","darkgray","darkgreen","darkgrey","darkkhaki","darkmagenta","darkolivegreen","darkorange","darkorchid","darkred","darksalmon","darkseagreen","darkslateblue","darkslategray","darkslategrey","darkturquoise","darkviolet","deeppink","deepskyblue","dimgray","dimgrey","dodgerblue","firebrick","floralwhite","forestgreen","fuchsia","gainsboro","ghostwhite","gold","goldenrod","gray","green","greenyellow","grey","honeydew","hotpink","indianred","indigo","ivory","khaki","lavender","lavenderblush","lawngreen","lemonchiffon","lightblue","lightcoral","lightcyan","lightgoldenrodyellow","lightgray","lightgreen","lightgrey","lightpink","lightsalmon","lightseagreen","lightskyblue","lightslategray","lightslategrey","lightsteelblue","lightyellow","lime","limegreen","linen","magenta","maroon","mediumaquamarine","mediumblue","mediumorchid","mediumpurple","mediumseagreen","mediumslateblue","mediumspringgreen","mediumturquoise","mediumvioletred","midnightblue","mintcream","mistyrose","moccasin","navajowhite","navy","oldlace","olive","olivedrab","orange","orangered","orchid","palegoldenrod","palegreen","paleturquoise","palevioletred","papayawhip","peachpuff","peru","pink","plum","powderblue","purple","rebeccapurple","red","rosybrown","royalblue","saddlebrown","salmon","sandybrown","seagreen","seashell","sienna","silver","skyblue","slateblue","slategray","slategrey","snow","springgreen","steelblue","tan","teal","thistle","tomato","turquoise","violet","wheat","white","whitesmoke","yellow","yellowgreen","canvas","canvastext","linktext","visitedtext","activetext","buttonface","buttontext","buttonborder","field","fieldtext","highlight","highlighttext","selecteditem","selecteditemtext","mark","marktext","graytext","colorFunctions","rgb","rgba","hsl","hsla","hwb","lab","lch","oklab","oklch","color","systemKeywords","inherit","initial","unset","revert","caption","icon","menu","keywordDisallowList","normal","small","medium","large","larger","smaller","bold","bolder","lighter","condensed","expanded","italic","oblique","isFontFamilyKeyword","firstChild","children","first","sizeKeywords","keywords","isFontSizeKeyword","auto","none","isValueKeyword","size","timingKeywords","linear","ease","hasVendorPrefix","keyword","indexOf","isAstVendorPrefixed","toArray","index","CountableCollection","this","_items","_total","_totalUnique","item","count","total","totalUnique","unique","uniquenessRatio","ContextCollection","_list","_contexts","context","itemsPerContext","Object","assign","AggregateCollection","add","aggregate","min","max","mean","mode","median","range","sum","arr","middle","lowerMiddleRank","sorted","slice","reduce","num","frequencies","create","maxOccurrences","maxOccurenceCount","element","updatedCount","Mode","Math","floor","isCustom","property","isProperty","basename","OccurrenceCounter","keys","analyze","css","start","Date","lines","split","stringifyNode","loc","end","line","substring","column","maxSpecificity","minSpecificity","totalComments","commentsSize","embeds","startParse","ast","parse","parseAtrulePrelude","parseCustomProperty","positions","onComment","comment","startAnalysis","totalAtRules","fontfaces","layers","imports","medias","charsets","supports","keyframes","prefixedKeyframes","containers","totalRules","emptyRules","ruleSizes","selectorsPerRule","declarationsPerRule","keyframeSelectors","uniqueSelectors","specificityA","specificityB","specificityC","uniqueSpecificities","selectorComplexities","specificities","ids","a11y","uniqueDeclarations","totalDeclarations","importantDeclarations","importantsInKeyframes","properties","propertyHacks","propertyVendorPrefixes","customProperties","vendorPrefixedValues","zindex","textShadows","boxShadows","fontFamilies","fontSizes","timingFunctions","durations","colors","units","atRuleName","descriptors","block","forEach","descriptor","prelude","trim","map","numSelectors","numDeclarations","atrule","undefined","declaration","unit","parts","reverse","fontNode","quote","getFamilyFromFont","operator","getSizeFromFont","durationFound","child","analyzeAnimation","times","fns","valueNode","toLowerCase","strEquals","important","code","isHack","embeddedContent","embedSize","join","totalUniqueDeclarations","totalSelectors","specificitiesA","specificitiesB","specificitiesC","uniqueSpecificitiesCount","complexityCount","totalUniqueSelectors","uniqueRuleSize","uniqueSelectorsPerRule","uniqueDeclarationsPerRule","stylesheet","sourceLinesOfCode","linesOfCode","comments","ratio","atrules","fontface","import","media","charset","prefixed","container","layer","rules","empty","sizes","items","selectors","declarations","id","accessibility","importants","inKeyframes","custom","browserhacks","values","zindexes","animations","prefixes","__meta__","parseTime","analyzeTime"],"mappings":"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"}
|
package/dist/analyzer.umd.js
CHANGED
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
!function(e,t){"object"==typeof exports&&"undefined"!=typeof module?t(exports,require("css-tree/parser"),require("css-tree/walker")):"function"==typeof define&&define.amd?define(["exports","css-tree/parser","css-tree/walker"],t):t((e||self).cssAnalyzer={},e.parse,e.walk)}(this,function(e,t,n){function r(e){return e&&"object"==typeof e&&"default"in e?e:{default:e}}var i=/*#__PURE__*/r(t),a=/*#__PURE__*/r(n);function s(){return s=Object.assign||function(e){for(var t=1;t<arguments.length;t++){var n=arguments[t];for(var r in n)Object.prototype.hasOwnProperty.call(n,r)&&(e[r]=n[r])}return e},s.apply(this,arguments)}function o(e,t){return t>=65&&t<=90&&(t|=32),e===t}function u(e,t){if(e.length!==t.length)return!1;for(var n=0;n<e.length;n++)if(!1===o(e.charCodeAt(n),t.charCodeAt(n)))return!1;return!0}function l(e,t){var n=t.length-e.length;if(n<0)return!1;for(var r=t.length-1;r>=n;r--)if(!1===o(e.charCodeAt(r-n),t.charCodeAt(r)))return!1;return!0}function c(e,t){if(t.length<e.length)return!1;for(var n=0;n<e.length;n++)if(!1===o(e.charCodeAt(n),t.charCodeAt(n)))return!1;return!0}function h(e,t){return e[0]===t[0]?e[1]===t[1]?t[2]-e[2]:t[1]-e[1]:t[0]-e[0]}var d=function(e){var t=0,n=0,r=0,i=0,s=!1;return a.default(e,function(e){switch(e.type){case"IdSelector":t++,i++;break;case"ClassSelector":n++,i++;break;case"AttributeSelector":n++,i++,Boolean(e.value)&&i++,s="role"===e.name.name||c("aria-",e.name.name);break;case"PseudoElementSelector":case"TypeSelector":if(i++,42===e.name.charCodeAt(0)&&1===e.name.length)break;r++;break;case"PseudoClassSelector":switch(e.name){case"before":case"after":case"first-letter":case"first-line":return r++,i++,this.skip;case"where":case"is":case"has":case"matches":case"-webkit-any":case"-moz-any":case"not":case"nth-child":case"nth-last-child":"nth-child"!==e.name&&"nth-last-child"!==e.name||n++;var o=(m=[],a.default(e,{visit:"Selector",enter:function(e){m.push(d(e))}}),m.sort(function(e,t){return h(e.specificity,t.specificity)}));if(0===o.length)return;if("where"!==e.name){var u=o[0].specificity;t+=u[0],n+=u[1],r+=u[2]}for(var l=0;l<o.length;l++){var f=o[l];f.isA11y&&(s=!0),i+=f.complexity}return i++,this.skip;default:return i++,n++,this.skip}case"Combinator":i++}var m}),{specificity:[t,n,r],complexity:i,isId:t>0,isA11y:s}},f={aliceblue:1,antiquewhite:1,aqua:1,aquamarine:1,azure:1,beige:1,bisque:1,black:1,blanchedalmond:1,blue:1,blueviolet:1,brown:1,burlywood:1,cadetblue:1,chartreuse:1,chocolate:1,coral:1,cornflowerblue:1,cornsilk:1,crimson:1,cyan:1,darkblue:1,darkcyan:1,darkgoldenrod:1,darkgray:1,darkgreen:1,darkgrey:1,darkkhaki:1,darkmagenta:1,darkolivegreen:1,darkorange:1,darkorchid:1,darkred:1,darksalmon:1,darkseagreen:1,darkslateblue:1,darkslategray:1,darkslategrey:1,darkturquoise:1,darkviolet:1,deeppink:1,deepskyblue:1,dimgray:1,dimgrey:1,dodgerblue:1,firebrick:1,floralwhite:1,forestgreen:1,fuchsia:1,gainsboro:1,ghostwhite:1,gold:1,goldenrod:1,gray:1,green:1,greenyellow:1,grey:1,honeydew:1,hotpink:1,indianred:1,indigo:1,ivory:1,khaki:1,lavender:1,lavenderblush:1,lawngreen:1,lemonchiffon:1,lightblue:1,lightcoral:1,lightcyan:1,lightgoldenrodyellow:1,lightgray:1,lightgreen:1,lightgrey:1,lightpink:1,lightsalmon:1,lightseagreen:1,lightskyblue:1,lightslategray:1,lightslategrey:1,lightsteelblue:1,lightyellow:1,lime:1,limegreen:1,linen:1,magenta:1,maroon:1,mediumaquamarine:1,mediumblue:1,mediumorchid:1,mediumpurple:1,mediumseagreen:1,mediumslateblue:1,mediumspringgreen:1,mediumturquoise:1,mediumvioletred:1,midnightblue:1,mintcream:1,mistyrose:1,moccasin:1,navajowhite:1,navy:1,oldlace:1,olive:1,olivedrab:1,orange:1,orangered:1,orchid:1,palegoldenrod:1,palegreen:1,paleturquoise:1,palevioletred:1,papayawhip:1,peachpuff:1,peru:1,pink:1,plum:1,powderblue:1,purple:1,rebeccapurple:1,red:1,rosybrown:1,royalblue:1,saddlebrown:1,salmon:1,sandybrown:1,seagreen:1,seashell:1,sienna:1,silver:1,skyblue:1,slateblue:1,slategray:1,slategrey:1,snow:1,springgreen:1,steelblue:1,tan:1,teal:1,thistle:1,tomato:1,turquoise:1,violet:1,wheat:1,white:1,whitesmoke:1,yellow:1,yellowgreen:1,canvas:1,canvastext:1,linktext:1,visitedtext:1,activetext:1,buttonface:1,buttontext:1,buttonborder:1,field:1,fieldtext:1,highlight:1,highlighttext:1,selecteditem:1,selecteditemtext:1,mark:1,marktext:1,graytext:1},m={rgb:1,rgba:1,hsl:1,hsla:1,hwb:1,lab:1,lch:1,oklab:1,oklch:1,color:1},p=/*#__PURE__*/function(){function e(e){if(this.items={},this.total=0,this.totalUnique=0,e)for(var t=0;t<e.length;t++)this.push(e[t])}var t=e.prototype;return t.push=function(e){this.total++,this.items[e]?this.items[e]++:(this.items[e]=1,this.totalUnique++)},t.size=function(){return this.total},t.count=function(){return{total:this.total,totalUnique:this.totalUnique,unique:this.items,uniquenessRatio:0===this.total?0:this.totalUnique/this.total}},e}(),g={inherit:1,initial:1,unset:1,revert:1,caption:1,icon:1,menu:1,"message-box":1,"small-caption":1,"status-bar":1},y={normal:1,"xx-small":1,"x-small":1,small:1,medium:1,large:1,"x-large":1,"xx-large":1,larger:1,smaller:1,bold:1,bolder:1,lighter:1,"ultra-condensed":1,"extra-condensed":1,condensed:1,"semi-condensed":1,"semi-expanded":1,expanded:1,"extra-expanded":1,"ultra-expanded":1,italic:1,oblique:1},v=function(e){for(var t=e.fontValues,n=e.stringifyNode,r=new p(e.fontFamilyValues),i=function(e){var i=t[e],s=i.children.first;if("Identifier"===s.type&&g[s.name])return"continue";var o="";a.default(i,{reverse:!0,enter:function(e){if("String"===e.type){var t=e.loc.start,r=n({loc:{start:{line:t.line,column:t.column},end:{line:t.line,column:t.column+1}}});return o=r+e.value+r+o}return"Operator"===e.type&&44===e.value.charCodeAt(0)?o=e.value+o:"Identifier"===e.type?y[e.name]?this.skip:o=e.name+o:void 0}}),r.push(o)},s=0;s<t.length;s++)i(s);return r.count()},b={"xx-small":1,"x-small":1,small:1,medium:1,large:1,"x-large":1,"xx-large":1,larger:1,smaller:1},k={inherit:1,initial:1,unset:1,revert:1,caption:1,icon:1,menu:1,"message-box":1,"small-caption":1,"status-bar":1},w=function(e){for(var t=e.stringifyNode,n=e.fontValues,r=new p(e.fontSizeValues),i=function(e){var i=n[e],s=i.children.first;if("Identifier"===s.type&&k[s.name])return"continue";var o=!1,u=void 0;a.default(i,function(e){switch(e.type){case"Number":if(48===e.value.charCodeAt(0))return u="0",this.break;case"Operator":47===e.value.charCodeAt(0)&&(o=!0);break;case"Dimension":if(!o)return u=t(e),this.break;case"Identifier":if(b[e.name])return u=e.name,this.break}}),u&&r.push(u)},s=0;s<n.length;s++)i(s);return r.count()},x={auto:1,inherit:1,initial:1,unset:1,revert:1,none:1},z=function(e){for(var t=e.values,n=e.stringifyNode,r=new p,i=0;i<t.length;i++){var a=t[i],s=a.children.first;s&&("Identifier"===s.type&&x[s.name]||r.push(n(a)))}return r.count()},q={linear:1,ease:1,"ease-in":1,"ease-out":1,"ease-in-out":1,"step-start":1,"step-end":1},A=function(e){for(var t=e.animations,n=e.timingFunctions,r=e.stringifyNode,i=new p(e.durations),a=new p(n),s=function(e){var n=!1;t[e].forEach(function(e){return"Operator"===e.type?n=!1:"Dimension"===e.type&&!1===n?(n=!0,i.push(r(e))):"Identifier"===e.type&&q[e.name]?a.push(r(e)):"Function"!==e.type||"cubic-bezier"!==e.name&&"steps"!==e.name?void 0:a.push(r(e))})},o=0;o<t.length;o++)s(o);return{durations:i.count(),timingFunctions:a.count()}};function C(e){return 45===e.charCodeAt(0)&&45!==e.charCodeAt(1)&&-1!==e.indexOf("-",2)}function O(e){e=e.toArray();for(var t=0;t<e.length;t++){var n=e[t];if("Identifier"===n.type&&n.name.length>=3&&C(n.name))return!0;if("Function"===n.type){if(C(n.name))return!0;if(n.children&&O(n.children))return!0}}return!1}var S=function(e){for(var t=e.values,n=e.stringifyNode,r=new p,i=0;i<t.length;i++){var a=t[i];a.children&&O(a.children)&&r.push(n(a))}return r.count()},N=function(e){for(var t=e.atrules,n=e.stringifyNode,r=[],i=new p,a=new p,o=new p,u=new p,c=new p,h=new p,d=new p,f=new p,m={"font-face":function(e){var t={};e.block.children.forEach(function(e){return t[e.property]=n(e.value)}),r.push(t)},media:function(e){return o.push(e.prelude)},supports:function(e){return c.push(e.prelude)},keyframes:function(e){return h.push("@"+e.name+" "+e.prelude)},import:function(e){return a.push(e.prelude)},charset:function(e){return u.push(e.prelude)},container:function(e){return f.push(e.prelude)},layer:function(e){e.prelude.trim().split(",").map(function(e){return e.trim()}).forEach(function(e){return i.push(e)})}},g=0;g<t.length;g++){var y=t[g],v=y.name,b=m[v];if(b)b(y);else if(l("keyframes",v)){var k="@"+v+" "+y.prelude;h.push(k),C(v)&&d.push(k)}}return{fontface:{total:r.length,totalUnique:r.length,unique:r,uniquenessRatio:0===r.length?0:1},import:a.count(),media:o.count(),charset:u.count(),supports:c.count(),keyframes:s({},h.count(),{prefixed:s({},d.count(),{ratio:0===h.size()?0:d.size()/h.size()})}),container:f.count(),layer:i.count()}},j=/*#__PURE__*/function(){function e(){this.list=new p,this.contexts={}}var t=e.prototype;return t.push=function(e,t){this.list.push(e),this.contexts[t]||(this.contexts[t]=new p),this.contexts[t].push(e)},t.count=function(){var e={};for(var t in this.contexts)e[t]=this.contexts[t].count();return Object.assign(this.list.count(),{itemsPerContext:e})},e}(),I=/*#__PURE__*/function(){function e(){this.items=[]}var t=e.prototype;return t.add=function(e){this.items.push(e)},t.aggregate=function(){if(0===this.items.length)return{min:0,max:0,mean:0,mode:0,median:0,range:0,sum:0};var e,t,n,r=this.items.slice().sort(function(e,t){return e-t}),i=r[0],a=r[r.length-1],s=this.items.reduce(function(e,t){return e+t}),o=function(e){for(var t=Object.create(null),n=-1,r=0,i=0,a=0;a<e.length;a++){var s=e[a],o=(t[s]||0)+1;t[s]=o,o>n&&(n=o,r=0,i=0),o>=n&&(r++,i+=s)}return i/r}(this.items),u=(t=(e=r).length/2)!==(n=Math.floor(t))?e[n]:(e[n]+e[n-1])/2;return{min:i,max:a,mean:s/this.items.length,mode:o,median:u,range:a-i,sum:s}},t.toArray=function(){return this.items},e}();function D(e){return!(e.length<3)&&45===e.charCodeAt(0)&&45===e.charCodeAt(1)}function F(e,t){return!D(t)&&l(e,t)}e.analyze=function(e){var t=new Date,n=e.split(/\r?\n/);function r(e){var t=e.loc.start,r=e.loc.end;if(0==r.line-t.line)return n[t.line-1].substring(t.column-1,r.column-1);for(var i="",a=t.line;a<=r.line;a++){var s=n[a-1];i+=a!==t.line?a!==r.line?s+"\n":s.substring(0,r.column-1):s.substring(t.column-1)+"\n"}return i}var o,g,y=new Date,b=0,k=0,x=i.default(e,{parseAtrulePrelude:!1,parseCustomProperty:!0,positions:!0,onComment:function(e){b++,k+=e.length}}),q=new Date,O=new p,U=[],V=0,P=0,T=new I,E=new I,R=new p,_=Object.create(null),L=0,B=0,H=0,K=new p,M=new p,G=new p,J=new p,Q=[],W=[],X=[],Y=[],Z=[],$=[],ee=[],te=[],ne=[],re=[],ie=new j,ae=new j,se=Object.create(null),oe=new I,ue=new I,le=new I,ce=new I,he=[],de=[],fe=new p,me=new p;a.default(x,function(e){switch(e.type){case"Atrule":U.push({name:e.name,prelude:e.prelude&&e.prelude.value,block:u("font-face",e.name)&&e.block});break;case"Rule":var t=function(e){var t=0,n=0;return a.default(e,function(e){return"Selector"===e.type?(t++,this.skip):"Declaration"===e.type?(n++,this.skip):void 0}),[t,n]}(e),n=t[1];V++,0===n&&P++,T.add(t[0]),E.add(n);break;case"Selector":var i=r(e);if(this.atrule&&l("keyframes",this.atrule.name))return R.push(i),this.skip;var s=d(e),p=s.specificity,y=s.complexity,v=s.isA11y;return s.isId&&fe.push(i),v&&me.push(i),se[i]?se[i]++:se[i]=1,ce.add(y),void 0===o&&(o=p),void 0===g&&(g=p),oe.add(p[0]),ue.add(p[1]),le.add(p[2]),void 0!==g&&h(g,p)<0&&(g=p),void 0!==o&&h(o,p)>0&&(o=p),he.push(p),de.push(y),this.skip;case"Dimension":if(!this.declaration)break;return ae.push(e.unit,this.declaration.property),this.skip;case"Url":c("data:",e.value)&&O.push(e.value);break;case"Declaration":L++;var b=r(e);_[b]?_[b]++:_[b]=1,e.important&&(B++,this.atrule&&l("keyframes",this.atrule.name)&&H++);var k=e.value,w=e.property;if(K.push(w),Q.push(k),C(w)?G.push(w):function(e){if(D(e)||C(e))return!1;var t=e.charCodeAt(0);return 47===t||95===t||43===t||42===t||38===t||36===t||35===t}(w)?M.push(w):D(w)&&J.push(w),F("z-index",w))return W.push(k),this.skip;if(F("font",w)){Z.push(k);break}if(F("font-size",w)){ee.push(r(k));break}if(F("font-family",w)){$.push(r(k));break}if(F("transition",w)||F("animation",w)){te.push(k.children);break}if(F("animation-duration",w)||F("transition-duration",w)){re.push(r(k));break}if(F("transition-timing-function",w)||F("animation-timing-function",w)){ne.push(r(k));break}F("text-shadow",w)?X.push(k):F("box-shadow",w)&&Y.push(k),a.default(k,function(e){switch(e.type){case"Hash":return ie.push("#"+e.value,w),this.skip;case"Identifier":var t=e.name;return t.length>20||t.length<3||f[t.toLowerCase()]&&ie.push(r(e),w),this.skip;case"Function":m[e.name.toLowerCase()]&&ie.push(r(e),w)}})}});var pe=O.count(),ge=Object.keys(pe.unique).join("").length,ye=Object.keys(_).length,ve=de.length,be=oe.aggregate(),ke=ue.aggregate(),we=le.aggregate(),xe=new p(de).count(),ze=Object.values(se).length;return{stylesheet:{sourceLinesOfCode:U.length+ve+L+R.size(),linesOfCode:n.length,size:e.length,comments:{total:b,size:k},embeddedContent:Object.assign(pe,{size:{total:ge,ratio:0===e.length?0:ge/e.length}})},atrules:N({atrules:U,stringifyNode:r}),rules:{total:V,empty:{total:P,ratio:0===V?0:P/V},selectors:s({},T.aggregate(),{items:T.toArray()}),declarations:s({},E.aggregate(),{items:E.toArray()})},selectors:{total:ve,totalUnique:ze,uniquenessRatio:0===ve?0:ze/ve,specificity:{min:void 0===g?[0,0,0]:g,max:void 0===o?[0,0,0]:o,sum:[be.sum,ke.sum,we.sum],mean:[be.mean,ke.mean,we.mean],mode:[be.mode,ke.mode,we.mode],median:[be.median,ke.median,we.median],items:he},complexity:s({},ce.aggregate(),xe,{items:de}),id:s({},fe.count(),{ratio:0===ve?0:fe.size()/ve}),accessibility:s({},me.count(),{ratio:0===ve?0:me.size()/ve}),keyframes:R.count()},declarations:{total:L,unique:{total:ye,ratio:0===L?0:ye/L},importants:{total:B,ratio:0===L?0:B/L,inKeyframes:{total:H,ratio:0===B?0:H/B}}},properties:s({},K.count(),{prefixed:s({},G.count(),{ratio:0===K.size()?0:G.size()/K.size()}),custom:s({},J.count(),{ratio:0===K.size()?0:J.size()/K.size()}),browserhacks:s({},M.count(),{ratio:0===K.size()?0:M.size()/K.size()})}),values:{colors:ie.count(),fontFamilies:v({stringifyNode:r,fontValues:Z,fontFamilyValues:$}),fontSizes:w({stringifyNode:r,fontValues:Z,fontSizeValues:ee}),zindexes:z({values:W,stringifyNode:r}),textShadows:z({values:X,stringifyNode:r}),boxShadows:z({values:Y,stringifyNode:r}),animations:A({animations:te,timingFunctions:ne,durations:re,stringifyNode:r}),prefixes:S({values:Q,stringifyNode:r}),units:ae.count()},__meta__:{parseTime:q-y,analyzeTime:new Date-q,total:new Date-t}}},e.compareSpecificity=h});
|
|
1
|
+
!function(e,t){"object"==typeof exports&&"undefined"!=typeof module?t(exports,require("css-tree/parser"),require("css-tree/walker")):"function"==typeof define&&define.amd?define(["exports","css-tree/parser","css-tree/walker"],t):t((e||self).cssAnalyzer={},e.parse,e.walk)}(this,function(e,t,r){function n(e){return e&&"object"==typeof e&&"default"in e?e:{default:e}}var i=/*#__PURE__*/n(t),a=/*#__PURE__*/n(r);function o(e,t){return t>=65&&t<=90&&(t|=32),e===t}function s(e,t){var r=t.length-e.length;if(r<0)return!1;for(var n=t.length-1;n>=r;n--)if(!1===o(e.charCodeAt(n-r),t.charCodeAt(n)))return!1;return!0}function u(e,t){if(t.length<e.length)return!1;for(var r=0;r<e.length;r++)if(!1===o(e.charCodeAt(r),t.charCodeAt(r)))return!1;return!0}function l(e,t){return e[0]===t[0]?e[1]===t[1]?t[2]-e[2]:t[1]-e[1]:t[0]-e[0]}var c=function(e){var t=0,r=0,n=0,i=0,o=!1;return a.default(e,function(e){switch(e.type){case"IdSelector":t++,i++;break;case"ClassSelector":r++,i++;break;case"AttributeSelector":r++,i++,Boolean(e.value)&&i++,o="role"===e.name.name||u("aria-",e.name.name);break;case"PseudoElementSelector":case"TypeSelector":if(i++,42===e.name.charCodeAt(0)&&1===e.name.length)break;n++;break;case"PseudoClassSelector":switch(e.name){case"before":case"after":case"first-letter":case"first-line":return n++,i++,this.skip;case"where":case"is":case"has":case"matches":case"-webkit-any":case"-moz-any":case"not":case"nth-child":case"nth-last-child":"nth-child"!==e.name&&"nth-last-child"!==e.name||r++;var s=(m=[],a.default(e,{visit:"Selector",enter:function(e){m.push(c(e))}}),m.sort(function(e,t){return l(e.specificity,t.specificity)}));if(0===s.length)return;if("where"!==e.name){var f=s[0].specificity;t+=f[0],r+=f[1],n+=f[2]}for(var d=0;d<s.length;d++){var h=s[d];h.isA11y&&(o=!0),i+=h.complexity}return i++,this.skip;default:return i++,r++,this.skip}case"Combinator":i++}var m}),{specificity:[t,r,n],complexity:i,isId:t>0,isA11y:o}},f={aliceblue:1,antiquewhite:1,aqua:1,aquamarine:1,azure:1,beige:1,bisque:1,black:1,blanchedalmond:1,blue:1,blueviolet:1,brown:1,burlywood:1,cadetblue:1,chartreuse:1,chocolate:1,coral:1,cornflowerblue:1,cornsilk:1,crimson:1,cyan:1,darkblue:1,darkcyan:1,darkgoldenrod:1,darkgray:1,darkgreen:1,darkgrey:1,darkkhaki:1,darkmagenta:1,darkolivegreen:1,darkorange:1,darkorchid:1,darkred:1,darksalmon:1,darkseagreen:1,darkslateblue:1,darkslategray:1,darkslategrey:1,darkturquoise:1,darkviolet:1,deeppink:1,deepskyblue:1,dimgray:1,dimgrey:1,dodgerblue:1,firebrick:1,floralwhite:1,forestgreen:1,fuchsia:1,gainsboro:1,ghostwhite:1,gold:1,goldenrod:1,gray:1,green:1,greenyellow:1,grey:1,honeydew:1,hotpink:1,indianred:1,indigo:1,ivory:1,khaki:1,lavender:1,lavenderblush:1,lawngreen:1,lemonchiffon:1,lightblue:1,lightcoral:1,lightcyan:1,lightgoldenrodyellow:1,lightgray:1,lightgreen:1,lightgrey:1,lightpink:1,lightsalmon:1,lightseagreen:1,lightskyblue:1,lightslategray:1,lightslategrey:1,lightsteelblue:1,lightyellow:1,lime:1,limegreen:1,linen:1,magenta:1,maroon:1,mediumaquamarine:1,mediumblue:1,mediumorchid:1,mediumpurple:1,mediumseagreen:1,mediumslateblue:1,mediumspringgreen:1,mediumturquoise:1,mediumvioletred:1,midnightblue:1,mintcream:1,mistyrose:1,moccasin:1,navajowhite:1,navy:1,oldlace:1,olive:1,olivedrab:1,orange:1,orangered:1,orchid:1,palegoldenrod:1,palegreen:1,paleturquoise:1,palevioletred:1,papayawhip:1,peachpuff:1,peru:1,pink:1,plum:1,powderblue:1,purple:1,rebeccapurple:1,red:1,rosybrown:1,royalblue:1,saddlebrown:1,salmon:1,sandybrown:1,seagreen:1,seashell:1,sienna:1,silver:1,skyblue:1,slateblue:1,slategray:1,slategrey:1,snow:1,springgreen:1,steelblue:1,tan:1,teal:1,thistle:1,tomato:1,turquoise:1,violet:1,wheat:1,white:1,whitesmoke:1,yellow:1,yellowgreen:1,canvas:1,canvastext:1,linktext:1,visitedtext:1,activetext:1,buttonface:1,buttontext:1,buttonborder:1,field:1,fieldtext:1,highlight:1,highlighttext:1,selecteditem:1,selecteditemtext:1,mark:1,marktext:1,graytext:1},d={rgb:1,rgba:1,hsl:1,hsla:1,hwb:1,lab:1,lch:1,oklab:1,oklch:1,color:1},h={inherit:1,initial:1,unset:1,revert:1,caption:1,icon:1,menu:1,"message-box":1,"small-caption":1,"status-bar":1},m={normal:1,"xx-small":1,"x-small":1,small:1,medium:1,large:1,"x-large":1,"xx-large":1,larger:1,smaller:1,bold:1,bolder:1,lighter:1,"ultra-condensed":1,"extra-condensed":1,condensed:1,"semi-condensed":1,"semi-expanded":1,expanded:1,"extra-expanded":1,"ultra-expanded":1,italic:1,oblique:1};function b(e){var t=e.children.first;return"Identifier"===t.type&&h[t.name]}var g={"xx-small":1,"x-small":1,small:1,medium:1,large:1,"x-large":1,"xx-large":1,larger:1,smaller:1},v={inherit:1,initial:1,unset:1,revert:1,caption:1,icon:1,menu:1,"message-box":1,"small-caption":1,"status-bar":1};function w(e){var t=e.children.first;return"Identifier"===t.type&&v[t.name]}var k={auto:1,inherit:1,initial:1,unset:1,revert:1,none:1};function p(e){if(!e.children)return!1;var t=e.children.first;return!!t&&!(e.children.size>1)&&"Identifier"===t.type&&k[t.name]}var y={linear:1,ease:1,"ease-in":1,"ease-out":1,"ease-in-out":1,"step-start":1,"step-end":1};function x(e){return 45===e.charCodeAt(0)&&45!==e.charCodeAt(1)&&-1!==e.indexOf("-",2)}function q(e){if(!e.children)return!1;for(var t=e.children.toArray(),r=0;r<t.length;r++){var n=t[r],i=n.type,a=n.name;if("Identifier"===i&&x(a))return!0;if("Function"===i){if(x(a))return!0;if(q(n))return!0}}return!1}var z=/*#__PURE__*/function(){function e(e){if(this.t={},this.i=0,this.o=0,e)for(var t=0;t<e.length;t++)this.push(e[t])}var t=e.prototype;return t.push=function(e){this.i++,this.t[e]?this.t[e]++:(this.t[e]=1,this.o++)},t.size=function(){return this.i},t.count=function(){return{total:this.i,totalUnique:this.o,unique:this.t,uniquenessRatio:0===this.i?0:this.o/this.i}},e}(),S=/*#__PURE__*/function(){function e(){this.u=new z,this.l={}}var t=e.prototype;return t.push=function(e,t){this.u.push(e),this.l[t]||(this.l[t]=new z),this.l[t].push(e)},t.count=function(){var e={};for(var t in this.l)e[t]=this.l[t].count();return Object.assign(this.u.count(),{itemsPerContext:e})},e}(),O=/*#__PURE__*/function(){function e(){this.t=[]}var t=e.prototype;return t.add=function(e){this.t.push(e)},t.size=function(){return this.t.length},t.aggregate=function(){if(0===this.t.length)return{min:0,max:0,mean:0,mode:0,median:0,range:0,sum:0};var e,t,r,n=this.t.slice().sort(function(e,t){return e-t}),i=n[0],a=n[n.length-1],o=this.t.reduce(function(e,t){return e+t}),s=function(e){for(var t=Object.create(null),r=-1,n=0,i=0,a=0;a<e.length;a++){var o=e[a],s=(t[o]||0)+1;t[o]=s,s>r&&(r=s,n=0,i=0),s>=r&&(n++,i+=o)}return i/n}(this.t),u=(t=(e=n).length/2)!==(r=Math.floor(t))?e[r]:(e[r]+e[r-1])/2;return{min:i,max:a,mean:o/this.t.length,mode:s,median:u,range:a-i,sum:o}},t.toArray=function(){return this.t},e}();function I(e){return!(e.length<3)&&45===e.charCodeAt(0)&&45===e.charCodeAt(1)}function j(e,t){return!I(t)&&s(e,t)}var C=/*#__PURE__*/function(){function e(){this.t=Object.create(null)}var t=e.prototype;return t.push=function(e){return this.t[e]?this.t[e]++:this.t[e]=1},t.count=function(){return Object.keys(this.t).length},e}();e.analyze=function(e){var t=new Date,r=e.split(/\r?\n/);function n(e){var t=e.loc.start,n=e.loc.end;if(0==n.line-t.line)return r[t.line-1].substring(t.column-1,n.column-1);for(var i="",a=t.line;a<=n.line;a++){var o=r[a-1];i+=a!==t.line?a!==n.line?o+"\n":o.substring(0,n.column-1):o.substring(t.column-1)+"\n"}return i}var h,v,k=0,D=0,R=new z,U=new Date,F=i.default(e,{parseAtrulePrelude:!1,parseCustomProperty:!0,positions:!0,onComment:function(e){k++,D+=e.length}}),P=new Date,T=0,A=[],_=new z,B=new z,E=new z,H=new z,K=new z,L=new z,M=new z,N=new z,V=0,G=0,J=new O,Q=new O,W=new O,X=new z,Y=new C,Z=new O,$=new O,ee=new O,te=new z,re=new O,ne=[],ie=new z,ae=new z,oe=new C,se=0,ue=0,le=0,ce=new z,fe=new z,de=new z,he=new z,me=new z,be=new z,ge=new z,ve=new z,we=new z,ke=new z,pe=new z,ye=new z,xe=new S,qe=new S;a.default(F,function(e){switch(e.type){case"Atrule":T++;var t=e.name;if("font-face"===t){var r={};e.block.children.forEach(function(e){return r[e.property]=n(e.value)}),A.push(r);break}if("media"===t){E.push(e.prelude.value);break}if("supports"===t){K.push(e.prelude.value);break}if(s("keyframes",t)){var i="@"+t+" "+e.prelude.value;x(t)&&M.push(i),L.push(i);break}if("import"===t){B.push(e.prelude.value);break}if("charset"===t){H.push(e.prelude.value);break}if("container"===t){N.push(e.prelude.value);break}"layer"===t&&N.push(e.prelude.value.trim().split(",").map(function(e){return e.trim()}).forEach(function(e){return _.push(e)}));break;case"Rule":var k=e.prelude.children?e.prelude.children.size:0,z=e.block.children?e.block.children.size:0;J.add(k+z),Q.add(k),W.add(z),V++,0===z&&G++;break;case"Selector":var S=n(e);if(this.atrule&&s("keyframes",this.atrule.name))return X.push(S),this.skip;var O=c(e),C=O.specificity,D=O.complexity,U=O.isA11y;return O.isId&&ie.push(S),U&&ae.push(S),Y.push(S),re.add(D),te.push(C),void 0===h&&(h=C),void 0===v&&(v=C),Z.add(C[0]),$.add(C[1]),ee.add(C[2]),void 0!==v&&l(v,C)<0&&(v=C),void 0!==h&&l(h,C)>0&&(h=C),ne.push(C),this.skip;case"Dimension":if(!this.declaration)break;return qe.push(e.unit,this.declaration.property),this.skip;case"Url":u("data:",e.value)&&R.push(e.value);break;case"Value":if(p(e))break;var F=this.declaration.property;if(q(e)&&me.push(n(e)),j("z-index",F))return p(e)||be.push(n(e)),this.skip;if(j("font",F)){if(b(e)||we.push(function(e,t){var r="";return a.default(e,{reverse:!0,enter:function(e){if("String"===e.type){var n=e.loc.start,i=t({loc:{start:{line:n.line,column:n.column},end:{line:n.line,column:n.column+1}}});return r=i+e.value+i+r}return"Operator"===e.type&&44===e.value.charCodeAt(0)?r=e.value+r:"Identifier"===e.type?m[e.name]?this.skip:r=e.name+r:void 0}}),r}(e,n)),!w(e)){var P=function(e){var t,r=!1;return a.default(e,function(e){switch(e.type){case"Number":if(48===e.value.charCodeAt(0))return t="0",this.break;case"Operator":47===e.value.charCodeAt(0)&&(r=!0);break;case"Dimension":if(!r)return t=e.value+e.unit,this.break;case"Identifier":if(g[e.name])return t=e.name,this.break}}),t}(e);P&&ke.push(P)}break}if(j("font-size",F)){w(e)||ke.push(n(e));break}if(j("font-family",F)){b(e)||we.push(n(e));break}if(j("transition",F)||j("animation",F)){for(var ze=function(e,t){var r=!1,n=[],i=[];return e.forEach(function(e){return"Operator"===e.type?r=!1:"Dimension"===e.type&&!1===r?(r=!0,n.push(t(e))):"Identifier"===e.type&&y[e.name]?i.push(t(e)):"Function"!==e.type||"cubic-bezier"!==e.name&&"steps"!==e.name?void 0:i.push(t(e))}),[n,i]}(e.children,n),Se=ze[0],Oe=ze[1],Ie=0;Ie<Se.length;Ie++)ye.push(Se[Ie]);for(var je=0;je<Oe.length;je++)pe.push(Oe[je]);break}if(j("animation-duration",F)||j("transition-duration",F)){ye.push(n(e));break}if(j("transition-timing-function",F)||j("animation-timing-function",F)){pe.push(n(e));break}j("text-shadow",F)?p(e)||ge.push(n(e)):j("box-shadow",F)&&(p(e)||ve.push(n(e))),a.default(e,function(e){switch(e.type){case"Hash":return xe.push("#"+e.value,F),this.skip;case"Identifier":var t=e.name;return t.length>20||t.length<3||f[t.toLowerCase()]&&xe.push(n(e),F),this.skip;case"Function":if(function(e,t){if(e.length!==t.length)return!1;for(var r=0;r<e.length;r++)if(!1===o(e.charCodeAt(r),t.charCodeAt(r)))return!1;return!0}("var",e.name))return this.skip;d[e.name.toLowerCase()]&&xe.push(n(e),F)}});break;case"Declaration":se++;var Ce=n(e);oe.push(Ce),e.important&&(ue++,this.atrule&&s("keyframes",this.atrule.name)&&le++);var De=e.property;ce.push(De),x(De)?de.push(De):function(e){if(I(e)||x(e))return!1;var t=e.charCodeAt(0);return 47===t||95===t||43===t||42===t||38===t||36===t||35===t}(De)?fe.push(De):I(De)&&he.push(De)}});var ze=R.count(),Se=Object.keys(ze.unique).join("").length,Oe=oe.count(),Ie=re.size(),je=Z.aggregate(),Ce=$.aggregate(),De=ee.aggregate(),Re=te.count(),Ue=new z(re.toArray()).count(),Fe=Y.count(),Pe=new z(J.toArray()).count(),Te=new z(Q.toArray()).count(),Ae=new z(W.toArray()).count(),_e=Object.assign;return{stylesheet:{sourceLinesOfCode:T+Ie+se+X.size(),linesOfCode:r.length,size:e.length,comments:{total:k,size:D},embeddedContent:_e(ze,{size:{total:Se,ratio:0===e.length?0:Se/e.length}})},atrules:{fontface:{total:A.length,totalUnique:A.length,unique:A,uniquenessRatio:0===A.length?0:1},import:B.count(),media:E.count(),charset:H.count(),supports:K.count(),keyframes:_e(L.count(),{prefixed:_e(M.count(),{ratio:0===L.size()?0:M.size()/L.size()})}),container:N.count(),layer:_.count()},rules:{total:V,empty:{total:G,ratio:0===V?0:G/V},sizes:_e(J.aggregate(),{items:J.toArray(),unique:Pe.unique,totalUnique:Pe.totalUnique,uniquenessRatio:Pe.uniquenessRatio}),selectors:_e(Q.aggregate(),{items:Q.toArray(),unique:Te.unique,totalUnique:Te.totalUnique,uniquenessRatio:Te.uniquenessRatio}),declarations:_e(W.aggregate(),{items:W.toArray(),unique:Ae.unique,totalUnique:Ae.totalUnique,uniquenessRatio:Ae.uniquenessRatio})},selectors:{total:Ie,totalUnique:Fe,uniquenessRatio:0===Ie?0:Fe/Ie,specificity:{min:void 0===v?[0,0,0]:v,max:void 0===h?[0,0,0]:h,sum:[je.sum,Ce.sum,De.sum],mean:[je.mean,Ce.mean,De.mean],mode:[je.mode,Ce.mode,De.mode],median:[je.median,Ce.median,De.median],items:ne,unique:Re.unique,totalUnique:Re.totalUnique,uniquenessRatio:Re.uniquenessRatio},complexity:_e(re.aggregate(),Ue,{items:re.toArray()}),id:_e(ie.count(),{ratio:0===Ie?0:ie.size()/Ie}),accessibility:_e(ae.count(),{ratio:0===Ie?0:ae.size()/Ie}),keyframes:X.count()},declarations:{total:se,unique:{total:Oe,ratio:0===se?0:Oe/se},importants:{total:ue,ratio:0===se?0:ue/se,inKeyframes:{total:le,ratio:0===ue?0:le/ue}}},properties:_e(ce.count(),{prefixed:_e(de.count(),{ratio:0===ce.size()?0:de.size()/ce.size()}),custom:_e(he.count(),{ratio:0===ce.size()?0:he.size()/ce.size()}),browserhacks:_e(fe.count(),{ratio:0===ce.size()?0:fe.size()/ce.size()})}),values:{colors:xe.count(),fontFamilies:we.count(),fontSizes:ke.count(),zindexes:be.count(),textShadows:ge.count(),boxShadows:ve.count(),animations:{durations:ye.count(),timingFunctions:pe.count()},prefixes:me.count(),units:qe.count()},__meta__:{parseTime:P-U,analyzeTime:new Date-P,total:new Date-t}}},e.compareSpecificity=l});
|
|
2
2
|
//# sourceMappingURL=analyzer.umd.js.map
|