all files / src/optimizations/MergeDeclarations/ DeclarationMapper.js

95.79% Statements 91/95
81.25% Branches 13/16
100% Functions 6/6
95.74% Lines 90/94
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168              34× 34× 34× 679× 679× 511× 679× 511× 679×   34× 34× 34× 34× 34× 34× 151×   151× 151× 185×   185×   151× 169× 169× 169× 169×   169×                           169× 169× 255×   169×       34× 34× 34× 34× 169×   169× 169× 255× 255× 255× 255× 255×       83× 83× 83× 83× 224× 224× 224×     224× 224× 224× 224×   83×       172× 172× 172×     172× 172× 172×             34×         396×                         396×     255× 396×   255×     396× 650× 650× 114× 114× 114×     536×         1022×   1022× 511× 511×   1283×    
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
const util_1 = require("@opticss/util");
const propParser = require("css-property-parser");
const specificity = require("specificity");
const typescript_collections_1 = require("typescript-collections");
const Match_1 = require("../../Match");
const cssIntrospection_1 = require("../../util/cssIntrospection");
const shorthandProperties_1 = require("../../util/shorthandProperties");
const OptimizationContext_1 = require("./OptimizationContext");
/**
 * Efficient navigation of the selectors and declarations of a stylesheet
 * and how they intersect at elements.
 */
class DeclarationMapper {
    constructor(pass, analyses, files) {
        this.declarationInfos = new util_1.MultiMap();
        this.contexts = new OptimizationContext_1.OptimizationContexts();
        this.selectorTree = new typescript_collections_1.BSTree((s1, s2) => {
            let cmp = specificity.compare(s1.specificity.specificityArray, s2.specificity.specificityArray);
            if (cmp === 0)
                cmp = compare(s1.fileIndex, s2.fileIndex);
            if (cmp === 0)
                cmp = compare(s1.sourceIndex, s2.sourceIndex);
            return cmp;
        });
        let declSourceIndex = 0;
        let declSourceIndexMap = new Map();
        this.elementDeclarations = new Map();
        files.forEach((file, fileIndex) => {
            let sourceIndex = 0;
            cssIntrospection_1.walkRules(file.content.root, (rule, scope) => {
                let selectors = pass.cache.getParsedSelectors(rule);
                /** all the declarations of this rule after expanding longhand properties. */
                let declarations = new typescript_collections_1.MultiDictionary(undefined, undefined, true);
                rule.walkDecls(decl => {
                    declSourceIndexMap.set(decl, declSourceIndex++);
                    // TODO: normalize values. E.g colors of different formats, etc.
                    declarations.setValue(decl.prop, [decl.value, decl.important, decl]);
                });
                selectors.forEach(selector => {
                    sourceIndex++;
                    let elements = new Array();
                    analyses.forEach(analysis => {
                        elements.splice(elements.length, 0, ...querySelector(analysis, selector));
                    });
                    let selectorInfo = {
                        rule,
                        container: rule.parent,
                        scope,
                        selector,
                        specificity: specificity.calculate(selector.toString())[0],
                        file,
                        fileIndex,
                        sourceIndex,
                        elements,
                        ordinal: -1,
                        declarations,
                        declarationInfos: new typescript_collections_1.MultiDictionary(),
                    };
                    let context = this.contexts.getContext(selectorInfo.rule.root(), selectorInfo.scope, selectorInfo.selector.toContext());
                    for (let prop of declarations.keys()) {
                        context.authoredProps.add(prop);
                    }
                    this.selectorTree.add(selectorInfo);
                });
            });
        });
        let selectorOrdinal = 1;
        let declarationOrdinal = 1;
        let importantDeclInfos = new Array();
        this.selectorTree.inorderTraversal((selectorInfo) => {
            selectorInfo.ordinal = selectorOrdinal++;
            // map properties to selector info
            let context = this.contexts.getContext(selectorInfo.rule.root(), selectorInfo.scope, selectorInfo.selector.toContext());
            for (let prop of selectorInfo.declarations.keys()) {
                let values = selectorInfo.declarations.getValue(prop);
                for (let value of values) {
                    declarationOrdinal++;
                    let [v, important, decl] = value;
                    if (propParser.isShorthandProperty(decl.prop)) {
                        // We only expand shorthand declarations as much as is necessary within the current optimization context
                        // but we need to track declarations globally and against elements according to the fully expanded
                        // values because selectors can conflict across different optimization contexts.
                        let longhandDeclarations = shorthandProperties_1.expandIfNecessary(context.authoredProps, decl.prop, decl.value, pass.actions);
                        let longHandProps = Object.keys(longhandDeclarations);
                        let longHandDeclInfos = new Array();
                        for (let longHandProp of longHandProps) {
                            let declInfo = this.makeDeclInfo(selectorInfo, longHandProp, longhandDeclarations[longHandProp], important, decl, declSourceIndexMap.get(decl), declarationOrdinal);
                            longHandDeclInfos.push(declInfo);
                            Iif (important) {
                                importantDeclInfos.push(declInfo);
                            }
                            selectorInfo.declarationInfos.setValue([prop, v], declInfo);
                            let valueInfo = context.getDeclarationValues(longHandProp);
                            valueInfo.setValue(longhandDeclarations[longHandProp], declInfo);
                            this.addDeclInfoToElements(selectorInfo.elements, longHandProp, declInfo);
                        }
                        this.trackDeclarationInfo(context, longHandDeclInfos);
                    }
                    else {
                        // normal long hand props are just set directly
                        let declInfo = this.makeDeclInfo(selectorInfo, prop, v, important, decl, declSourceIndexMap.get(decl), declarationOrdinal);
                        this.trackDeclarationInfo(context, [declInfo]);
                        Iif (important) {
                            importantDeclInfos.push(declInfo);
                        }
                        let valueInfo = context.getDeclarationValues(prop);
                        valueInfo.setValue(v, declInfo);
                        this.addDeclInfoToElements(selectorInfo.elements, prop, declInfo);
                    }
                }
            }
        });
        // we add the max declaration ordinal to all the important declaration infos
        // this makes those declarations resolve higher than all the non-important values.
        for (let declInfo of importantDeclInfos) {
            declInfo.ordinal = declInfo.ordinal + declarationOrdinal;
        }
    }
    makeDeclInfo(selectorInfo, prop, value, important, decl, sourceOrdinal, ordinal, dupeCount = 0) {
        let declInfo = {
            decl,
            prop,
            value,
            important,
            selectorInfo,
            sourceOrdinal,
            originalSourceOrdinal: sourceOrdinal,
            ordinal,
            originalOrdinal: ordinal,
            dupeCount,
            expanded: false,
        };
        return declInfo;
    }
    trackDeclarationInfo(context, declInfos) {
        for (let info of declInfos) {
            context.declarationInfos.add(info);
        }
        this.declarationInfos.set(declInfos[0].decl, declInfos);
    }
    addDeclInfoToElements(elements, property, declInfo) {
        for (let el of elements) {
            let declarations = this.elementDeclarations.get(el);
            if (!declarations) {
                let newDecls = new typescript_collections_1.MultiDictionary();
                this.elementDeclarations.set(el, newDecls);
                newDecls.setValue(property, declInfo);
            }
            else {
                declarations.setValue(property, declInfo);
            }
        }
    }
}
exports.DeclarationMapper = DeclarationMapper;
function compare(n1, n2) {
    Iif (n1 < n2)
        return -1;
    if (n1 > n2)
        return 1;
    return 0;
}
function querySelector(analysis, selector) {
    return analysis.elements.filter(e => Match_1.matches(Match_1.ElementMatcher.instance.matchSelector(e, selector, true)));
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"DeclarationMapper.js","sourceRoot":"","sources":["../../../../src/optimizations/MergeDeclarations/DeclarationMapper.ts"],"names":[],"mappings":"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"}