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