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;
 |