462 lines
14 KiB
JavaScript
462 lines
14 KiB
JavaScript
|
var hasInherit = require('./has-inherit');
|
||
|
var hasUnset = require('./has-unset');
|
||
|
var everyValuesPair = require('./every-values-pair');
|
||
|
var findComponentIn = require('./find-component-in');
|
||
|
var isComponentOf = require('./is-component-of');
|
||
|
var isMergeableShorthand = require('./is-mergeable-shorthand');
|
||
|
var overridesNonComponentShorthand = require('./overrides-non-component-shorthand');
|
||
|
var sameVendorPrefixesIn = require('./../../vendor-prefixes').same;
|
||
|
|
||
|
var configuration = require('../../configuration');
|
||
|
var deepClone = require('../../clone').deep;
|
||
|
var restoreWithComponents = require('../restore-with-components');
|
||
|
var shallowClone = require('../../clone').shallow;
|
||
|
|
||
|
var restoreFromOptimizing = require('../../restore-from-optimizing');
|
||
|
|
||
|
var Token = require('../../../tokenizer/token');
|
||
|
var Marker = require('../../../tokenizer/marker');
|
||
|
|
||
|
var serializeProperty = require('../../../writer/one-time').property;
|
||
|
|
||
|
function sameValue(_validator, value1, value2) {
|
||
|
return value1 === value2;
|
||
|
}
|
||
|
|
||
|
function wouldBreakCompatibility(property, validator) {
|
||
|
for (var i = 0; i < property.components.length; i++) {
|
||
|
var component = property.components[i];
|
||
|
var descriptor = configuration[component.name];
|
||
|
var canOverride = descriptor && descriptor.canOverride || sameValue;
|
||
|
|
||
|
var _component = shallowClone(component);
|
||
|
_component.value = [[Token.PROPERTY_VALUE, descriptor.defaultValue]];
|
||
|
|
||
|
if (!everyValuesPair(canOverride.bind(null, validator), _component, component)) {
|
||
|
return true;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
function overrideIntoMultiplex(property, by) {
|
||
|
by.unused = true;
|
||
|
|
||
|
turnIntoMultiplex(by, multiplexSize(property));
|
||
|
property.value = by.value;
|
||
|
}
|
||
|
|
||
|
function overrideByMultiplex(property, by) {
|
||
|
by.unused = true;
|
||
|
property.multiplex = true;
|
||
|
property.value = by.value;
|
||
|
}
|
||
|
|
||
|
function overrideSimple(property, by) {
|
||
|
by.unused = true;
|
||
|
property.value = by.value;
|
||
|
}
|
||
|
|
||
|
function override(property, by) {
|
||
|
if (by.multiplex) {
|
||
|
overrideByMultiplex(property, by);
|
||
|
} else if (property.multiplex) {
|
||
|
overrideIntoMultiplex(property, by);
|
||
|
} else {
|
||
|
overrideSimple(property, by);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
function overrideShorthand(property, by) {
|
||
|
by.unused = true;
|
||
|
|
||
|
for (var i = 0, l = property.components.length; i < l; i++) {
|
||
|
override(property.components[i], by.components[i]);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
function turnIntoMultiplex(property, size) {
|
||
|
property.multiplex = true;
|
||
|
|
||
|
if (configuration[property.name].shorthand) {
|
||
|
turnShorthandValueIntoMultiplex(property, size);
|
||
|
} else {
|
||
|
turnLonghandValueIntoMultiplex(property, size);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
function turnShorthandValueIntoMultiplex(property, size) {
|
||
|
var component;
|
||
|
var i, l;
|
||
|
|
||
|
for (i = 0, l = property.components.length; i < l; i++) {
|
||
|
component = property.components[i];
|
||
|
|
||
|
if (!component.multiplex) {
|
||
|
turnLonghandValueIntoMultiplex(component, size);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
function turnLonghandValueIntoMultiplex(property, size) {
|
||
|
var descriptor = configuration[property.name];
|
||
|
var withRealValue = descriptor.intoMultiplexMode == 'real';
|
||
|
var withValue = descriptor.intoMultiplexMode == 'real'
|
||
|
? property.value.slice(0)
|
||
|
: (descriptor.intoMultiplexMode == 'placeholder' ? descriptor.placeholderValue : descriptor.defaultValue);
|
||
|
var i = multiplexSize(property);
|
||
|
var j;
|
||
|
var m = withValue.length;
|
||
|
|
||
|
for (; i < size; i++) {
|
||
|
property.value.push([Token.PROPERTY_VALUE, Marker.COMMA]);
|
||
|
|
||
|
if (Array.isArray(withValue)) {
|
||
|
for (j = 0; j < m; j++) {
|
||
|
property.value.push(withRealValue ? withValue[j] : [Token.PROPERTY_VALUE, withValue[j]]);
|
||
|
}
|
||
|
} else {
|
||
|
property.value.push(withRealValue ? withValue : [Token.PROPERTY_VALUE, withValue]);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
function multiplexSize(component) {
|
||
|
var size = 0;
|
||
|
|
||
|
for (var i = 0, l = component.value.length; i < l; i++) {
|
||
|
if (component.value[i][1] == Marker.COMMA) { size++; }
|
||
|
}
|
||
|
|
||
|
return size + 1;
|
||
|
}
|
||
|
|
||
|
function lengthOf(property) {
|
||
|
var fakeAsArray = [
|
||
|
Token.PROPERTY,
|
||
|
[Token.PROPERTY_NAME, property.name]
|
||
|
].concat(property.value);
|
||
|
return serializeProperty([fakeAsArray], 0).length;
|
||
|
}
|
||
|
|
||
|
function moreSameShorthands(properties, startAt, name) {
|
||
|
// Since we run the main loop in `compactOverrides` backwards, at this point some
|
||
|
// properties may not be marked as unused.
|
||
|
// We should consider reverting the order if possible
|
||
|
var count = 0;
|
||
|
|
||
|
for (var i = startAt; i >= 0; i--) {
|
||
|
if (properties[i].name == name && !properties[i].unused) { count++; }
|
||
|
if (count > 1) { break; }
|
||
|
}
|
||
|
|
||
|
return count > 1;
|
||
|
}
|
||
|
|
||
|
function overridingFunction(shorthand, validator) {
|
||
|
for (var i = 0, l = shorthand.components.length; i < l; i++) {
|
||
|
if (!anyValue(validator.isUrl, shorthand.components[i])
|
||
|
&& anyValue(validator.isFunction, shorthand.components[i])) { return true; }
|
||
|
}
|
||
|
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
function anyValue(fn, property) {
|
||
|
for (var i = 0, l = property.value.length; i < l; i++) {
|
||
|
if (property.value[i][1] == Marker.COMMA) { continue; }
|
||
|
|
||
|
if (fn(property.value[i][1])) { return true; }
|
||
|
}
|
||
|
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
function wouldResultInLongerValue(left, right) {
|
||
|
if (!left.multiplex && !right.multiplex || left.multiplex && right.multiplex) { return false; }
|
||
|
|
||
|
var multiplex = left.multiplex ? left : right;
|
||
|
var simple = left.multiplex ? right : left;
|
||
|
var component;
|
||
|
|
||
|
var multiplexClone = deepClone(multiplex);
|
||
|
restoreFromOptimizing([multiplexClone], restoreWithComponents);
|
||
|
|
||
|
var simpleClone = deepClone(simple);
|
||
|
restoreFromOptimizing([simpleClone], restoreWithComponents);
|
||
|
|
||
|
var lengthBefore = lengthOf(multiplexClone) + 1 + lengthOf(simpleClone);
|
||
|
|
||
|
if (left.multiplex) {
|
||
|
component = findComponentIn(multiplexClone, simpleClone);
|
||
|
overrideIntoMultiplex(component, simpleClone);
|
||
|
} else {
|
||
|
component = findComponentIn(simpleClone, multiplexClone);
|
||
|
turnIntoMultiplex(simpleClone, multiplexSize(multiplexClone));
|
||
|
overrideByMultiplex(component, multiplexClone);
|
||
|
}
|
||
|
|
||
|
restoreFromOptimizing([simpleClone], restoreWithComponents);
|
||
|
|
||
|
var lengthAfter = lengthOf(simpleClone);
|
||
|
|
||
|
return lengthBefore <= lengthAfter;
|
||
|
}
|
||
|
|
||
|
function isCompactable(property) {
|
||
|
return property.name in configuration;
|
||
|
}
|
||
|
|
||
|
function noneOverrideHack(left, right) {
|
||
|
return !left.multiplex
|
||
|
&& (left.name == 'background' || left.name == 'background-image')
|
||
|
&& right.multiplex
|
||
|
&& (right.name == 'background' || right.name == 'background-image')
|
||
|
&& anyLayerIsNone(right.value);
|
||
|
}
|
||
|
|
||
|
function anyLayerIsNone(values) {
|
||
|
var layers = intoLayers(values);
|
||
|
|
||
|
for (var i = 0, l = layers.length; i < l; i++) {
|
||
|
if (layers[i].length == 1 && layers[i][0][1] == 'none') { return true; }
|
||
|
}
|
||
|
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
function intoLayers(values) {
|
||
|
var layers = [];
|
||
|
|
||
|
for (var i = 0, layer = [], l = values.length; i < l; i++) {
|
||
|
var value = values[i];
|
||
|
if (value[1] == Marker.COMMA) {
|
||
|
layers.push(layer);
|
||
|
layer = [];
|
||
|
} else {
|
||
|
layer.push(value);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
layers.push(layer);
|
||
|
return layers;
|
||
|
}
|
||
|
|
||
|
function overrideProperties(properties, withMerging, compatibility, validator) {
|
||
|
var mayOverride, right, left, component;
|
||
|
var overriddenComponents;
|
||
|
var overriddenComponent;
|
||
|
var overridingComponent;
|
||
|
var overridable;
|
||
|
var i, j, k;
|
||
|
|
||
|
propertyLoop:
|
||
|
for (i = properties.length - 1; i >= 0; i--) {
|
||
|
right = properties[i];
|
||
|
|
||
|
if (!isCompactable(right)) { continue; }
|
||
|
|
||
|
if (right.block) { continue; }
|
||
|
|
||
|
mayOverride = configuration[right.name].canOverride || sameValue;
|
||
|
|
||
|
traverseLoop:
|
||
|
for (j = i - 1; j >= 0; j--) {
|
||
|
left = properties[j];
|
||
|
|
||
|
if (!isCompactable(left)) { continue; }
|
||
|
|
||
|
if (left.block) { continue; }
|
||
|
|
||
|
if (left.dynamic || right.dynamic) { continue; }
|
||
|
|
||
|
if (left.unused || right.unused) { continue; }
|
||
|
|
||
|
if (left.hack && !right.hack && !right.important || !left.hack && !left.important && right.hack) { continue; }
|
||
|
|
||
|
if (left.important == right.important && left.hack[0] != right.hack[0]) { continue; }
|
||
|
|
||
|
if (left.important == right.important
|
||
|
&& (left.hack[0] != right.hack[0] || (left.hack[1] && left.hack[1] != right.hack[1]))) { continue; }
|
||
|
|
||
|
if (hasInherit(right)) { continue; }
|
||
|
|
||
|
if (noneOverrideHack(left, right)) { continue; }
|
||
|
|
||
|
if (right.shorthand && isComponentOf(right, left)) {
|
||
|
// maybe `left` can be overridden by `right` which is a shorthand?
|
||
|
if (!right.important && left.important) { continue; }
|
||
|
|
||
|
if (!sameVendorPrefixesIn([left], right.components)) { continue; }
|
||
|
|
||
|
if (!anyValue(validator.isFunction, left) && overridingFunction(right, validator)) { continue; }
|
||
|
|
||
|
if (!isMergeableShorthand(right)) {
|
||
|
left.unused = true;
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
component = findComponentIn(right, left);
|
||
|
mayOverride = configuration[left.name].canOverride || sameValue;
|
||
|
if (everyValuesPair(mayOverride.bind(null, validator), left, component)) {
|
||
|
left.unused = true;
|
||
|
}
|
||
|
} else if (right.shorthand && overridesNonComponentShorthand(right, left)) {
|
||
|
// `right` is a shorthand while `left` can be overriden by it, think `border` and `border-top`
|
||
|
if (!right.important && left.important) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
if (!sameVendorPrefixesIn([left], right.components)) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
if (!anyValue(validator.isFunction, left) && overridingFunction(right, validator)) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
overriddenComponents = left.shorthand
|
||
|
? left.components
|
||
|
: [left];
|
||
|
|
||
|
for (k = overriddenComponents.length - 1; k >= 0; k--) {
|
||
|
overriddenComponent = overriddenComponents[k];
|
||
|
overridingComponent = findComponentIn(right, overriddenComponent);
|
||
|
mayOverride = configuration[overriddenComponent.name].canOverride || sameValue;
|
||
|
|
||
|
if (!everyValuesPair(mayOverride.bind(null, validator), left, overridingComponent)) {
|
||
|
continue traverseLoop;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
left.unused = true;
|
||
|
} else if (withMerging && left.shorthand && !right.shorthand && isComponentOf(left, right, true)) {
|
||
|
// maybe `right` can be pulled into `left` which is a shorthand?
|
||
|
if (right.important && !left.important) { continue; }
|
||
|
|
||
|
if (!right.important && left.important) {
|
||
|
right.unused = true;
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
// Pending more clever algorithm in #527
|
||
|
if (moreSameShorthands(properties, i - 1, left.name)) { continue; }
|
||
|
|
||
|
if (overridingFunction(left, validator)) { continue; }
|
||
|
|
||
|
if (!isMergeableShorthand(left)) { continue; }
|
||
|
|
||
|
if (hasUnset(left) || hasUnset(right)) { continue; }
|
||
|
|
||
|
component = findComponentIn(left, right);
|
||
|
if (everyValuesPair(mayOverride.bind(null, validator), component, right)) {
|
||
|
var disabledBackgroundMerging = !compatibility.properties.backgroundClipMerging && component.name.indexOf('background-clip') > -1
|
||
|
|| !compatibility.properties.backgroundOriginMerging && component.name.indexOf('background-origin') > -1
|
||
|
|| !compatibility.properties.backgroundSizeMerging && component.name.indexOf('background-size') > -1;
|
||
|
var nonMergeableValue = configuration[right.name].nonMergeableValue === right.value[0][1];
|
||
|
|
||
|
if (disabledBackgroundMerging || nonMergeableValue) { continue; }
|
||
|
|
||
|
if (!compatibility.properties.merging && wouldBreakCompatibility(left, validator)) { continue; }
|
||
|
|
||
|
if (component.value[0][1] != right.value[0][1] && (hasInherit(left) || hasInherit(right))) { continue; }
|
||
|
|
||
|
if (wouldResultInLongerValue(left, right)) { continue; }
|
||
|
|
||
|
if (!left.multiplex && right.multiplex) { turnIntoMultiplex(left, multiplexSize(right)); }
|
||
|
|
||
|
override(component, right);
|
||
|
left.dirty = true;
|
||
|
}
|
||
|
} else if (withMerging && left.shorthand && right.shorthand && left.name == right.name) {
|
||
|
// merge if all components can be merged
|
||
|
|
||
|
if (!left.multiplex && right.multiplex) { continue; }
|
||
|
|
||
|
if (!right.important && left.important) {
|
||
|
right.unused = true;
|
||
|
continue propertyLoop;
|
||
|
}
|
||
|
|
||
|
if (right.important && !left.important) {
|
||
|
left.unused = true;
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
if (!isMergeableShorthand(right)) {
|
||
|
left.unused = true;
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
for (k = left.components.length - 1; k >= 0; k--) {
|
||
|
var leftComponent = left.components[k];
|
||
|
var rightComponent = right.components[k];
|
||
|
|
||
|
mayOverride = configuration[leftComponent.name].canOverride || sameValue;
|
||
|
if (!everyValuesPair(mayOverride.bind(null, validator), leftComponent, rightComponent)) {
|
||
|
continue propertyLoop;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
overrideShorthand(left, right);
|
||
|
left.dirty = true;
|
||
|
} else if (withMerging && left.shorthand && right.shorthand && isComponentOf(left, right)) {
|
||
|
// border is a shorthand but any of its components is a shorthand too
|
||
|
|
||
|
if (!left.important && right.important) { continue; }
|
||
|
|
||
|
component = findComponentIn(left, right);
|
||
|
mayOverride = configuration[right.name].canOverride || sameValue;
|
||
|
if (!everyValuesPair(mayOverride.bind(null, validator), component, right)) { continue; }
|
||
|
|
||
|
if (left.important && !right.important) {
|
||
|
right.unused = true;
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
var rightRestored = configuration[right.name].restore(right, configuration);
|
||
|
if (rightRestored.length > 1) { continue; }
|
||
|
|
||
|
component = findComponentIn(left, right);
|
||
|
override(component, right);
|
||
|
right.dirty = true;
|
||
|
} else if (left.name == right.name) {
|
||
|
// two non-shorthands should be merged based on understandability
|
||
|
overridable = true;
|
||
|
|
||
|
if (right.shorthand) {
|
||
|
for (k = right.components.length - 1; k >= 0 && overridable; k--) {
|
||
|
overriddenComponent = left.components[k];
|
||
|
overridingComponent = right.components[k];
|
||
|
mayOverride = configuration[overridingComponent.name].canOverride || sameValue;
|
||
|
|
||
|
overridable = everyValuesPair(mayOverride.bind(null, validator), overriddenComponent, overridingComponent);
|
||
|
}
|
||
|
} else {
|
||
|
mayOverride = configuration[right.name].canOverride || sameValue;
|
||
|
overridable = everyValuesPair(mayOverride.bind(null, validator), left, right);
|
||
|
}
|
||
|
|
||
|
if (left.important && !right.important && overridable) {
|
||
|
right.unused = true;
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
if (!left.important && right.important && overridable) {
|
||
|
left.unused = true;
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
if (!overridable) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
left.unused = true;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
module.exports = overrideProperties;
|