You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
77 lines
2.5 KiB
JavaScript
77 lines
2.5 KiB
JavaScript
5 years ago
|
var optimizeProperties = require('../properties/optimizer');
|
||
|
var stringifySelectors = require('../stringifier/one-time').selectors;
|
||
|
var extractProperties = require('./extractor');
|
||
|
var canReorder = require('./reorderable').canReorder;
|
||
|
|
||
|
function mergeNonAdjacentBySelector(tokens, options, context) {
|
||
|
var allSelectors = {};
|
||
|
var repeatedSelectors = [];
|
||
|
var i;
|
||
|
|
||
|
for (i = tokens.length - 1; i >= 0; i--) {
|
||
|
if (tokens[i][0] != 'selector')
|
||
|
continue;
|
||
|
if (tokens[i][2].length === 0)
|
||
|
continue;
|
||
|
|
||
|
var selector = stringifySelectors(tokens[i][1]);
|
||
|
allSelectors[selector] = [i].concat(allSelectors[selector] || []);
|
||
|
|
||
|
if (allSelectors[selector].length == 2)
|
||
|
repeatedSelectors.push(selector);
|
||
|
}
|
||
|
|
||
|
for (i = repeatedSelectors.length - 1; i >= 0; i--) {
|
||
|
var positions = allSelectors[repeatedSelectors[i]];
|
||
|
|
||
|
selectorIterator:
|
||
|
for (var j = positions.length - 1; j > 0; j--) {
|
||
|
var positionOne = positions[j - 1];
|
||
|
var tokenOne = tokens[positionOne];
|
||
|
var positionTwo = positions[j];
|
||
|
var tokenTwo = tokens[positionTwo];
|
||
|
|
||
|
directionIterator:
|
||
|
for (var direction = 1; direction >= -1; direction -= 2) {
|
||
|
var topToBottom = direction == 1;
|
||
|
var from = topToBottom ? positionOne + 1 : positionTwo - 1;
|
||
|
var to = topToBottom ? positionTwo : positionOne;
|
||
|
var delta = topToBottom ? 1 : -1;
|
||
|
var moved = topToBottom ? tokenOne : tokenTwo;
|
||
|
var target = topToBottom ? tokenTwo : tokenOne;
|
||
|
var movedProperties = extractProperties(moved);
|
||
|
var joinAt;
|
||
|
|
||
|
while (from != to) {
|
||
|
var traversedProperties = extractProperties(tokens[from]);
|
||
|
from += delta;
|
||
|
|
||
|
// traversed then moved as we move selectors towards the start
|
||
|
var reorderable = topToBottom ?
|
||
|
canReorder(movedProperties, traversedProperties) :
|
||
|
canReorder(traversedProperties, movedProperties);
|
||
|
|
||
|
if (!reorderable && !topToBottom)
|
||
|
continue selectorIterator;
|
||
|
if (!reorderable && topToBottom)
|
||
|
continue directionIterator;
|
||
|
}
|
||
|
|
||
|
if (topToBottom) {
|
||
|
joinAt = [moved[2].length];
|
||
|
Array.prototype.push.apply(moved[2], target[2]);
|
||
|
target[2] = moved[2];
|
||
|
} else {
|
||
|
joinAt = [target[2].length];
|
||
|
Array.prototype.push.apply(target[2], moved[2]);
|
||
|
}
|
||
|
|
||
|
optimizeProperties(target[1], target[2], joinAt, true, options, context);
|
||
|
moved[2] = [];
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
module.exports = mergeNonAdjacentBySelector;
|