merge-non-adjacent-by-body.js 3.0 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788
  1. var isMergeable = require('./is-mergeable');
  2. var sortSelectors = require('../level-1/sort-selectors');
  3. var tidyRules = require('../level-1/tidy-rules');
  4. var OptimizationLevel = require('../../options/optimization-level').OptimizationLevel;
  5. var serializeBody = require('../../writer/one-time').body;
  6. var serializeRules = require('../../writer/one-time').rules;
  7. var Token = require('../../tokenizer/token');
  8. function unsafeSelector(value) {
  9. return /\.|\*| :/.test(value);
  10. }
  11. function isBemElement(token) {
  12. var asString = serializeRules(token[1]);
  13. return asString.indexOf('__') > -1 || asString.indexOf('--') > -1;
  14. }
  15. function withoutModifier(selector) {
  16. return selector.replace(/--[^ ,>+~:]+/g, '');
  17. }
  18. function removeAnyUnsafeElements(left, candidates) {
  19. var leftSelector = withoutModifier(serializeRules(left[1]));
  20. for (var body in candidates) {
  21. var right = candidates[body];
  22. var rightSelector = withoutModifier(serializeRules(right[1]));
  23. if (rightSelector.indexOf(leftSelector) > -1 || leftSelector.indexOf(rightSelector) > -1) {
  24. delete candidates[body];
  25. }
  26. }
  27. }
  28. function mergeNonAdjacentByBody(tokens, context) {
  29. var options = context.options;
  30. var mergeSemantically = options.level[OptimizationLevel.Two].mergeSemantically;
  31. var adjacentSpace = options.compatibility.selectors.adjacentSpace;
  32. var selectorsSortingMethod = options.level[OptimizationLevel.One].selectorsSortingMethod;
  33. var mergeablePseudoClasses = options.compatibility.selectors.mergeablePseudoClasses;
  34. var mergeablePseudoElements = options.compatibility.selectors.mergeablePseudoElements;
  35. var multiplePseudoMerging = options.compatibility.selectors.multiplePseudoMerging;
  36. var candidates = {};
  37. for (var i = tokens.length - 1; i >= 0; i--) {
  38. var token = tokens[i];
  39. if (token[0] != Token.RULE) { continue; }
  40. if (token[2].length > 0 && (!mergeSemantically && unsafeSelector(serializeRules(token[1])))) { candidates = {}; }
  41. if (token[2].length > 0 && mergeSemantically && isBemElement(token)) { removeAnyUnsafeElements(token, candidates); }
  42. var candidateBody = serializeBody(token[2]);
  43. var oldToken = candidates[candidateBody];
  44. if (oldToken
  45. && isMergeable(
  46. serializeRules(token[1]),
  47. mergeablePseudoClasses,
  48. mergeablePseudoElements,
  49. multiplePseudoMerging
  50. )
  51. && isMergeable(
  52. serializeRules(oldToken[1]),
  53. mergeablePseudoClasses,
  54. mergeablePseudoElements,
  55. multiplePseudoMerging
  56. )
  57. ) {
  58. if (token[2].length > 0) {
  59. token[1] = tidyRules(oldToken[1].concat(token[1]), false, adjacentSpace, false, context.warnings);
  60. token[1] = token[1].length > 1 ? sortSelectors(token[1], selectorsSortingMethod) : token[1];
  61. } else {
  62. token[1] = oldToken[1].concat(token[1]);
  63. }
  64. oldToken[2] = [];
  65. candidates[candidateBody] = null;
  66. }
  67. candidates[serializeBody(token[2])] = token;
  68. }
  69. }
  70. module.exports = mergeNonAdjacentByBody;