@projectwallace/css-analyzer 5.1.0 → 5.1.1
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/package.json +1 -1
package/dist/analyzer.cjs
CHANGED
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
function e(e){if(e&&e.__esModule)return e;var t=Object.create(null);return e&&Object.keys(e).forEach(function(n){if("default"!==n){var r=Object.getOwnPropertyDescriptor(e,n);Object.defineProperty(t,n,r.get?r:{enumerable:!0,get:function(){return e[n]}})}}),t.default=e,t}var t=/*#__PURE__*/e(require("css-tree"));function n(){return n=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},n.apply(this,arguments)}function r(e,t){return e[0]===t[0]?e[1]===t[1]?t[2]-e[2]:t[1]-e[1]:t[0]-e[0]}function i(e){var n=[];return t.walk(e,{visit:"Selector",enter:function(e){n.push(a(e))}}),n.sort(function(e,t){return r(e.specificity,t.specificity)})}var a=function(e){var n=0,r=0,a=0,s=0,o=!1;return t.walk(e,{enter:function(e){switch(e.type){case"IdSelector":n++,s++;break;case"ClassSelector":r++,s++;break;case"AttributeSelector":r++,s++,Boolean(e.value)&&s++,o="role"===e.name.name||e.name.name.startsWith("aria-");break;case"PseudoElementSelector":case"TypeSelector":"*"!==e.name&&a++,s++;break;case"PseudoClassSelector":if(["before","after","first-letter","first-line"].includes(e.name))return a++,s++,this.skip;if(["is","has","matches"].includes(e.name)){var u=t.find(e,function(e){return"Raw"===e.type}),l=i(t.parse(u.value,{context:"selectorList"})),c=l[0].specificity;n+=c[0],r+=c[1],a+=c[2];for(var h=0;h<l.length;h++)s+=l[h].complexity;return void s++}if("not"===e.name){var d=i(e),m=d[0].specificity;n+=m[0],r+=m[1],a+=m[2];for(var f=0;f<d.length;f++)s+=d[f].complexity;return s++,this.skip}if(["nth-child","nth-last-child"].includes(e.name)){r++;var p=i(e);if(0===p.length)return;var g=p[0].specificity;n+=g[0],r+=g[1],a+=g[2];for(var y=0;y<p.length;y++)s+=p[y].complexity;return void s++}if("where"===e.name){for(var v=t.find(e,function(e){return"Raw"===e.type}),b=i(t.parse(v.value,{context:"selectorList"})),k=0;k<b.length;k++)s+=b[k].complexity;return void s++}s++,r++;break;case"Combinator":s++}}}),{specificity:[n,r,a],complexity:s,isId:n>0,isA11y:o}},s=/*#__PURE__*/function(){function e(e){this.items=new Uint8Array(e),this.sum=0,this.cursor=0}var t=e.prototype;return t.add=function(e){this.items[this.cursor]=e,this.sum+=e,this.cursor++},t.aggregate=function(){if(0===this.cursor)return{min:0,max:0,mean:0,mode:0,median:0,range:0,sum:0};var e,t,n,r=new Uint8Array(this.items.slice(0,this.cursor)).sort(function(e,t){return e-t}),i=r[0],a=r[r.length-1],s=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}(r),o=(t=(e=r).length/2)!==(n=Math.floor(t))?e[n]:(e[n]+e[n-1])/2;return{min:i,max:a,mean:this.sum/this.cursor,mode:s,median:o,range:a-i,sum:this.sum}},t.toArray=function(){return Array.from(this.items.subarray(0,this.cursor))},e}(),o=function(e){for(var r=e.rules,i=r.length,a=new s(i),o=new s(i),u=0,l=function(e){var n=0,i=0;t.walk(r[e],{enter:function(e){return"Selector"===e.type?(n++,this.skip):"Declaration"===e.type?(i++,this.skip):void 0}}),0===i&&u++,a.add(n),o.add(i)},c=0;c<i;c++)l(c);return{total:i,empty:{total:u,ratio:0===i?0:u/i},selectors:n({},a.aggregate(),{items:a.toArray()}),declarations:n({},o.aggregate(),{items:o.toArray()})}},u={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},l={rgb:1,rgba:1,hsl:1,hsla:1,hwb:1,lab:1,lch:1,oklab:1,oklch:1,color:1},c=/*#__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}(),h={inherit:1,initial:1,unset:1,revert:1,caption:1,icon:1,menu:1,"message-box":1,"small-caption":1,"status-bar":1},d={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},m=function(e){for(var n=e.fontValues,r=new c(e.fontFamilyValues),i=function(e){var i=n[e],a=i.children.first;if("Identifier"===a.type&&h[a.name])return"continue";var s=[];t.walk(i,{reverse:!0,enter:function(e){return"String"===e.type||"Operator"===e.type&&44===e.value.charCodeAt(0)?s.unshift(e):"Identifier"===e.type?d[e.name]?this.skip:s.unshift(e):void 0}}),r.push(s.map(t.generate).join(""))},a=0;a<n.length;a++)i(a);return r.count()},f={"xx-small":1,"x-small":1,small:1,medium:1,large:1,"x-large":1,"xx-large":1,larger:1,smaller:1},p={inherit:1,initial:1,unset:1,revert:1,caption:1,icon:1,menu:1,"message-box":1,"small-caption":1,"status-bar":1},g=function(e){for(var n=e.stringifyNode,r=e.fontValues,i=new c(e.fontSizeValues),a=function(e){var a=r[e],s=a.children.first;if("Identifier"===s.type&&p[s.name])return"continue";var o=!1,u=void 0;t.walk(a,{enter: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=n(e),this.break;case"Identifier":if(f[e.name])return u=e.name,this.break}}}),u&&i.push(u)},s=0;s<r.length;s++)a(s);return i.count()},y=function(e){for(var t=e.stringifyNode,n=e.declarations,r=n.length,i=Object.create(null),a=0,s=0,o=0,u=0;u<r;u++){var l=n[u];!0===l.important&&(a++,l.inKeyframe&&o++);var c=t(l);i[c]||(i[c]=1,s++)}return{total:r,unique:{total:s,ratio:0===r?0:s/r},importants:{total:a,ratio:0===r?0:a/r,inKeyframes:{total:o,ratio:0===a?0:o/a}}}},v=function(e){for(var t,i,o=e.stringifyNode,u=e.selectors,l=Object.create(null),h=Object.create(null),d=u.length,m=new s(d),f=new s(d),p=new s(d),g=0,y=new s(d),v=[],b=[],k=new c,w=new c,x=new c,q=0;q<d;q++){var z=u[q],O=o(z);if(z.isKeyframeSelector)x.push(O);else{var A=h[O]||a(z),C=A.specificity,N=A.complexity,S=A.isId,j=A.isA11y;S&&k.push(O),j&&w.push(O),h[O]?l[O]++:(h[O]={complexity:N,specificity:C,isId:S,isA11y:j},g++,l[O]=1),y.add(N),void 0===t&&(t=C),void 0===i&&(i=C),m.add(C[0]),f.add(C[1]),p.add(C[2]),void 0!==i&&r(i,C)<0&&(i=C),void 0!==t&&r(t,C)>0&&(t=C),v.push(C),b.push(N)}}var I=m.aggregate(),D=f.aggregate(),U=p.aggregate(),F=new c(b).count();return{total:d,totalUnique:g,uniquenessRatio:0===d?0:g/d,specificity:{min:i,max:t,sum:[I.sum,D.sum,U.sum],mean:[I.mean,D.mean,U.mean],mode:[I.mode,D.mode,U.mode],median:[I.median,D.median,U.median],items:v},complexity:n({},y.aggregate(),F,{items:b}),id:n({},k.count(),{ratio:0===d?0:k.size()/d}),accessibility:n({},w.count(),{ratio:0===d?0:w.size()/d}),keyframes:n({},x.count(),{ratio:0===d?0:x.size()/d})}},b=function(e){for(var t=e.properties,r=new c(t.map(function(e){return e.authored})),i=new c,a=new c,s=new c,o=t.length,u=0;u<o;u++){var l=t[u];l.vendor?i.push(l.authored):l.hack?a.push(l.authored):l.custom&&s.push(l.authored)}return n({},r.count(),{prefixed:n({},i.count(),{ratio:i.size()/o}),custom:n({},s.count(),{ratio:s.size()/o}),browserhacks:n({},a.count(),{ratio:a.size()/o})})},k={auto:1,inherit:1,initial:1,unset:1,revert:1,none:1},w=function(e){for(var t=e.values,n=e.stringifyNode,r=new c,i=0;i<t.length;i++){var a=t[i],s=a.children.first;s&&("Identifier"===s.type&&k[s.name]||r.push(n(a)))}return r.count()},x={linear:1,ease:1,"ease-in":1,"ease-out":1,"ease-in-out":1,"step-start":1,"step-end":1},q=function(e){for(var t=e.animations,n=e.timingFunctions,r=e.stringifyNode,i=new c(e.durations.map(r)),a=new c(n.map(r)),s=function(e){var n=!1;t[e].value.children.forEach(function(e){return"Operator"===e.type?n=!1:"Dimension"===e.type&&!1===n?(n=!0,i.push(r(e))):"Identifier"===e.type&&x[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 z(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&&z(n.name))return!0;if("Function"===n.type){if(z(n.name))return!0;if(n.children&&O(n.children))return!0}}return!1}var A=function(e){for(var t=e.values,n=e.stringifyNode,r=new c,i=0;i<t.length;i++){var a=t[i];a.children&&O(a.children)&&r.push(n(a))}return r.count()},C=function(e){for(var t=e.atrules,r=e.stringifyNode,i=[],a=new c,s=new c,o=new c,u=new c,l=new c,h=new c,d=new c,m={"font-face":function(e){var t={};e.block.children.forEach(function(e){return t[e.property]=r(e.value)}),i.push(t)},media:function(e){return s.push(e.prelude.value)},supports:function(e){return u.push(e.prelude.value)},keyframes:function(e){return l.push("@"+e.name+" "+e.prelude.value)},import:function(e){return a.push(e.prelude.value)},charset:function(e){return o.push(e.prelude.value)},container:function(e){return d.push(e.prelude.value)}},f=0;f<t.length;f++){var p=t[f],g=p.name,y=m[g];if(y)y(p);else if(g.endsWith("keyframes")){var v="@"+g+" "+p.prelude.value;l.push(v),z(g)&&h.push(v)}}return{fontface:{total:i.length,totalUnique:i.length,unique:i,uniquenessRatio:1},import:a.count(),media:s.count(),charset:o.count(),supports:u.count(),keyframes:n({},l.count(),{prefixed:n({},h.count(),{ratio:0===l.size()?0:h.size()/l.size()})}),container:d.count()}},N=/*#__PURE__*/function(){function e(){this.list=new c,this.contexts={},this.contextCount=0}var t=e.prototype;return t.push=function(e,t){this.list.push(e),this.contexts[t]||(this.contexts[t]=new c,this.contextCount++),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}();exports.analyze=function(e){var r=new Date,i=e.split(/\r?\n/);function a(e){var t=e.loc.start,n=e.loc.end;if(0==n.line-t.line)return i[t.line-1].substring(t.column-1,n.column-1);for(var r=[],a=t.line;a<=n.line;a++){var s=i[a-1];r.push(a!==t.line?a!==n.line?s:s.substring(0,n.column-1):s.substring(t.column-1))}return r.join("\n")}var s=new Date,h=0,d=0,f=t.parse(e,{parseAtrulePrelude:!1,parseCustomProperty:!0,positions:!0,onComment:function(e){h++,d+=e.length}}),p=new Date,k=[],x=[],z=[],O=[],S=[],j=[],I=[],D=[],U=[],F=[],P=[],V=[],R=[],_=[],K=[],L=new N,W=new N,E=new c;t.walk(f,{enter:function(e){switch(e.type){case"Atrule":k.push(e);break;case"Rule":x.push(e);break;case"Selector":return z.push(n({},e,{isKeyframeSelector:this.atrule&&this.atrule.name.endsWith("keyframes")})),this.skip;case"Dimension":if(!this.declaration)break;return W.push(e.unit,this.declaration.property),this.skip;case"Url":e.value.startsWith("data:")&&E.push(e.value);break;case"Declaration":O.push(n({},e,{inKeyframe:this.atrule&&this.atrule.name.endsWith("keyframes")}));var r=e.value,i=e.property,s=n({authored:i},t.property(i));switch(S.push(s),j.push(r),s.basename){case"z-index":I.push(r);break;case"text-shadow":D.push(r);break;case"box-shadow":U.push(r);break;case"font":F.push(r);break;case"font-family":return P.push(a(r)),this.skip;case"font-size":V.push(a(r));break;case"transition":case"animation":R.push(e);break;case"animation-duration":case"transition-duration":K.push(r);break;case"transition-timing-function":case"animation-timing-function":_.push(r)}t.walk(e.value,{enter:function(e){switch(e.type){case"Hash":return L.push(a(e),i),this.skip;case"Identifier":var t=e.name;return t.length>20||t.length<3||u[t.toLowerCase()]&&L.push(a(e),i),this.skip;case"Function":l[e.name.toLowerCase()]&&L.push(a(e),i)}}})}}});var T=E.count(),M=Object.keys(T.unique).join("").length;return{stylesheet:{sourceLinesOfCode:k.length+z.length+O.length,linesOfCode:i.length,size:e.length,comments:{total:h,size:d},embeddedContent:Object.assign(T,{size:{total:M,ratio:0===e.length?0:M/e.length}})},atrules:C({atrules:k,stringifyNode:a}),rules:o({rules:x}),selectors:v({stringifyNode:a,selectors:z}),declarations:y({stringifyNode:a,declarations:O}),properties:b({properties:S}),values:{colors:L.count(),fontFamilies:m({stringifyNode:a,fontValues:F,fontFamilyValues:P}),fontSizes:g({stringifyNode:a,fontValues:F,fontSizeValues:V}),zindexes:w({values:I,stringifyNode:a}),textShadows:w({values:D,stringifyNode:a}),boxShadows:w({values:U,stringifyNode:a}),animations:q({animations:R,timingFunctions:_,durations:K,stringifyNode:a}),prefixes:A({values:j,stringifyNode:a}),units:W.count()},__meta__:{parseTime:p-s,analyzeTime:new Date-p,total:new Date-r}}},exports.compareSpecificity=r;
|
|
1
|
+
var e=require("css-tree/parser"),t=require("css-tree/walker"),n=require("css-tree/generator");function r(e){return e&&"object"==typeof e&&"default"in e?e:{default:e}}var i=/*#__PURE__*/r(e),a=/*#__PURE__*/r(t),s=/*#__PURE__*/r(n);function o(){return o=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},o.apply(this,arguments)}const u=new Map,l=function(e){if(u.has(e))return u.get(e);let t=e,n=e[0];"/"===n?n="/"===e[1]?"//":"/":"_"!==n&&"*"!==n&&"$"!==n&&"#"!==n&&"+"!==n&&"&"!==n&&(n="");const r=(i=t).length-(a=(a=n.length)||0)>=2&&45===i.charCodeAt(a)&&45===i.charCodeAt(a+1);var i,a;if(!r&&(t=t.toLowerCase(),u.has(t))){const n=u.get(t);return u.set(e,n),n}const s=r?"":function(e,t){if(e.length-(t=t||0)>=3&&45===e.charCodeAt(t)&&45!==e.charCodeAt(t+1)){const n=e.indexOf("-",t+2);if(-1!==n)return e.substring(t,n+1)}return""}(t,n.length),o=t.substr(0,n.length+s.length),l=Object.freeze({basename:t.substr(o.length),name:t.substr(n.length),hack:n,vendor:s,prefix:o,custom:r});return u.set(e,l),l};function c(e,t){return e[0]===t[0]?e[1]===t[1]?t[2]-e[2]:t[1]-e[1]:t[0]-e[0]}function h(e){var t=[];return a.default(e,{visit:"Selector",enter:function(e){t.push(d(e))}}),t.sort(function(e,t){return c(e.specificity,t.specificity)})}var d=function(e){var t=0,n=0,r=0,s=0,o=!1;return a.default(e,{enter:function(e){switch(e.type){case"IdSelector":t++,s++;break;case"ClassSelector":n++,s++;break;case"AttributeSelector":n++,s++,Boolean(e.value)&&s++,o="role"===e.name.name||e.name.name.startsWith("aria-");break;case"PseudoElementSelector":case"TypeSelector":"*"!==e.name&&r++,s++;break;case"PseudoClassSelector":if(["before","after","first-letter","first-line"].includes(e.name))return r++,s++,this.skip;if(["is","has","matches","-webkit-any","-moz-any"].includes(e.name)){var a=h(i.default(e.children.first.value,{context:"selectorList"})),u=a[0].specificity;t+=u[0],n+=u[1],r+=u[2];for(var l=0;l<a.length;l++)s+=a[l].complexity;return void s++}if("not"===e.name){var c=h(e),d=c[0].specificity;t+=d[0],n+=d[1],r+=d[2];for(var f=0;f<c.length;f++)s+=c[f].complexity;return s++,this.skip}if(["nth-child","nth-last-child"].includes(e.name)){n++;var m=h(e);if(0===m.length)return;var p=m[0].specificity;t+=p[0],n+=p[1],r+=p[2];for(var g=0;g<m.length;g++)s+=m[g].complexity;return void s++}if("where"===e.name){for(var y=h(i.default(e.children.first.value,{context:"selectorList"})),v=0;v<y.length;v++)s+=y[v].complexity;return void s++}s++,n++;break;case"Combinator":s++}}}),{specificity:[t,n,r],complexity:s,isId:t>0,isA11y:o}},f=/*#__PURE__*/function(){function e(e){this.items=new Uint8Array(e),this.sum=0,this.cursor=0}var t=e.prototype;return t.add=function(e){this.items[this.cursor]=e,this.sum+=e,this.cursor++},t.aggregate=function(){if(0===this.cursor)return{min:0,max:0,mean:0,mode:0,median:0,range:0,sum:0};var e,t,n,r=new Uint8Array(this.items.slice(0,this.cursor)).sort(function(e,t){return e-t}),i=r[0],a=r[r.length-1],s=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}(r),o=(t=(e=r).length/2)!==(n=Math.floor(t))?e[n]:(e[n]+e[n-1])/2;return{min:i,max:a,mean:this.sum/this.cursor,mode:s,median:o,range:a-i,sum:this.sum}},t.toArray=function(){return Array.from(this.items.subarray(0,this.cursor))},e}(),m=function(e){for(var t=e.rules,n=t.length,r=new f(n),i=new f(n),s=0,u=function(e){var n=0,o=0;a.default(t[e],{enter:function(e){return"Selector"===e.type?(n++,this.skip):"Declaration"===e.type?(o++,this.skip):void 0}}),0===o&&s++,r.add(n),i.add(o)},l=0;l<n;l++)u(l);return{total:n,empty:{total:s,ratio:0===n?0:s/n},selectors:o({},r.aggregate(),{items:r.toArray()}),declarations:o({},i.aggregate(),{items:i.toArray()})}},p={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},g={rgb:1,rgba:1,hsl:1,hsla:1,hwb:1,lab:1,lch:1,oklab:1,oklch:1,color:1},y=/*#__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}(),v={inherit:1,initial:1,unset:1,revert:1,caption:1,icon:1,menu:1,"message-box":1,"small-caption":1,"status-bar":1},b={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},k=function(e){for(var t=e.fontValues,n=new y(e.fontFamilyValues),r=function(e){var r=t[e],i=r.children.first;if("Identifier"===i.type&&v[i.name])return"continue";var o=[];a.default(r,{reverse:!0,enter:function(e){return"String"===e.type||"Operator"===e.type&&44===e.value.charCodeAt(0)?o.unshift(e):"Identifier"===e.type?b[e.name]?this.skip:o.unshift(e):void 0}}),n.push(o.map(s.default).join(""))},i=0;i<t.length;i++)r(i);return n.count()},w={"xx-small":1,"x-small":1,small:1,medium:1,large:1,"x-large":1,"xx-large":1,larger:1,smaller:1},x={inherit:1,initial:1,unset:1,revert:1,caption:1,icon:1,menu:1,"message-box":1,"small-caption":1,"status-bar":1},q=function(e){for(var t=e.stringifyNode,n=e.fontValues,r=new y(e.fontSizeValues),i=function(e){var i=n[e],s=i.children.first;if("Identifier"===s.type&&x[s.name])return"continue";var o=!1,u=void 0;a.default(i,{enter: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(w[e.name])return u=e.name,this.break}}}),u&&r.push(u)},s=0;s<n.length;s++)i(s);return r.count()},z=function(e){for(var t=e.stringifyNode,n=e.declarations,r=n.length,i=Object.create(null),a=0,s=0,o=0,u=0;u<r;u++){var l=n[u];!0===l.important&&(a++,l.inKeyframe&&o++);var c=t(l);i[c]||(i[c]=1,s++)}return{total:r,unique:{total:s,ratio:0===r?0:s/r},importants:{total:a,ratio:0===r?0:a/r,inKeyframes:{total:o,ratio:0===a?0:o/a}}}},C=function(e){for(var t,n,r=e.stringifyNode,i=e.selectors,a=Object.create(null),s=Object.create(null),u=i.length,l=new f(u),h=new f(u),m=new f(u),p=0,g=new f(u),v=[],b=[],k=new y,w=new y,x=new y,q=0;q<u;q++){var z=i[q],C=r(z);if(z.isKeyframeSelector)x.push(C);else{var A=s[C]||d(z),O=A.specificity,N=A.complexity,S=A.isId,j=A.isA11y;S&&k.push(C),j&&w.push(C),s[C]?a[C]++:(s[C]={complexity:N,specificity:O,isId:S,isA11y:j},p++,a[C]=1),g.add(N),void 0===t&&(t=O),void 0===n&&(n=O),l.add(O[0]),h.add(O[1]),m.add(O[2]),void 0!==n&&c(n,O)<0&&(n=O),void 0!==t&&c(t,O)>0&&(t=O),v.push(O),b.push(N)}}var I=l.aggregate(),D=h.aggregate(),U=m.aggregate(),F=new y(b).count();return{total:u,totalUnique:p,uniquenessRatio:0===u?0:p/u,specificity:{min:n,max:t,sum:[I.sum,D.sum,U.sum],mean:[I.mean,D.mean,U.mean],mode:[I.mode,D.mode,U.mode],median:[I.median,D.median,U.median],items:v},complexity:o({},g.aggregate(),F,{items:b}),id:o({},k.count(),{ratio:0===u?0:k.size()/u}),accessibility:o({},w.count(),{ratio:0===u?0:w.size()/u}),keyframes:o({},x.count(),{ratio:0===u?0:x.size()/u})}},A=function(e){for(var t=e.properties,n=new y(t.map(function(e){return e.authored})),r=new y,i=new y,a=new y,s=t.length,u=0;u<s;u++){var l=t[u];l.vendor?r.push(l.authored):l.hack?i.push(l.authored):l.custom&&a.push(l.authored)}return o({},n.count(),{prefixed:o({},r.count(),{ratio:r.size()/s}),custom:o({},a.count(),{ratio:a.size()/s}),browserhacks:o({},i.count(),{ratio:i.size()/s})})},O={auto:1,inherit:1,initial:1,unset:1,revert:1,none:1},N=function(e){for(var t=e.values,n=e.stringifyNode,r=new y,i=0;i<t.length;i++){var a=t[i],s=a.children.first;s&&("Identifier"===s.type&&O[s.name]||r.push(n(a)))}return r.count()},S={linear:1,ease:1,"ease-in":1,"ease-out":1,"ease-in-out":1,"step-start":1,"step-end":1},j=function(e){for(var t=e.animations,n=e.timingFunctions,r=e.stringifyNode,i=new y(e.durations.map(r)),a=new y(n.map(r)),s=function(e){var n=!1;t[e].value.children.forEach(function(e){return"Operator"===e.type?n=!1:"Dimension"===e.type&&!1===n?(n=!0,i.push(r(e))):"Identifier"===e.type&&S[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 I(e){return 45===e.charCodeAt(0)&&45!==e.charCodeAt(1)&&-1!==e.indexOf("-",2)}function D(e){e=e.toArray();for(var t=0;t<e.length;t++){var n=e[t];if("Identifier"===n.type&&n.name.length>=3&&I(n.name))return!0;if("Function"===n.type){if(I(n.name))return!0;if(n.children&&D(n.children))return!0}}return!1}var U=function(e){for(var t=e.values,n=e.stringifyNode,r=new y,i=0;i<t.length;i++){var a=t[i];a.children&&D(a.children)&&r.push(n(a))}return r.count()},F=function(e){for(var t=e.atrules,n=e.stringifyNode,r=[],i=new y,a=new y,s=new y,u=new y,l=new y,c=new y,h=new y,d={"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 a.push(e.prelude.value)},supports:function(e){return u.push(e.prelude.value)},keyframes:function(e){return l.push("@"+e.name+" "+e.prelude.value)},import:function(e){return i.push(e.prelude.value)},charset:function(e){return s.push(e.prelude.value)},container:function(e){return h.push(e.prelude.value)}},f=0;f<t.length;f++){var m=t[f],p=m.name,g=d[p];if(g)g(m);else if(p.endsWith("keyframes")){var v="@"+p+" "+m.prelude.value;l.push(v),I(p)&&c.push(v)}}return{fontface:{total:r.length,totalUnique:r.length,unique:r,uniquenessRatio:1},import:i.count(),media:a.count(),charset:s.count(),supports:u.count(),keyframes:o({},l.count(),{prefixed:o({},c.count(),{ratio:0===l.size()?0:c.size()/l.size()})}),container:h.count()}},V=/*#__PURE__*/function(){function e(){this.list=new y,this.contexts={},this.contextCount=0}var t=e.prototype;return t.push=function(e,t){this.list.push(e),this.contexts[t]||(this.contexts[t]=new y,this.contextCount++),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}();exports.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.push(a!==t.line?a!==r.line?s:s.substring(0,r.column-1):s.substring(t.column-1))}return i.join("\n")}var s=new Date,u=0,c=0,h=i.default(e,{parseAtrulePrelude:!1,parseCustomProperty:!0,positions:!0,onComment:function(e){u++,c+=e.length}}),d=new Date,f=[],v=[],b=[],w=[],x=[],O=[],S=[],I=[],D=[],L=[],P=[],K=[],W=[],_=[],R=[],E=new V,T=new V,M=new y;a.default(h,{enter:function(e){switch(e.type){case"Atrule":f.push(e);break;case"Rule":v.push(e);break;case"Selector":return b.push(o({},e,{isKeyframeSelector:this.atrule&&this.atrule.name.endsWith("keyframes")})),this.skip;case"Dimension":if(!this.declaration)break;return T.push(e.unit,this.declaration.property),this.skip;case"Url":e.value.startsWith("data:")&&M.push(e.value);break;case"Declaration":w.push(o({},e,{inKeyframe:this.atrule&&this.atrule.name.endsWith("keyframes")}));var t=e.value,n=e.property,i=Object.assign({authored:n},l(n));switch(x.push(i),O.push(t),i.basename){case"z-index":S.push(t);break;case"text-shadow":I.push(t);break;case"box-shadow":D.push(t);break;case"font":L.push(t);break;case"font-family":return P.push(r(t)),this.skip;case"font-size":K.push(r(t));break;case"transition":case"animation":W.push(e);break;case"animation-duration":case"transition-duration":R.push(t);break;case"transition-timing-function":case"animation-timing-function":_.push(t)}a.default(t,{enter:function(e){switch(e.type){case"Hash":return E.push(r(e),n),this.skip;case"Identifier":var t=e.name;return t.length>20||t.length<3||p[t.toLowerCase()]&&E.push(r(e),n),this.skip;case"Function":g[e.name.toLowerCase()]&&E.push(r(e),n)}}})}}});var B=M.count(),H=Object.keys(B.unique).join("").length;return{stylesheet:{sourceLinesOfCode:f.length+b.length+w.length,linesOfCode:n.length,size:e.length,comments:{total:u,size:c},embeddedContent:Object.assign(B,{size:{total:H,ratio:0===e.length?0:H/e.length}})},atrules:F({atrules:f,stringifyNode:r}),rules:m({rules:v}),selectors:C({stringifyNode:r,selectors:b}),declarations:z({stringifyNode:r,declarations:w}),properties:A({properties:x}),values:{colors:E.count(),fontFamilies:k({stringifyNode:r,fontValues:L,fontFamilyValues:P}),fontSizes:q({stringifyNode:r,fontValues:L,fontSizeValues:K}),zindexes:N({values:S,stringifyNode:r}),textShadows:N({values:I,stringifyNode:r}),boxShadows:N({values:D,stringifyNode:r}),animations:j({animations:W,timingFunctions:_,durations:R,stringifyNode:r}),prefixes:U({values:O,stringifyNode:r}),units:T.count()},__meta__:{parseTime:d-s,analyzeTime:new Date-d,total:new Date-t}}},exports.compareSpecificity=c;
|
|
2
2
|
//# sourceMappingURL=analyzer.cjs.map
|
package/dist/analyzer.cjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"analyzer.cjs","sources":["../src/selectors/specificity.js","../src/aggregate-collection.js","../src/rules/rules.js","../src/values/colors.js","../src/countable-collection.js","../src/values/font-families.js","../src/values/font-sizes.js","../src/declarations/declarations.js","../src/selectors/selectors.js","../src/properties/properties.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/index.js"],"sourcesContent":["import * as csstree from 'css-tree'\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 csstree.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 = (ast) => {\n let A = 0\n let B = 0\n let C = 0\n let complexity = 0\n let isA11y = false\n\n csstree.walk(ast, {\n enter: 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 if (Boolean(selector.value)) {\n complexity++\n }\n isA11y = selector.name.name === 'role' || selector.name.name.startsWith('aria-')\n break\n }\n case 'PseudoElementSelector':\n case 'TypeSelector': {\n if (selector.name !== '*') {\n C++\n }\n complexity++\n break\n }\n case 'PseudoClassSelector': {\n if (['before', 'after', 'first-letter', 'first-line'].includes(selector.name)) {\n C++\n complexity++\n return this.skip\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\n // CSSTree doesn't parse the arguments of :is, :has and :matches,\n // so we need to create an AST out of them ourselves\n if (['is', 'has', 'matches'].includes(selector.name)) {\n const rawSelectorList = csstree.find(selector, ({ type }) => type === 'Raw')\n const childAst = csstree.parse(rawSelectorList.value, { context: 'selectorList' })\n const selectorList = selectorListSpecificities(childAst)\n const [topA, topB, topC] = selectorList[0].specificity\n A += topA\n B += topB\n C += topC\n\n for (let i = 0; i < selectorList.length; i++) {\n complexity += selectorList[i].complexity\n }\n complexity++\n return\n }\n\n // CSSTree *does* parse the arguments of the :not() pseudo-class,\n // so we have direct access to the AST, instead of having to parse\n // the arguments ourselves.\n if (selector.name === 'not') {\n const selectorList = selectorListSpecificities(selector)\n const [topA, topB, topC] = selectorList[0].specificity\n A += topA\n B += topB\n C += topC\n\n for (let i = 0; i < selectorList.length; i++) {\n complexity += selectorList[i].complexity\n }\n complexity++\n return this.skip\n }\n\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 (['nth-child', 'nth-last-child'].includes(selector.name)) {\n // +1 for the pseudo class itself\n B++\n\n const childSelectors = selectorListSpecificities(selector)\n\n if (childSelectors.length === 0) {\n return\n }\n\n const [topA, topB, topC] = childSelectors[0].specificity\n A += topA\n B += topB\n C += topC\n\n for (let i = 0; i < childSelectors.length; i++) {\n complexity += childSelectors[i].complexity;\n }\n\n complexity++\n return\n }\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 rawSelectorList = csstree.find(selector, ({ type }) => type === 'Raw')\n const childAst = csstree.parse(rawSelectorList.value, { context: 'selectorList' })\n const childSelectors = selectorListSpecificities(childAst)\n\n for (let i = 0; i < childSelectors.length; i++) {\n complexity += childSelectors[i].complexity;\n }\n\n complexity++\n return\n }\n\n // Regular pseudo classes have specificity [0,1,0]\n complexity++\n B++\n break\n }\n case 'Combinator': {\n complexity++\n break\n }\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}","/**\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(size) {\n /** @type number[] */\n this.items = new Uint8Array(size)\n this.sum = 0\n this.cursor = 0\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[this.cursor] = item\n this.sum += item\n this.cursor++\n }\n\n aggregate() {\n if (this.cursor === 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 = new Uint8Array(\n this.items.slice(0, this.cursor)\n ).sort((a, b) => a - b)\n const min = sorted[0]\n const max = sorted[sorted.length - 1]\n\n const mode = Mode(sorted)\n const median = Median(sorted)\n\n return {\n min,\n max,\n mean: this.sum / this.cursor,\n mode,\n median,\n range: max - min,\n sum: this.sum,\n }\n }\n\n /**\n * @returns {number[]} All items in this collection\n */\n toArray() {\n return Array.from(\n this.items.subarray(0, this.cursor)\n )\n }\n}\n\nexport {\n AggregateCollection\n}","import * as csstree from 'css-tree'\nimport { AggregateCollection } from '../aggregate-collection.js'\n\nconst analyzeRules = ({ rules }) => {\n /** @type number */\n const totalRules = rules.length\n const selectorsPerRule = new AggregateCollection(totalRules)\n const declarationsPerRule = new AggregateCollection(totalRules)\n\n let emptyRules = 0\n\n for (let i = 0; i < totalRules; i++) {\n let selectors = 0\n let declarations = 0\n\n csstree.walk(rules[i], {\n enter: function (childNode) {\n if (childNode.type === 'Selector') {\n selectors++\n return this.skip\n }\n\n if (childNode.type === 'Declaration') {\n declarations++\n return this.skip\n }\n }\n })\n\n if (declarations === 0) {\n emptyRules++\n }\n\n // For later aggregations\n selectorsPerRule.add(selectors)\n declarationsPerRule.add(declarations)\n }\n\n return {\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}\n\nexport {\n analyzeRules\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 * as csstree from 'css-tree'\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 }) => {\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 const parts = []\n\n csstree.walk(value, {\n reverse: true,\n enter: function (fontNode) {\n if (fontNode.type === 'String') {\n return parts.unshift(fontNode)\n }\n if (fontNode.type === 'Operator' && fontNode.value.charCodeAt(0) === COMMA) {\n return parts.unshift(fontNode)\n }\n if (fontNode.type === 'Identifier') {\n if (keywordDisallowList[fontNode.name]) {\n return this.skip\n }\n return parts.unshift(fontNode)\n }\n }\n })\n\n all.push(parts.map(csstree.generate).join(''))\n }\n\n return all.count()\n}\n\nexport {\n analyzeFontFamilies\n}","import * as csstree from 'css-tree'\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 csstree.walk(fontNode, {\n enter: 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\n if (size) {\n all.push(size)\n }\n }\n\n return all.count()\n}\n\nexport {\n analyzeFontSizes\n}","const analyzeDeclarations = ({ stringifyNode, declarations }) => {\n /** @type number */\n const total = declarations.length\n const cache = Object.create(null)\n let importants = 0\n let totalUnique = 0\n let totalInKeyframes = 0\n\n for (let i = 0; i < total; i++) {\n const declaration = declarations[i]\n\n if (declaration.important === true) {\n importants++\n\n if (declaration.inKeyframe) {\n totalInKeyframes++\n }\n }\n\n const stringified = stringifyNode(declaration)\n\n if (!cache[stringified]) {\n cache[stringified] = 1\n totalUnique++\n }\n }\n\n return {\n total,\n unique: {\n total: totalUnique,\n ratio: total === 0 ? 0 : totalUnique / total,\n },\n importants: {\n total: importants,\n ratio: total === 0 ? 0 : importants / total,\n inKeyframes: {\n total: totalInKeyframes,\n ratio: importants === 0 ? 0 : totalInKeyframes / importants,\n },\n },\n }\n}\n\nexport {\n analyzeDeclarations\n}","import { analyzeSpecificity, compareSpecificity } from './specificity.js'\nimport { AggregateCollection } from '../aggregate-collection.js'\nimport { CountableCollection } from '../countable-collection.js'\n\nconst analyzeSelectors = ({ stringifyNode, selectors }) => {\n const counts = Object.create(null)\n const cache = Object.create(null)\n /** @type number */\n const totalSelectors = selectors.length\n\n /** @type [number,number,number] */\n let maxSpecificity\n /** @type [number,number,number] */\n let minSpecificity\n let specificityA = new AggregateCollection(totalSelectors)\n let specificityB = new AggregateCollection(totalSelectors)\n let specificityC = new AggregateCollection(totalSelectors)\n let totalUnique = 0\n const complexityAggregator = new AggregateCollection(totalSelectors);\n\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 const keyframes = new CountableCollection()\n\n for (let i = 0; i < totalSelectors; i++) {\n /** @type import('css-tree').Selector */\n const node = selectors[i];\n /** @type string */\n const value = stringifyNode(node)\n\n if (node.isKeyframeSelector) {\n keyframes.push(value)\n // Do not attempt to further analyze <keyframe-selectors>\n continue\n }\n\n const { specificity, complexity, isId, isA11y } = cache[value] || analyzeSpecificity(node)\n\n if (isId) {\n ids.push(value)\n }\n\n if (isA11y) {\n a11y.push(value)\n }\n\n if (!cache[value]) {\n cache[value] = { complexity, specificity, isId, isA11y }\n totalUnique++\n counts[value] = 1\n } else {\n counts[value]++\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\n const aggregatesA = specificityA.aggregate()\n const aggregatesB = specificityB.aggregate()\n const aggregatesC = specificityC.aggregate()\n const complexityCount = new CountableCollection(complexities).count()\n\n return {\n total: totalSelectors,\n totalUnique,\n uniquenessRatio: totalSelectors === 0 ? 0 : totalUnique / totalSelectors,\n specificity: {\n /** @type [number, number, number] */\n min: minSpecificity,\n /** @type [number, number, number] */\n max: maxSpecificity,\n /** @type [number, number, number] */\n sum: [aggregatesA.sum, aggregatesB.sum, aggregatesC.sum],\n /** @type [number, number, number] */\n mean: [aggregatesA.mean, aggregatesB.mean, aggregatesC.mean],\n /** @type [number, number, number] */\n mode: [aggregatesA.mode, aggregatesB.mode, aggregatesC.mode],\n /** @type [number, number, number] */\n median: [aggregatesA.median, aggregatesB.median, aggregatesC.median],\n /** @type [number, number, number][] */\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: {\n ...keyframes.count(),\n ratio: totalSelectors === 0 ? 0 : keyframes.size() / totalSelectors,\n }\n }\n}\n\nexport {\n analyzeSelectors\n}","import { CountableCollection } from '../countable-collection.js'\n\nconst analyzeProperties = ({ properties }) => {\n const all = new CountableCollection(properties.map(p => p.authored))\n const prefixed = new CountableCollection()\n const hacks = new CountableCollection()\n const customs = new CountableCollection()\n const totalProperties = properties.length\n\n for (let i = 0; i < totalProperties; i++) {\n const property = properties[i]\n\n if (property.vendor) {\n prefixed.push(property.authored)\n continue\n }\n\n if (property.hack) {\n hacks.push(property.authored)\n continue\n }\n\n if (property.custom) {\n customs.push(property.authored)\n continue\n }\n }\n\n return {\n ...all.count(),\n prefixed: {\n ...prefixed.count(),\n ratio: prefixed.size() / totalProperties,\n },\n custom: {\n ...customs.count(),\n ratio: customs.size() / totalProperties,\n },\n browserhacks: {\n ...hacks.count(),\n ratio: hacks.size() / totalProperties,\n }\n }\n}\n\nexport {\n analyzeProperties\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`\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.map(stringifyNode))\n const allTimingFunctions = new CountableCollection(timingFunctions.map(stringifyNode))\n\n for (let index = 0; index < animations.length; index++) {\n const node = animations[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 node.value.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 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'\n\nconst analyzeAtRules = ({ atrules, stringifyNode }) => {\n /** @type {{[index: string]: string}[]} */\n const fontfaces = []\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.value),\n 'supports': node => supports.push(node.prelude.value),\n 'keyframes': node => keyframes.push(`@${node.name} ${node.prelude.value}`),\n 'import': node => imports.push(node.prelude.value),\n 'charset': node => charsets.push(node.prelude.value),\n 'container': node => containers.push(node.prelude.value),\n }\n\n for (let i = 0; i < atrules.length; i++) {\n /** @type {import('css-tree').Atrule} */\n const node = atrules[i]\n const nodeName = node.name\n const action = machine[nodeName]\n if (action) {\n action(node)\n continue\n }\n\n if (nodeName.endsWith('keyframes')) {\n const name = `@${nodeName} ${node.prelude.value}`\n keyframes.push(name)\n\n if (hasVendorPrefix(nodeName)) {\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: 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 }\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 this.contextCount = 0\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 this.contextCount++\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}","import * as csstree from 'css-tree'\nimport { compareSpecificity } from './selectors/specificity.js'\nimport { analyzeRules } from './rules/rules.js'\nimport { colorFunctions, colorNames } from './values/colors.js'\nimport { analyzeFontFamilies } from './values/font-families.js'\nimport { analyzeFontSizes } from './values/font-sizes.js'\nimport { analyzeDeclarations } from './declarations/declarations.js'\nimport { analyzeSelectors } from './selectors/selectors.js'\nimport { analyzeProperties } from './properties/properties.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'\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 const 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.push(line.substring(start.column - 1))\n continue\n }\n // Last line\n if (i === end.line) {\n value.push(line.substring(0, end.column - 1))\n continue\n }\n // All lines in between first and last\n value.push(line)\n }\n\n return value.join('\\n')\n }\n\n const startParse = new Date()\n let totalComments = 0\n let commentsSize = 0\n\n const ast = csstree.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 atrules = []\n const rules = []\n const selectors = []\n const declarations = []\n const properties = []\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 const embeds = new CountableCollection()\n\n csstree.walk(ast, {\n enter: function (node) {\n switch (node.type) {\n case 'Atrule': {\n atrules.push(node)\n break\n }\n case 'Rule': {\n rules.push(node)\n break\n }\n case 'Selector': {\n selectors.push({\n ...node,\n isKeyframeSelector: this.atrule && this.atrule.name.endsWith('keyframes')\n })\n\n // Avoid further 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 (node.value.startsWith('data:')) {\n embeds.push(node.value)\n }\n break\n }\n case 'Declaration': {\n declarations.push({\n ...node,\n inKeyframe: this.atrule && this.atrule.name.endsWith('keyframes')\n })\n\n const { value, property } = node\n const fullProperty = {\n authored: property,\n ...csstree.property(property)\n }\n\n properties.push(fullProperty)\n values.push(value)\n\n switch (fullProperty.basename) {\n case 'z-index': {\n zindex.push(value)\n break\n }\n case 'text-shadow': {\n textShadows.push(value)\n break\n }\n case 'box-shadow': {\n boxShadows.push(value)\n break\n }\n case 'font': {\n fontValues.push(value)\n break\n }\n case 'font-family': {\n fontFamilyValues.push(stringifyNode(value))\n // Prevent analyzer to find color names in this property\n return this.skip\n }\n case 'font-size': {\n fontSizeValues.push(stringifyNode(value))\n break\n }\n case 'transition':\n case 'animation': {\n animations.push(node)\n break\n }\n case 'animation-duration':\n case 'transition-duration': {\n durations.push(value)\n break\n }\n case 'transition-timing-function':\n case 'animation-timing-function': {\n timingFunctions.push(value)\n break\n }\n }\n\n csstree.walk(node.value, {\n enter: function (valueNode) {\n switch (valueNode.type) {\n case 'Hash': {\n colors.push(stringifyNode(valueNode), 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 })\n const embeddedContent = embeds.count()\n const embedSize = Object.keys(embeddedContent.unique).join('').length\n\n return {\n stylesheet: {\n sourceLinesOfCode: atrules.length + selectors.length + declarations.length,\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: analyzeRules({ rules }),\n selectors: analyzeSelectors({ stringifyNode, selectors }),\n declarations: analyzeDeclarations({ stringifyNode, declarations }),\n properties: analyzeProperties({ properties }),\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"],"names":["compareSpecificity","a","b","selectorListSpecificities","selectorListAst","childSelectors","csstree","walk","visit","enter","node","push","analyzeSpecificity","sort","specificity","ast","A","B","C","complexity","isA11y","selector","type","Boolean","value","name","startsWith","includes","skip","rawSelectorList","find","selectorList","parse","context","i","length","isId","AggregateCollection","size","this","items","Uint8Array","sum","cursor","add","item","aggregate","min","max","mean","mode","median","range","arr","middle","lowerMiddleRank","sorted","slice","frequencies","Object","create","maxOccurrences","maxOccurenceCount","element","updatedCount","Mode","Math","floor","toArray","Array","from","subarray","analyzeRules","rules","totalRules","selectorsPerRule","declarationsPerRule","emptyRules","selectors","declarations","childNode","total","empty","ratio","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","totalUnique","index","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","all","fontFamilyValues","firstChild","children","first","parts","reverse","fontNode","charCodeAt","unshift","map","generate","join","sizeKeywords","keywords","analyzeFontSizes","stringifyNode","fontSizeValues","operator","analyzeDeclarations","cache","importants","totalInKeyframes","declaration","important","inKeyframe","stringified","inKeyframes","analyzeSelectors","maxSpecificity","minSpecificity","counts","totalSelectors","specificityA","specificityB","specificityC","complexityAggregator","specificities","complexities","ids","a11y","keyframes","isKeyframeSelector","undefined","aggregatesA","aggregatesB","aggregatesC","complexityCount","id","accessibility","analyzeProperties","properties","p","authored","prefixed","hacks","customs","totalProperties","property","vendor","hack","custom","browserhacks","auto","none","analyzeValues","values","timingKeywords","linear","ease","analyzeAnimations","animations","timingFunctions","allDurations","durations","allTimingFunctions","durationFound","forEach","child","hasVendorPrefix","keyword","indexOf","isAstVendorPrefixed","analyzeVendorPrefixes","analyzeAtRules","atrules","fontfaces","imports","medias","charsets","supports","prefixedKeyframes","containers","machine","descriptors","block","descriptor","media","prelude","import","charset","container","nodeName","action","endsWith","fontface","ContextCollection","list","contexts","contextCount","itemsPerContext","assign","css","start","Date","lines","split","loc","end","line","substring","column","startParse","totalComments","commentsSize","parseAtrulePrelude","parseCustomProperty","positions","onComment","comment","startAnalysis","zindex","textShadows","boxShadows","colors","units","embeds","atrule","unit","fullProperty","basename","valueNode","toLowerCase","embeddedContent","embedSize","keys","stylesheet","sourceLinesOfCode","linesOfCode","comments","fontFamilies","fontSizes","zindexes","prefixes","__meta__","parseTime","analyzeTime"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"analyzer.cjs","sources":["../node_modules/css-tree/lib/utils/names.js","../src/selectors/specificity.js","../src/aggregate-collection.js","../src/rules/rules.js","../src/values/colors.js","../src/countable-collection.js","../src/values/font-families.js","../src/values/font-sizes.js","../src/declarations/declarations.js","../src/selectors/selectors.js","../src/properties/properties.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/index.js"],"sourcesContent":["const keywords = new Map();\nconst properties = new Map();\nconst HYPHENMINUS = 45; // '-'.charCodeAt()\n\nexport const keyword = getKeywordDescriptor;\nexport const property = getPropertyDescriptor;\nexport const vendorPrefix = getVendorPrefix;\nexport function isCustomProperty(str, offset) {\n offset = offset || 0;\n\n return str.length - offset >= 2 &&\n str.charCodeAt(offset) === HYPHENMINUS &&\n str.charCodeAt(offset + 1) === HYPHENMINUS;\n}\n\nfunction getVendorPrefix(str, offset) {\n offset = offset || 0;\n\n // verdor prefix should be at least 3 chars length\n if (str.length - offset >= 3) {\n // vendor prefix starts with hyper minus following non-hyper minus\n if (str.charCodeAt(offset) === HYPHENMINUS &&\n str.charCodeAt(offset + 1) !== HYPHENMINUS) {\n // vendor prefix should contain a hyper minus at the ending\n const secondDashIndex = str.indexOf('-', offset + 2);\n\n if (secondDashIndex !== -1) {\n return str.substring(offset, secondDashIndex + 1);\n }\n }\n }\n\n return '';\n}\n\nfunction getKeywordDescriptor(keyword) {\n if (keywords.has(keyword)) {\n return keywords.get(keyword);\n }\n\n const name = keyword.toLowerCase();\n let descriptor = keywords.get(name);\n\n if (descriptor === undefined) {\n const custom = isCustomProperty(name, 0);\n const vendor = !custom ? getVendorPrefix(name, 0) : '';\n descriptor = Object.freeze({\n basename: name.substr(vendor.length),\n name,\n prefix: vendor,\n vendor,\n custom\n });\n }\n\n keywords.set(keyword, descriptor);\n\n return descriptor;\n}\n\nfunction getPropertyDescriptor(property) {\n if (properties.has(property)) {\n return properties.get(property);\n }\n\n let name = property;\n let hack = property[0];\n\n if (hack === '/') {\n hack = property[1] === '/' ? '//' : '/';\n } else if (hack !== '_' &&\n hack !== '*' &&\n hack !== '$' &&\n hack !== '#' &&\n hack !== '+' &&\n hack !== '&') {\n hack = '';\n }\n\n const custom = isCustomProperty(name, hack.length);\n\n // re-use result when possible (the same as for lower case)\n if (!custom) {\n name = name.toLowerCase();\n if (properties.has(name)) {\n const descriptor = properties.get(name);\n properties.set(property, descriptor);\n return descriptor;\n }\n }\n\n const vendor = !custom ? getVendorPrefix(name, hack.length) : '';\n const prefix = name.substr(0, hack.length + vendor.length);\n const descriptor = Object.freeze({\n basename: name.substr(prefix.length),\n name: name.substr(hack.length),\n hack,\n vendor,\n prefix,\n custom\n });\n\n properties.set(property, descriptor);\n\n return descriptor;\n}\n","import parse from 'css-tree/parser'\nimport walk from 'css-tree/walker'\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 = (ast) => {\n let A = 0\n let B = 0\n let C = 0\n let complexity = 0\n let isA11y = false\n\n walk(ast, {\n enter: 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 if (Boolean(selector.value)) {\n complexity++\n }\n isA11y = selector.name.name === 'role' || selector.name.name.startsWith('aria-')\n break\n }\n case 'PseudoElementSelector':\n case 'TypeSelector': {\n if (selector.name !== '*') {\n C++\n }\n complexity++\n break\n }\n case 'PseudoClassSelector': {\n if (['before', 'after', 'first-letter', 'first-line'].includes(selector.name)) {\n C++\n complexity++\n return this.skip\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\n // CSSTree doesn't parse the arguments of :is, :has and :matches,\n // so we need to create an AST out of them ourselves\n if (['is', 'has', 'matches', '-webkit-any', '-moz-any'].includes(selector.name)) {\n const childAst = parse(selector.children.first.value, { context: 'selectorList' })\n const selectorList = selectorListSpecificities(childAst)\n const [topA, topB, topC] = selectorList[0].specificity\n A += topA\n B += topB\n C += topC\n\n for (let i = 0; i < selectorList.length; i++) {\n complexity += selectorList[i].complexity\n }\n complexity++\n return\n }\n\n // CSSTree *does* parse the arguments of the :not() pseudo-class,\n // so we have direct access to the AST, instead of having to parse\n // the arguments ourselves.\n if (selector.name === 'not') {\n const selectorList = selectorListSpecificities(selector)\n const [topA, topB, topC] = selectorList[0].specificity\n A += topA\n B += topB\n C += topC\n\n for (let i = 0; i < selectorList.length; i++) {\n complexity += selectorList[i].complexity\n }\n complexity++\n return this.skip\n }\n\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 (['nth-child', 'nth-last-child'].includes(selector.name)) {\n // +1 for the pseudo class itself\n B++\n\n const childSelectors = selectorListSpecificities(selector)\n\n if (childSelectors.length === 0) {\n return\n }\n\n const [topA, topB, topC] = childSelectors[0].specificity\n A += topA\n B += topB\n C += topC\n\n for (let i = 0; i < childSelectors.length; i++) {\n complexity += childSelectors[i].complexity;\n }\n\n complexity++\n return\n }\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 childAst = parse(selector.children.first.value, { context: 'selectorList' })\n const childSelectors = selectorListSpecificities(childAst)\n\n for (let i = 0; i < childSelectors.length; i++) {\n complexity += childSelectors[i].complexity;\n }\n\n complexity++\n return\n }\n\n // Regular pseudo classes have specificity [0,1,0]\n complexity++\n B++\n break\n }\n case 'Combinator': {\n complexity++\n break\n }\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}","/**\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(size) {\n /** @type number[] */\n this.items = new Uint8Array(size)\n this.sum = 0\n this.cursor = 0\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[this.cursor] = item\n this.sum += item\n this.cursor++\n }\n\n aggregate() {\n if (this.cursor === 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 = new Uint8Array(\n this.items.slice(0, this.cursor)\n ).sort((a, b) => a - b)\n const min = sorted[0]\n const max = sorted[sorted.length - 1]\n\n const mode = Mode(sorted)\n const median = Median(sorted)\n\n return {\n min,\n max,\n mean: this.sum / this.cursor,\n mode,\n median,\n range: max - min,\n sum: this.sum,\n }\n }\n\n /**\n * @returns {number[]} All items in this collection\n */\n toArray() {\n return Array.from(\n this.items.subarray(0, this.cursor)\n )\n }\n}\n\nexport {\n AggregateCollection\n}","import walk from 'css-tree/walker'\nimport { AggregateCollection } from '../aggregate-collection.js'\n\nconst analyzeRules = ({ rules }) => {\n /** @type number */\n const totalRules = rules.length\n const selectorsPerRule = new AggregateCollection(totalRules)\n const declarationsPerRule = new AggregateCollection(totalRules)\n\n let emptyRules = 0\n\n for (let i = 0; i < totalRules; i++) {\n let selectors = 0\n let declarations = 0\n\n walk(rules[i], {\n enter: function (childNode) {\n if (childNode.type === 'Selector') {\n selectors++\n return this.skip\n }\n\n if (childNode.type === 'Declaration') {\n declarations++\n return this.skip\n }\n }\n })\n\n if (declarations === 0) {\n emptyRules++\n }\n\n // For later aggregations\n selectorsPerRule.add(selectors)\n declarationsPerRule.add(declarations)\n }\n\n return {\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}\n\nexport {\n analyzeRules\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 generate from 'css-tree/generator'\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 }) => {\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 const parts = []\n\n walk(value, {\n reverse: true,\n enter: function (fontNode) {\n if (fontNode.type === 'String') {\n return parts.unshift(fontNode)\n }\n if (fontNode.type === 'Operator' && fontNode.value.charCodeAt(0) === COMMA) {\n return parts.unshift(fontNode)\n }\n if (fontNode.type === 'Identifier') {\n if (keywordDisallowList[fontNode.name]) {\n return this.skip\n }\n return parts.unshift(fontNode)\n }\n }\n })\n\n all.push(parts.map(generate).join(''))\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, {\n enter: 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\n if (size) {\n all.push(size)\n }\n }\n\n return all.count()\n}\n\nexport {\n analyzeFontSizes\n}","const analyzeDeclarations = ({ stringifyNode, declarations }) => {\n /** @type number */\n const total = declarations.length\n const cache = Object.create(null)\n let importants = 0\n let totalUnique = 0\n let totalInKeyframes = 0\n\n for (let i = 0; i < total; i++) {\n const declaration = declarations[i]\n\n if (declaration.important === true) {\n importants++\n\n if (declaration.inKeyframe) {\n totalInKeyframes++\n }\n }\n\n const stringified = stringifyNode(declaration)\n\n if (!cache[stringified]) {\n cache[stringified] = 1\n totalUnique++\n }\n }\n\n return {\n total,\n unique: {\n total: totalUnique,\n ratio: total === 0 ? 0 : totalUnique / total,\n },\n importants: {\n total: importants,\n ratio: total === 0 ? 0 : importants / total,\n inKeyframes: {\n total: totalInKeyframes,\n ratio: importants === 0 ? 0 : totalInKeyframes / importants,\n },\n },\n }\n}\n\nexport {\n analyzeDeclarations\n}","import { analyzeSpecificity, compareSpecificity } from './specificity.js'\nimport { AggregateCollection } from '../aggregate-collection.js'\nimport { CountableCollection } from '../countable-collection.js'\n\nconst analyzeSelectors = ({ stringifyNode, selectors }) => {\n const counts = Object.create(null)\n const cache = Object.create(null)\n /** @type number */\n const totalSelectors = selectors.length\n\n /** @type [number,number,number] */\n let maxSpecificity\n /** @type [number,number,number] */\n let minSpecificity\n let specificityA = new AggregateCollection(totalSelectors)\n let specificityB = new AggregateCollection(totalSelectors)\n let specificityC = new AggregateCollection(totalSelectors)\n let totalUnique = 0\n const complexityAggregator = new AggregateCollection(totalSelectors);\n\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 const keyframes = new CountableCollection()\n\n for (let i = 0; i < totalSelectors; i++) {\n /** @type import('css-tree').Selector */\n const node = selectors[i];\n /** @type string */\n const value = stringifyNode(node)\n\n if (node.isKeyframeSelector) {\n keyframes.push(value)\n // Do not attempt to further analyze <keyframe-selectors>\n continue\n }\n\n const { specificity, complexity, isId, isA11y } = cache[value] || analyzeSpecificity(node)\n\n if (isId) {\n ids.push(value)\n }\n\n if (isA11y) {\n a11y.push(value)\n }\n\n if (!cache[value]) {\n cache[value] = { complexity, specificity, isId, isA11y }\n totalUnique++\n counts[value] = 1\n } else {\n counts[value]++\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\n const aggregatesA = specificityA.aggregate()\n const aggregatesB = specificityB.aggregate()\n const aggregatesC = specificityC.aggregate()\n const complexityCount = new CountableCollection(complexities).count()\n\n return {\n total: totalSelectors,\n totalUnique,\n uniquenessRatio: totalSelectors === 0 ? 0 : totalUnique / totalSelectors,\n specificity: {\n /** @type [number, number, number] */\n min: minSpecificity,\n /** @type [number, number, number] */\n max: maxSpecificity,\n /** @type [number, number, number] */\n sum: [aggregatesA.sum, aggregatesB.sum, aggregatesC.sum],\n /** @type [number, number, number] */\n mean: [aggregatesA.mean, aggregatesB.mean, aggregatesC.mean],\n /** @type [number, number, number] */\n mode: [aggregatesA.mode, aggregatesB.mode, aggregatesC.mode],\n /** @type [number, number, number] */\n median: [aggregatesA.median, aggregatesB.median, aggregatesC.median],\n /** @type [number, number, number][] */\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: {\n ...keyframes.count(),\n ratio: totalSelectors === 0 ? 0 : keyframes.size() / totalSelectors,\n }\n }\n}\n\nexport {\n analyzeSelectors\n}","import { CountableCollection } from '../countable-collection.js'\n\nconst analyzeProperties = ({ properties }) => {\n const all = new CountableCollection(properties.map(p => p.authored))\n const prefixed = new CountableCollection()\n const hacks = new CountableCollection()\n const customs = new CountableCollection()\n const totalProperties = properties.length\n\n for (let i = 0; i < totalProperties; i++) {\n const property = properties[i]\n\n if (property.vendor) {\n prefixed.push(property.authored)\n continue\n }\n\n if (property.hack) {\n hacks.push(property.authored)\n continue\n }\n\n if (property.custom) {\n customs.push(property.authored)\n continue\n }\n }\n\n return {\n ...all.count(),\n prefixed: {\n ...prefixed.count(),\n ratio: prefixed.size() / totalProperties,\n },\n custom: {\n ...customs.count(),\n ratio: customs.size() / totalProperties,\n },\n browserhacks: {\n ...hacks.count(),\n ratio: hacks.size() / totalProperties,\n }\n }\n}\n\nexport {\n analyzeProperties\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`\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.map(stringifyNode))\n const allTimingFunctions = new CountableCollection(timingFunctions.map(stringifyNode))\n\n for (let index = 0; index < animations.length; index++) {\n const node = animations[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 node.value.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 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'\n\nconst analyzeAtRules = ({ atrules, stringifyNode }) => {\n /** @type {{[index: string]: string}[]} */\n const fontfaces = []\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.value),\n 'supports': node => supports.push(node.prelude.value),\n 'keyframes': node => keyframes.push(`@${node.name} ${node.prelude.value}`),\n 'import': node => imports.push(node.prelude.value),\n 'charset': node => charsets.push(node.prelude.value),\n 'container': node => containers.push(node.prelude.value),\n }\n\n for (let i = 0; i < atrules.length; i++) {\n /** @type {import('css-tree').Atrule} */\n const node = atrules[i]\n const nodeName = node.name\n const action = machine[nodeName]\n if (action) {\n action(node)\n continue\n }\n\n if (nodeName.endsWith('keyframes')) {\n const name = `@${nodeName} ${node.prelude.value}`\n keyframes.push(name)\n\n if (hasVendorPrefix(nodeName)) {\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: 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 }\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 this.contextCount = 0\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 this.contextCount++\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}","import parse from 'css-tree/parser'\nimport walk from 'css-tree/walker'\nimport { property as getProperty } from '../node_modules/css-tree/lib/utils/names.js'\nimport { compareSpecificity } from './selectors/specificity.js'\nimport { analyzeRules } from './rules/rules.js'\nimport { colorFunctions, colorNames } from './values/colors.js'\nimport { analyzeFontFamilies } from './values/font-families.js'\nimport { analyzeFontSizes } from './values/font-sizes.js'\nimport { analyzeDeclarations } from './declarations/declarations.js'\nimport { analyzeSelectors } from './selectors/selectors.js'\nimport { analyzeProperties } from './properties/properties.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'\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 const 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.push(line.substring(start.column - 1))\n continue\n }\n // Last line\n if (i === end.line) {\n value.push(line.substring(0, end.column - 1))\n continue\n }\n // All lines in between first and last\n value.push(line)\n }\n\n return value.join('\\n')\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 atrules = []\n const rules = []\n const selectors = []\n const declarations = []\n const properties = []\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 const embeds = new CountableCollection()\n\n walk(ast, {\n enter: function (node) {\n switch (node.type) {\n case 'Atrule': {\n atrules.push(node)\n break\n }\n case 'Rule': {\n rules.push(node)\n break\n }\n case 'Selector': {\n selectors.push({\n ...node,\n isKeyframeSelector: this.atrule && this.atrule.name.endsWith('keyframes')\n })\n\n // Avoid further 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 (node.value.startsWith('data:')) {\n embeds.push(node.value)\n }\n break\n }\n case 'Declaration': {\n declarations.push({\n ...node,\n inKeyframe: this.atrule && this.atrule.name.endsWith('keyframes')\n })\n\n const { value, property } = node\n const fullProperty = Object.assign({\n authored: property\n }, getProperty(property))\n\n properties.push(fullProperty)\n values.push(value)\n\n switch (fullProperty.basename) {\n case 'z-index': {\n zindex.push(value)\n break\n }\n case 'text-shadow': {\n textShadows.push(value)\n break\n }\n case 'box-shadow': {\n boxShadows.push(value)\n break\n }\n case 'font': {\n fontValues.push(value)\n break\n }\n case 'font-family': {\n fontFamilyValues.push(stringifyNode(value))\n // Prevent analyzer to find color names in this property\n return this.skip\n }\n case 'font-size': {\n fontSizeValues.push(stringifyNode(value))\n break\n }\n case 'transition':\n case 'animation': {\n animations.push(node)\n break\n }\n case 'animation-duration':\n case 'transition-duration': {\n durations.push(value)\n break\n }\n case 'transition-timing-function':\n case 'animation-timing-function': {\n timingFunctions.push(value)\n break\n }\n }\n\n walk(value, {\n enter: function (valueNode) {\n switch (valueNode.type) {\n case 'Hash': {\n colors.push(stringifyNode(valueNode), 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 })\n const embeddedContent = embeds.count()\n const embedSize = Object.keys(embeddedContent.unique).join('').length\n\n return {\n stylesheet: {\n sourceLinesOfCode: atrules.length + selectors.length + declarations.length,\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: analyzeRules({ rules }),\n selectors: analyzeSelectors({ stringifyNode, selectors }),\n declarations: analyzeDeclarations({ stringifyNode, declarations }),\n properties: analyzeProperties({ properties }),\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"],"names":["properties","Map","property","has","get","name","hack","custom","str","length","offset","charCodeAt","toLowerCase","descriptor","set","vendor","secondDashIndex","indexOf","substring","getVendorPrefix","prefix","substr","Object","freeze","basename","compareSpecificity","a","b","selectorListSpecificities","selectorListAst","childSelectors","walk","visit","enter","node","push","analyzeSpecificity","sort","specificity","ast","A","B","C","complexity","isA11y","selector","type","Boolean","value","startsWith","includes","skip","selectorList","parse","children","first","context","i","isId","AggregateCollection","size","this","items","Uint8Array","sum","cursor","add","item","aggregate","min","max","mean","mode","median","range","arr","middle","lowerMiddleRank","sorted","slice","frequencies","create","maxOccurrences","maxOccurenceCount","element","updatedCount","Mode","Math","floor","toArray","Array","from","subarray","analyzeRules","rules","totalRules","selectorsPerRule","declarationsPerRule","emptyRules","selectors","declarations","childNode","total","empty","ratio","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","totalUnique","index","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","all","fontFamilyValues","firstChild","parts","reverse","fontNode","unshift","map","generate","join","sizeKeywords","keywords","analyzeFontSizes","stringifyNode","fontSizeValues","operator","analyzeDeclarations","cache","importants","totalInKeyframes","declaration","important","inKeyframe","stringified","inKeyframes","analyzeSelectors","maxSpecificity","minSpecificity","counts","totalSelectors","specificityA","specificityB","specificityC","complexityAggregator","specificities","complexities","ids","a11y","keyframes","isKeyframeSelector","undefined","aggregatesA","aggregatesB","aggregatesC","complexityCount","id","accessibility","analyzeProperties","p","authored","prefixed","hacks","customs","totalProperties","browserhacks","auto","none","analyzeValues","values","timingKeywords","linear","ease","analyzeAnimations","animations","timingFunctions","allDurations","durations","allTimingFunctions","durationFound","forEach","child","hasVendorPrefix","keyword","isAstVendorPrefixed","analyzeVendorPrefixes","analyzeAtRules","atrules","fontfaces","imports","medias","charsets","supports","prefixedKeyframes","containers","machine","descriptors","block","media","prelude","import","charset","container","nodeName","action","endsWith","fontface","ContextCollection","list","contexts","contextCount","itemsPerContext","assign","css","start","Date","lines","split","loc","end","line","column","startParse","totalComments","commentsSize","parseAtrulePrelude","parseCustomProperty","positions","onComment","comment","startAnalysis","zindex","textShadows","boxShadows","colors","units","embeds","atrule","unit","fullProperty","getProperty","valueNode","embeddedContent","embedSize","keys","stylesheet","sourceLinesOfCode","linesOfCode","comments","fontFamilies","fontSizes","zindexes","prefixes","__meta__","parseTime","analyzeTime"],"mappings":"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"}
|
package/dist/analyzer.modern.js
CHANGED
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
import*as e from"css-tree";function t(){return t=Object.assign||function(e){for(var t=1;t<arguments.length;t++){var n=arguments[t];for(var i in n)Object.prototype.hasOwnProperty.call(n,i)&&(e[i]=n[i])}return e},t.apply(this,arguments)}function n(e,t){return e[0]===t[0]?e[1]===t[1]?t[2]-e[2]:t[1]-e[1]:t[0]-e[0]}function i(t){const i=[];return e.walk(t,{visit:"Selector",enter(e){i.push(s(e))}}),i.sort((e,t)=>n(e.specificity,t.specificity))}const s=t=>{let n=0,s=0,r=0,a=0,o=!1;return e.walk(t,{enter:function(t){switch(t.type){case"IdSelector":n++,a++;break;case"ClassSelector":s++,a++;break;case"AttributeSelector":s++,a++,Boolean(t.value)&&a++,o="role"===t.name.name||t.name.name.startsWith("aria-");break;case"PseudoElementSelector":case"TypeSelector":"*"!==t.name&&r++,a++;break;case"PseudoClassSelector":if(["before","after","first-letter","first-line"].includes(t.name))return r++,a++,this.skip;if(["is","has","matches"].includes(t.name)){const o=e.find(t,({type:e})=>"Raw"===e),l=i(e.parse(o.value,{context:"selectorList"})),[u,c,h]=l[0].specificity;n+=u,s+=c,r+=h;for(let e=0;e<l.length;e++)a+=l[e].complexity;return void a++}if("not"===t.name){const e=i(t),[o,l,u]=e[0].specificity;n+=o,s+=l,r+=u;for(let t=0;t<e.length;t++)a+=e[t].complexity;return a++,this.skip}if(["nth-child","nth-last-child"].includes(t.name)){s++;const e=i(t);if(0===e.length)return;const[o,l,u]=e[0].specificity;n+=o,s+=l,r+=u;for(let t=0;t<e.length;t++)a+=e[t].complexity;return void a++}if("where"===t.name){const n=e.find(t,({type:e})=>"Raw"===e),s=i(e.parse(n.value,{context:"selectorList"}));for(let e=0;e<s.length;e++)a+=s[e].complexity;return void a++}a++,s++;break;case"Combinator":a++}}}),{specificity:[n,s,r],complexity:a,isId:n>0,isA11y:o}};class r{constructor(e){this.items=new Uint8Array(e),this.sum=0,this.cursor=0}add(e){this.items[this.cursor]=e,this.sum+=e,this.cursor++}aggregate(){if(0===this.cursor)return{min:0,max:0,mean:0,mode:0,median:0,range:0,sum:0};const e=new Uint8Array(this.items.slice(0,this.cursor)).sort((e,t)=>e-t),t=e[0],n=e[e.length-1],i=function(e){const t=Object.create(null);let n=-1,i=0,s=0;for(let r=0;r<e.length;r++){const a=e[r],o=(t[a]||0)+1;t[a]=o,o>n&&(n=o,i=0,s=0),o>=n&&(i++,s+=a)}return s/i}(e),s=function(e){const t=e.length/2,n=Math.floor(t);return t!==n?e[n]:(e[n]+e[n-1])/2}(e);return{min:t,max:n,mean:this.sum/this.cursor,mode:i,median:s,range:n-t,sum:this.sum}}toArray(){return Array.from(this.items.subarray(0,this.cursor))}}const a=({rules:n})=>{const i=n.length,s=new r(i),a=new r(i);let o=0;for(let t=0;t<i;t++){let i=0,r=0;e.walk(n[t],{enter:function(e){return"Selector"===e.type?(i++,this.skip):"Declaration"===e.type?(r++,this.skip):void 0}}),0===r&&o++,s.add(i),a.add(r)}return{total:i,empty:{total:o,ratio:0===i?0:o/i},selectors:t({},s.aggregate(),{items:s.toArray()}),declarations:t({},a.aggregate(),{items:a.toArray()})}},o={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},l={rgb:1,rgba:1,hsl:1,hsla:1,hwb:1,lab:1,lch:1,oklab:1,oklch:1,color:1};class u{constructor(e){if(this.items={},this.total=0,this.totalUnique=0,e)for(let t=0;t<e.length;t++)this.push(e[t])}push(e){this.total++,this.items[e]?this.items[e]++:(this.items[e]=1,this.totalUnique++)}size(){return this.total}count(){return{total:this.total,totalUnique:this.totalUnique,unique:this.items,uniquenessRatio:0===this.total?0:this.totalUnique/this.total}}}const c={inherit:1,initial:1,unset:1,revert:1,caption:1,icon:1,menu:1,"message-box":1,"small-caption":1,"status-bar":1},h={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},d=({fontValues:t,fontFamilyValues:n})=>{const i=new u(n);for(let n=0;n<t.length;n++){const s=t[n],r=s.children.first;if("Identifier"===r.type&&c[r.name])continue;const a=[];e.walk(s,{reverse:!0,enter:function(e){return"String"===e.type||"Operator"===e.type&&44===e.value.charCodeAt(0)?a.unshift(e):"Identifier"===e.type?h[e.name]?this.skip:a.unshift(e):void 0}}),i.push(a.map(e.generate).join(""))}return i.count()},m={"xx-small":1,"x-small":1,small:1,medium:1,large:1,"x-large":1,"xx-large":1,larger:1,smaller:1},p={inherit:1,initial:1,unset:1,revert:1,caption:1,icon:1,menu:1,"message-box":1,"small-caption":1,"status-bar":1},g=({stringifyNode:t,fontSizeValues:n,fontValues:i})=>{const s=new u(n);for(let n=0;n<i.length;n++){const r=i[n],a=r.children.first;if("Identifier"===a.type&&p[a.name])continue;let o,l=!1;e.walk(r,{enter:function(e){switch(e.type){case"Number":if(48===e.value.charCodeAt(0))return o="0",this.break;case"Operator":47===e.value.charCodeAt(0)&&(l=!0);break;case"Dimension":if(!l)return o=t(e),this.break;case"Identifier":if(m[e.name])return o=e.name,this.break}}}),o&&s.push(o)}return s.count()},f=({stringifyNode:e,declarations:t})=>{const n=t.length,i=Object.create(null);let s=0,r=0,a=0;for(let o=0;o<n;o++){const n=t[o];!0===n.important&&(s++,n.inKeyframe&&a++);const l=e(n);i[l]||(i[l]=1,r++)}return{total:n,unique:{total:r,ratio:0===n?0:r/n},importants:{total:s,ratio:0===n?0:s/n,inKeyframes:{total:a,ratio:0===s?0:a/s}}}},y=({stringifyNode:e,selectors:i})=>{const a=Object.create(null),o=Object.create(null),l=i.length;let c,h,d=new r(l),m=new r(l),p=new r(l),g=0;const f=new r(l),y=[],b=[],k=new u,w=new u,v=new u;for(let t=0;t<l;t++){const r=i[t],l=e(r);if(r.isKeyframeSelector){v.push(l);continue}const{specificity:u,complexity:x,isId:q,isA11y:z}=o[l]||s(r);q&&k.push(l),z&&w.push(l),o[l]?a[l]++:(o[l]={complexity:x,specificity:u,isId:q,isA11y:z},g++,a[l]=1),f.add(x),void 0===c&&(c=u),void 0===h&&(h=u),d.add(u[0]),m.add(u[1]),p.add(u[2]),void 0!==h&&n(h,u)<0&&(h=u),void 0!==c&&n(c,u)>0&&(c=u),y.push(u),b.push(x)}const x=d.aggregate(),q=m.aggregate(),z=p.aggregate(),A=new u(b).count();return{total:l,totalUnique:g,uniquenessRatio:0===l?0:g/l,specificity:{min:h,max:c,sum:[x.sum,q.sum,z.sum],mean:[x.mean,q.mean,z.mean],mode:[x.mode,q.mode,z.mode],median:[x.median,q.median,z.median],items:y},complexity:t({},f.aggregate(),A,{items:b}),id:t({},k.count(),{ratio:0===l?0:k.size()/l}),accessibility:t({},w.count(),{ratio:0===l?0:w.size()/l}),keyframes:t({},v.count(),{ratio:0===l?0:v.size()/l})}},b=({properties:e})=>{const n=new u(e.map(e=>e.authored)),i=new u,s=new u,r=new u,a=e.length;for(let t=0;t<a;t++){const n=e[t];n.vendor?i.push(n.authored):n.hack?s.push(n.authored):n.custom&&r.push(n.authored)}return t({},n.count(),{prefixed:t({},i.count(),{ratio:i.size()/a}),custom:t({},r.count(),{ratio:r.size()/a}),browserhacks:t({},s.count(),{ratio:s.size()/a})})},k={auto:1,inherit:1,initial:1,unset:1,revert:1,none:1},w=({values:e,stringifyNode:t})=>{const n=new u;for(let i=0;i<e.length;i++){const s=e[i],r=s.children.first;r&&("Identifier"===r.type&&k[r.name]||n.push(t(s)))}return n.count()},v={linear:1,ease:1,"ease-in":1,"ease-out":1,"ease-in-out":1,"step-start":1,"step-end":1},x=({animations:e,durations:t,timingFunctions:n,stringifyNode:i})=>{const s=new u(t.map(i)),r=new u(n.map(i));for(let t=0;t<e.length;t++){let n=!1;e[t].value.children.forEach(e=>"Operator"===e.type?n=!1:"Dimension"===e.type&&!1===n?(n=!0,s.push(i(e))):"Identifier"===e.type&&v[e.name]?r.push(i(e)):"Function"!==e.type||"cubic-bezier"!==e.name&&"steps"!==e.name?void 0:r.push(i(e)))}return{durations:s.count(),timingFunctions:r.count()}};function q(e){return 45===e.charCodeAt(0)&&45!==e.charCodeAt(1)&&-1!==e.indexOf("-",2)}function z(e){e=e.toArray();for(let t=0;t<e.length;t++){const n=e[t];if("Identifier"===n.type&&n.name.length>=3&&q(n.name))return!0;if("Function"===n.type){if(q(n.name))return!0;if(n.children&&z(n.children))return!0}}return!1}const A=({values:e,stringifyNode:t})=>{const n=new u;for(let i=0;i<e.length;i++){const s=e[i];s.children&&z(s.children)&&n.push(t(s))}return n.count()},C=({atrules:e,stringifyNode:n})=>{const i=[],s=new u,r=new u,a=new u,o=new u,l=new u,c=new u,h=new u,d={"font-face":e=>{const t={};e.block.children.forEach(e=>t[e.property]=n(e.value)),i.push(t)},media:e=>r.push(e.prelude.value),supports:e=>o.push(e.prelude.value),keyframes:e=>l.push(`@${e.name} ${e.prelude.value}`),import:e=>s.push(e.prelude.value),charset:e=>a.push(e.prelude.value),container:e=>h.push(e.prelude.value)};for(let t=0;t<e.length;t++){const n=e[t],i=n.name,s=d[i];if(s)s(n);else if(i.endsWith("keyframes")){const e=`@${i} ${n.prelude.value}`;l.push(e),q(i)&&c.push(e)}}return{fontface:{total:i.length,totalUnique:i.length,unique:i,uniquenessRatio:1},import:s.count(),media:r.count(),charset:a.count(),supports:o.count(),keyframes:t({},l.count(),{prefixed:t({},c.count(),{ratio:0===l.size()?0:c.size()/l.size()})}),container:h.count()}};class N{constructor(){this.list=new u,this.contexts={},this.contextCount=0}push(e,t){this.list.push(e),this.contexts[t]||(this.contexts[t]=new u,this.contextCount++),this.contexts[t].push(e)}count(){const e={};for(let t in this.contexts)e[t]=this.contexts[t].count();return Object.assign(this.list.count(),{itemsPerContext:e})}}const S=n=>{const i=new Date,s=n.split(/\r?\n/);function r(e){const t=e.loc.start,n=e.loc.end;if(0==n.line-t.line)return s[t.line-1].substring(t.column-1,n.column-1);const i=[];for(let e=t.line;e<=n.line;e++){const r=s[e-1];i.push(e!==t.line?e!==n.line?r:r.substring(0,n.column-1):r.substring(t.column-1))}return i.join("\n")}const c=new Date;let h=0,m=0;const p=e.parse(n,{parseAtrulePrelude:!1,parseCustomProperty:!0,positions:!0,onComment:function(e){h++,m+=e.length}}),k=new Date,v=[],q=[],z=[],S=[],O=[],j=[],I=[],D=[],U=[],F=[],V=[],P=[],R=[],K=[],L=[],W=new N,$=new N,_=new u;e.walk(p,{enter:function(n){switch(n.type){case"Atrule":v.push(n);break;case"Rule":q.push(n);break;case"Selector":return z.push(t({},n,{isKeyframeSelector:this.atrule&&this.atrule.name.endsWith("keyframes")})),this.skip;case"Dimension":if(!this.declaration)break;return $.push(n.unit,this.declaration.property),this.skip;case"Url":n.value.startsWith("data:")&&_.push(n.value);break;case"Declaration":{S.push(t({},n,{inKeyframe:this.atrule&&this.atrule.name.endsWith("keyframes")}));const{value:i,property:s}=n,a=t({authored:s},e.property(s));switch(O.push(a),j.push(i),a.basename){case"z-index":I.push(i);break;case"text-shadow":D.push(i);break;case"box-shadow":U.push(i);break;case"font":F.push(i);break;case"font-family":return V.push(r(i)),this.skip;case"font-size":P.push(r(i));break;case"transition":case"animation":R.push(n);break;case"animation-duration":case"transition-duration":L.push(i);break;case"transition-timing-function":case"animation-timing-function":K.push(i)}e.walk(n.value,{enter:function(e){switch(e.type){case"Hash":return W.push(r(e),s),this.skip;case"Identifier":{const{name:t}=e;return t.length>20||t.length<3||o[t.toLowerCase()]&&W.push(r(e),s),this.skip}case"Function":l[e.name.toLowerCase()]&&W.push(r(e),s)}}})}}}});const E=_.count(),T=Object.keys(E.unique).join("").length;return{stylesheet:{sourceLinesOfCode:v.length+z.length+S.length,linesOfCode:s.length,size:n.length,comments:{total:h,size:m},embeddedContent:Object.assign(E,{size:{total:T,ratio:0===n.length?0:T/n.length}})},atrules:C({atrules:v,stringifyNode:r}),rules:a({rules:q}),selectors:y({stringifyNode:r,selectors:z}),declarations:f({stringifyNode:r,declarations:S}),properties:b({properties:O}),values:{colors:W.count(),fontFamilies:d({stringifyNode:r,fontValues:F,fontFamilyValues:V}),fontSizes:g({stringifyNode:r,fontValues:F,fontSizeValues:P}),zindexes:w({values:I,stringifyNode:r}),textShadows:w({values:D,stringifyNode:r}),boxShadows:w({values:U,stringifyNode:r}),animations:x({animations:R,timingFunctions:K,durations:L,stringifyNode:r}),prefixes:A({values:j,stringifyNode:r}),units:$.count()},__meta__:{parseTime:k-c,analyzeTime:new Date-k,total:new Date-i}}};export{S as analyze,n as compareSpecificity};
|
|
1
|
+
import e from"css-tree/parser";import t from"css-tree/walker";import n from"css-tree/generator";function i(){return i=Object.assign||function(e){for(var t=1;t<arguments.length;t++){var n=arguments[t];for(var i in n)Object.prototype.hasOwnProperty.call(n,i)&&(e[i]=n[i])}return e},i.apply(this,arguments)}const s=new Map,r=function(e){if(s.has(e))return s.get(e);let t=e,n=e[0];"/"===n?n="/"===e[1]?"//":"/":"_"!==n&&"*"!==n&&"$"!==n&&"#"!==n&&"+"!==n&&"&"!==n&&(n="");const i=(r=t).length-(a=(a=n.length)||0)>=2&&45===r.charCodeAt(a)&&45===r.charCodeAt(a+1);var r,a;if(!i&&(t=t.toLowerCase(),s.has(t))){const n=s.get(t);return s.set(e,n),n}const o=i?"":function(e,t){if(e.length-(t=t||0)>=3&&45===e.charCodeAt(t)&&45!==e.charCodeAt(t+1)){const n=e.indexOf("-",t+2);if(-1!==n)return e.substring(t,n+1)}return""}(t,n.length),l=t.substr(0,n.length+o.length),u=Object.freeze({basename:t.substr(l.length),name:t.substr(n.length),hack:n,vendor:o,prefix:l,custom:i});return s.set(e,u),u};function a(e,t){return e[0]===t[0]?e[1]===t[1]?t[2]-e[2]:t[1]-e[1]:t[0]-e[0]}function o(e){const n=[];return t(e,{visit:"Selector",enter(e){n.push(l(e))}}),n.sort((e,t)=>a(e.specificity,t.specificity))}const l=n=>{let i=0,s=0,r=0,a=0,l=!1;return t(n,{enter:function(t){switch(t.type){case"IdSelector":i++,a++;break;case"ClassSelector":s++,a++;break;case"AttributeSelector":s++,a++,Boolean(t.value)&&a++,l="role"===t.name.name||t.name.name.startsWith("aria-");break;case"PseudoElementSelector":case"TypeSelector":"*"!==t.name&&r++,a++;break;case"PseudoClassSelector":if(["before","after","first-letter","first-line"].includes(t.name))return r++,a++,this.skip;if(["is","has","matches","-webkit-any","-moz-any"].includes(t.name)){const n=o(e(t.children.first.value,{context:"selectorList"})),[l,u,c]=n[0].specificity;i+=l,s+=u,r+=c;for(let e=0;e<n.length;e++)a+=n[e].complexity;return void a++}if("not"===t.name){const e=o(t),[n,l,u]=e[0].specificity;i+=n,s+=l,r+=u;for(let t=0;t<e.length;t++)a+=e[t].complexity;return a++,this.skip}if(["nth-child","nth-last-child"].includes(t.name)){s++;const e=o(t);if(0===e.length)return;const[n,l,u]=e[0].specificity;i+=n,s+=l,r+=u;for(let t=0;t<e.length;t++)a+=e[t].complexity;return void a++}if("where"===t.name){const n=o(e(t.children.first.value,{context:"selectorList"}));for(let e=0;e<n.length;e++)a+=n[e].complexity;return void a++}a++,s++;break;case"Combinator":a++}}}),{specificity:[i,s,r],complexity:a,isId:i>0,isA11y:l}};class u{constructor(e){this.items=new Uint8Array(e),this.sum=0,this.cursor=0}add(e){this.items[this.cursor]=e,this.sum+=e,this.cursor++}aggregate(){if(0===this.cursor)return{min:0,max:0,mean:0,mode:0,median:0,range:0,sum:0};const e=new Uint8Array(this.items.slice(0,this.cursor)).sort((e,t)=>e-t),t=e[0],n=e[e.length-1],i=function(e){const t=Object.create(null);let n=-1,i=0,s=0;for(let r=0;r<e.length;r++){const a=e[r],o=(t[a]||0)+1;t[a]=o,o>n&&(n=o,i=0,s=0),o>=n&&(i++,s+=a)}return s/i}(e),s=function(e){const t=e.length/2,n=Math.floor(t);return t!==n?e[n]:(e[n]+e[n-1])/2}(e);return{min:t,max:n,mean:this.sum/this.cursor,mode:i,median:s,range:n-t,sum:this.sum}}toArray(){return Array.from(this.items.subarray(0,this.cursor))}}const c=({rules:e})=>{const n=e.length,s=new u(n),r=new u(n);let a=0;for(let i=0;i<n;i++){let n=0,o=0;t(e[i],{enter:function(e){return"Selector"===e.type?(n++,this.skip):"Declaration"===e.type?(o++,this.skip):void 0}}),0===o&&a++,s.add(n),r.add(o)}return{total:n,empty:{total:a,ratio:0===n?0:a/n},selectors:i({},s.aggregate(),{items:s.toArray()}),declarations:i({},r.aggregate(),{items:r.toArray()})}},h={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};class m{constructor(e){if(this.items={},this.total=0,this.totalUnique=0,e)for(let t=0;t<e.length;t++)this.push(e[t])}push(e){this.total++,this.items[e]?this.items[e]++:(this.items[e]=1,this.totalUnique++)}size(){return this.total}count(){return{total:this.total,totalUnique:this.totalUnique,unique:this.items,uniquenessRatio:0===this.total?0:this.totalUnique/this.total}}}const p={inherit:1,initial:1,unset:1,revert:1,caption:1,icon:1,menu:1,"message-box":1,"small-caption":1,"status-bar":1},g={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},f=({fontValues:e,fontFamilyValues:i})=>{const s=new m(i);for(let i=0;i<e.length;i++){const r=e[i],a=r.children.first;if("Identifier"===a.type&&p[a.name])continue;const o=[];t(r,{reverse:!0,enter:function(e){return"String"===e.type||"Operator"===e.type&&44===e.value.charCodeAt(0)?o.unshift(e):"Identifier"===e.type?g[e.name]?this.skip:o.unshift(e):void 0}}),s.push(o.map(n).join(""))}return s.count()},y={"xx-small":1,"x-small":1,small:1,medium:1,large:1,"x-large":1,"xx-large":1,larger:1,smaller:1},b={inherit:1,initial:1,unset:1,revert:1,caption:1,icon:1,menu:1,"message-box":1,"small-caption":1,"status-bar":1},k=({stringifyNode:e,fontSizeValues:n,fontValues:i})=>{const s=new m(n);for(let n=0;n<i.length;n++){const r=i[n],a=r.children.first;if("Identifier"===a.type&&b[a.name])continue;let o,l=!1;t(r,{enter:function(t){switch(t.type){case"Number":if(48===t.value.charCodeAt(0))return o="0",this.break;case"Operator":47===t.value.charCodeAt(0)&&(l=!0);break;case"Dimension":if(!l)return o=e(t),this.break;case"Identifier":if(y[t.name])return o=t.name,this.break}}}),o&&s.push(o)}return s.count()},w=({stringifyNode:e,declarations:t})=>{const n=t.length,i=Object.create(null);let s=0,r=0,a=0;for(let o=0;o<n;o++){const n=t[o];!0===n.important&&(s++,n.inKeyframe&&a++);const l=e(n);i[l]||(i[l]=1,r++)}return{total:n,unique:{total:r,ratio:0===n?0:r/n},importants:{total:s,ratio:0===n?0:s/n,inKeyframes:{total:a,ratio:0===s?0:a/s}}}},x=({stringifyNode:e,selectors:t})=>{const n=Object.create(null),s=Object.create(null),r=t.length;let o,c,h=new u(r),d=new u(r),p=new u(r),g=0;const f=new u(r),y=[],b=[],k=new m,w=new m,x=new m;for(let i=0;i<r;i++){const r=t[i],u=e(r);if(r.isKeyframeSelector){x.push(u);continue}const{specificity:m,complexity:v,isId:q,isA11y:z}=s[u]||l(r);q&&k.push(u),z&&w.push(u),s[u]?n[u]++:(s[u]={complexity:v,specificity:m,isId:q,isA11y:z},g++,n[u]=1),f.add(v),void 0===o&&(o=m),void 0===c&&(c=m),h.add(m[0]),d.add(m[1]),p.add(m[2]),void 0!==c&&a(c,m)<0&&(c=m),void 0!==o&&a(o,m)>0&&(o=m),y.push(m),b.push(v)}const v=h.aggregate(),q=d.aggregate(),z=p.aggregate(),C=new m(b).count();return{total:r,totalUnique:g,uniquenessRatio:0===r?0:g/r,specificity:{min:c,max:o,sum:[v.sum,q.sum,z.sum],mean:[v.mean,q.mean,z.mean],mode:[v.mode,q.mode,z.mode],median:[v.median,q.median,z.median],items:y},complexity:i({},f.aggregate(),C,{items:b}),id:i({},k.count(),{ratio:0===r?0:k.size()/r}),accessibility:i({},w.count(),{ratio:0===r?0:w.size()/r}),keyframes:i({},x.count(),{ratio:0===r?0:x.size()/r})}},v=({properties:e})=>{const t=new m(e.map(e=>e.authored)),n=new m,s=new m,r=new m,a=e.length;for(let t=0;t<a;t++){const i=e[t];i.vendor?n.push(i.authored):i.hack?s.push(i.authored):i.custom&&r.push(i.authored)}return i({},t.count(),{prefixed:i({},n.count(),{ratio:n.size()/a}),custom:i({},r.count(),{ratio:r.size()/a}),browserhacks:i({},s.count(),{ratio:s.size()/a})})},q={auto:1,inherit:1,initial:1,unset:1,revert:1,none:1},z=({values:e,stringifyNode:t})=>{const n=new m;for(let i=0;i<e.length;i++){const s=e[i],r=s.children.first;r&&("Identifier"===r.type&&q[r.name]||n.push(t(s)))}return n.count()},C={linear:1,ease:1,"ease-in":1,"ease-out":1,"ease-in-out":1,"step-start":1,"step-end":1},A=({animations:e,durations:t,timingFunctions:n,stringifyNode:i})=>{const s=new m(t.map(i)),r=new m(n.map(i));for(let t=0;t<e.length;t++){let n=!1;e[t].value.children.forEach(e=>"Operator"===e.type?n=!1:"Dimension"===e.type&&!1===n?(n=!0,s.push(i(e))):"Identifier"===e.type&&C[e.name]?r.push(i(e)):"Function"!==e.type||"cubic-bezier"!==e.name&&"steps"!==e.name?void 0:r.push(i(e)))}return{durations:s.count(),timingFunctions:r.count()}};function O(e){return 45===e.charCodeAt(0)&&45!==e.charCodeAt(1)&&-1!==e.indexOf("-",2)}function N(e){e=e.toArray();for(let t=0;t<e.length;t++){const n=e[t];if("Identifier"===n.type&&n.name.length>=3&&O(n.name))return!0;if("Function"===n.type){if(O(n.name))return!0;if(n.children&&N(n.children))return!0}}return!1}const S=({values:e,stringifyNode:t})=>{const n=new m;for(let i=0;i<e.length;i++){const s=e[i];s.children&&N(s.children)&&n.push(t(s))}return n.count()},j=({atrules:e,stringifyNode:t})=>{const n=[],s=new m,r=new m,a=new m,o=new m,l=new m,u=new m,c=new m,h={"font-face":e=>{const i={};e.block.children.forEach(e=>i[e.property]=t(e.value)),n.push(i)},media:e=>r.push(e.prelude.value),supports:e=>o.push(e.prelude.value),keyframes:e=>l.push(`@${e.name} ${e.prelude.value}`),import:e=>s.push(e.prelude.value),charset:e=>a.push(e.prelude.value),container:e=>c.push(e.prelude.value)};for(let t=0;t<e.length;t++){const n=e[t],i=n.name,s=h[i];if(s)s(n);else if(i.endsWith("keyframes")){const e=`@${i} ${n.prelude.value}`;l.push(e),O(i)&&u.push(e)}}return{fontface:{total:n.length,totalUnique:n.length,unique:n,uniquenessRatio:1},import:s.count(),media:r.count(),charset:a.count(),supports:o.count(),keyframes:i({},l.count(),{prefixed:i({},u.count(),{ratio:0===l.size()?0:u.size()/l.size()})}),container:c.count()}};class I{constructor(){this.list=new m,this.contexts={},this.contextCount=0}push(e,t){this.list.push(e),this.contexts[t]||(this.contexts[t]=new m,this.contextCount++),this.contexts[t].push(e)}count(){const e={};for(let t in this.contexts)e[t]=this.contexts[t].count();return Object.assign(this.list.count(),{itemsPerContext:e})}}const D=n=>{const s=new Date,a=n.split(/\r?\n/);function o(e){const t=e.loc.start,n=e.loc.end;if(0==n.line-t.line)return a[t.line-1].substring(t.column-1,n.column-1);const i=[];for(let e=t.line;e<=n.line;e++){const s=a[e-1];i.push(e!==t.line?e!==n.line?s:s.substring(0,n.column-1):s.substring(t.column-1))}return i.join("\n")}const l=new Date;let u=0,p=0;const g=e(n,{parseAtrulePrelude:!1,parseCustomProperty:!0,positions:!0,onComment:function(e){u++,p+=e.length}}),y=new Date,b=[],q=[],C=[],O=[],N=[],D=[],U=[],F=[],V=[],L=[],P=[],K=[],W=[],$=[],_=[],R=new I,E=new I,T=new m;t(g,{enter:function(e){switch(e.type){case"Atrule":b.push(e);break;case"Rule":q.push(e);break;case"Selector":return C.push(i({},e,{isKeyframeSelector:this.atrule&&this.atrule.name.endsWith("keyframes")})),this.skip;case"Dimension":if(!this.declaration)break;return E.push(e.unit,this.declaration.property),this.skip;case"Url":e.value.startsWith("data:")&&T.push(e.value);break;case"Declaration":{O.push(i({},e,{inKeyframe:this.atrule&&this.atrule.name.endsWith("keyframes")}));const{value:n,property:s}=e,a=Object.assign({authored:s},r(s));switch(N.push(a),D.push(n),a.basename){case"z-index":U.push(n);break;case"text-shadow":F.push(n);break;case"box-shadow":V.push(n);break;case"font":L.push(n);break;case"font-family":return P.push(o(n)),this.skip;case"font-size":K.push(o(n));break;case"transition":case"animation":W.push(e);break;case"animation-duration":case"transition-duration":_.push(n);break;case"transition-timing-function":case"animation-timing-function":$.push(n)}t(n,{enter:function(e){switch(e.type){case"Hash":return R.push(o(e),s),this.skip;case"Identifier":{const{name:t}=e;return t.length>20||t.length<3||h[t.toLowerCase()]&&R.push(o(e),s),this.skip}case"Function":d[e.name.toLowerCase()]&&R.push(o(e),s)}}})}}}});const M=T.count(),B=Object.keys(M.unique).join("").length;return{stylesheet:{sourceLinesOfCode:b.length+C.length+O.length,linesOfCode:a.length,size:n.length,comments:{total:u,size:p},embeddedContent:Object.assign(M,{size:{total:B,ratio:0===n.length?0:B/n.length}})},atrules:j({atrules:b,stringifyNode:o}),rules:c({rules:q}),selectors:x({stringifyNode:o,selectors:C}),declarations:w({stringifyNode:o,declarations:O}),properties:v({properties:N}),values:{colors:R.count(),fontFamilies:f({stringifyNode:o,fontValues:L,fontFamilyValues:P}),fontSizes:k({stringifyNode:o,fontValues:L,fontSizeValues:K}),zindexes:z({values:U,stringifyNode:o}),textShadows:z({values:F,stringifyNode:o}),boxShadows:z({values:V,stringifyNode:o}),animations:A({animations:W,timingFunctions:$,durations:_,stringifyNode:o}),prefixes:S({values:D,stringifyNode:o}),units:E.count()},__meta__:{parseTime:y-l,analyzeTime:new Date-y,total:new Date-s}}};export{D as analyze,a as compareSpecificity};
|
|
2
2
|
//# sourceMappingURL=analyzer.modern.js.map
|