Blame view

node_modules/csso/lib/compressor/restructure/7-mergeRuleset.js 2.57 KB
aaac7fed   liuqimichale   add
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
  var utils = require('./utils.js');
  var walkRules = require('../../utils/walk.js').rules;
  
  /*
      At this step all rules has single simple selector. We try to join by equal
      declaration blocks to first rule, e.g.
  
      .a { color: red }
      b { ... }
      .b { color: red }
      ->
      .a, .b { color: red }
      b { ... }
  */
  
  function processRuleset(node, item, list) {
      var selectors = node.selector.selectors;
      var declarations = node.block.declarations;
      var nodeCompareMarker = selectors.first().compareMarker;
      var skippedCompareMarkers = {};
  
      list.nextUntil(item.next, function(next, nextItem) {
          // skip non-ruleset node if safe
          if (next.type !== 'Ruleset') {
              return utils.unsafeToSkipNode.call(selectors, next);
          }
  
          if (node.pseudoSignature !== next.pseudoSignature) {
              return true;
          }
  
          var nextFirstSelector = next.selector.selectors.head;
          var nextDeclarations = next.block.declarations;
          var nextCompareMarker = nextFirstSelector.data.compareMarker;
  
          // if next ruleset has same marked as one of skipped then stop joining
          if (nextCompareMarker in skippedCompareMarkers) {
              return true;
          }
  
          // try to join by selectors
          if (selectors.head === selectors.tail) {
              if (selectors.first().id === nextFirstSelector.data.id) {
                  declarations.appendList(nextDeclarations);
                  list.remove(nextItem);
                  return;
              }
          }
  
          // try to join by properties
          if (utils.isEqualDeclarations(declarations, nextDeclarations)) {
              var nextStr = nextFirstSelector.data.id;
  
              selectors.some(function(data, item) {
                  var curStr = data.id;
  
                  if (nextStr < curStr) {
                      selectors.insert(nextFirstSelector, item);
                      return true;
                  }
  
                  if (!item.next) {
                      selectors.insert(nextFirstSelector);
                      return true;
                  }
              });
  
              list.remove(nextItem);
              return;
          }
  
          // go to next ruleset if current one can be skipped (has no equal specificity nor element selector)
          if (nextCompareMarker === nodeCompareMarker) {
              return true;
          }
  
          skippedCompareMarkers[nextCompareMarker] = true;
      });
  };
  
  module.exports = function mergeRuleset(ast) {
      walkRules(ast, function(node, item, list) {
          if (node.type === 'Ruleset') {
              processRuleset(node, item, list);
          }
      });
  };