698 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
		
		
			
		
	
	
			698 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| 
								 | 
							
								/*********************************************************************
							 | 
						||
| 
								 | 
							
								 * These are commonly used parsers for CSS Values they take a string *
							 | 
						||
| 
								 | 
							
								 * to parse and return a string after it's been converted, if needed *
							 | 
						||
| 
								 | 
							
								 ********************************************************************/
							 | 
						||
| 
								 | 
							
								'use strict';
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								const namedColors = require('./named_colors.json');
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								exports.TYPES = {
							 | 
						||
| 
								 | 
							
								  INTEGER: 1,
							 | 
						||
| 
								 | 
							
								  NUMBER: 2,
							 | 
						||
| 
								 | 
							
								  LENGTH: 3,
							 | 
						||
| 
								 | 
							
								  PERCENT: 4,
							 | 
						||
| 
								 | 
							
								  URL: 5,
							 | 
						||
| 
								 | 
							
								  COLOR: 6,
							 | 
						||
| 
								 | 
							
								  STRING: 7,
							 | 
						||
| 
								 | 
							
								  ANGLE: 8,
							 | 
						||
| 
								 | 
							
								  KEYWORD: 9,
							 | 
						||
| 
								 | 
							
								  NULL_OR_EMPTY_STR: 10,
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// rough regular expressions
							 | 
						||
| 
								 | 
							
								var integerRegEx = /^[-+]?[0-9]+$/;
							 | 
						||
| 
								 | 
							
								var numberRegEx = /^[-+]?[0-9]*\.[0-9]+$/;
							 | 
						||
| 
								 | 
							
								var lengthRegEx = /^(0|[-+]?[0-9]*\.?[0-9]+(in|cm|em|mm|pt|pc|px|ex|rem|vh|vw))$/;
							 | 
						||
| 
								 | 
							
								var percentRegEx = /^[-+]?[0-9]*\.?[0-9]+%$/;
							 | 
						||
| 
								 | 
							
								var urlRegEx = /^url\(\s*([^)]*)\s*\)$/;
							 | 
						||
| 
								 | 
							
								var stringRegEx = /^("[^"]*"|'[^']*')$/;
							 | 
						||
| 
								 | 
							
								var colorRegEx1 = /^#[0-9a-fA-F][0-9a-fA-F][0-9a-fA-F]([0-9a-fA-F][0-9a-fA-F][0-9a-fA-F])?$/;
							 | 
						||
| 
								 | 
							
								var colorRegEx2 = /^rgb\(([^)]*)\)$/;
							 | 
						||
| 
								 | 
							
								var colorRegEx3 = /^rgba\(([^)]*)\)$/;
							 | 
						||
| 
								 | 
							
								var colorRegEx4 = /^hsla?\(\s*(-?\d+|-?\d*.\d+)\s*,\s*(-?\d+|-?\d*.\d+)%\s*,\s*(-?\d+|-?\d*.\d+)%\s*(,\s*(-?\d+|-?\d*.\d+)\s*)?\)/;
							 | 
						||
| 
								 | 
							
								var angleRegEx = /^([-+]?[0-9]*\.?[0-9]+)(deg|grad|rad)$/;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// This will return one of the above types based on the passed in string
							 | 
						||
| 
								 | 
							
								exports.valueType = function valueType(val) {
							 | 
						||
| 
								 | 
							
								  if (val === '' || val === null) {
							 | 
						||
| 
								 | 
							
								    return exports.TYPES.NULL_OR_EMPTY_STR;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  if (typeof val === 'number') {
							 | 
						||
| 
								 | 
							
								    val = val.toString();
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  if (typeof val !== 'string') {
							 | 
						||
| 
								 | 
							
								    return undefined;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  if (integerRegEx.test(val)) {
							 | 
						||
| 
								 | 
							
								    return exports.TYPES.INTEGER;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  if (numberRegEx.test(val)) {
							 | 
						||
| 
								 | 
							
								    return exports.TYPES.NUMBER;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  if (lengthRegEx.test(val)) {
							 | 
						||
| 
								 | 
							
								    return exports.TYPES.LENGTH;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  if (percentRegEx.test(val)) {
							 | 
						||
| 
								 | 
							
								    return exports.TYPES.PERCENT;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  if (urlRegEx.test(val)) {
							 | 
						||
| 
								 | 
							
								    return exports.TYPES.URL;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  if (stringRegEx.test(val)) {
							 | 
						||
| 
								 | 
							
								    return exports.TYPES.STRING;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  if (angleRegEx.test(val)) {
							 | 
						||
| 
								 | 
							
								    return exports.TYPES.ANGLE;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  if (colorRegEx1.test(val)) {
							 | 
						||
| 
								 | 
							
								    return exports.TYPES.COLOR;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  var res = colorRegEx2.exec(val);
							 | 
						||
| 
								 | 
							
								  var parts;
							 | 
						||
| 
								 | 
							
								  if (res !== null) {
							 | 
						||
| 
								 | 
							
								    parts = res[1].split(/\s*,\s*/);
							 | 
						||
| 
								 | 
							
								    if (parts.length !== 3) {
							 | 
						||
| 
								 | 
							
								      return undefined;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    if (
							 | 
						||
| 
								 | 
							
								      parts.every(percentRegEx.test.bind(percentRegEx)) ||
							 | 
						||
| 
								 | 
							
								      parts.every(integerRegEx.test.bind(integerRegEx))
							 | 
						||
| 
								 | 
							
								    ) {
							 | 
						||
| 
								 | 
							
								      return exports.TYPES.COLOR;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    return undefined;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  res = colorRegEx3.exec(val);
							 | 
						||
| 
								 | 
							
								  if (res !== null) {
							 | 
						||
| 
								 | 
							
								    parts = res[1].split(/\s*,\s*/);
							 | 
						||
| 
								 | 
							
								    if (parts.length !== 4) {
							 | 
						||
| 
								 | 
							
								      return undefined;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    if (
							 | 
						||
| 
								 | 
							
								      parts.slice(0, 3).every(percentRegEx.test.bind(percentRegEx)) ||
							 | 
						||
| 
								 | 
							
								      parts.every(integerRegEx.test.bind(integerRegEx))
							 | 
						||
| 
								 | 
							
								    ) {
							 | 
						||
| 
								 | 
							
								      if (numberRegEx.test(parts[3])) {
							 | 
						||
| 
								 | 
							
								        return exports.TYPES.COLOR;
							 | 
						||
| 
								 | 
							
								      }
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    return undefined;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  if (colorRegEx4.test(val)) {
							 | 
						||
| 
								 | 
							
								    return exports.TYPES.COLOR;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  // could still be a color, one of the standard keyword colors
							 | 
						||
| 
								 | 
							
								  val = val.toLowerCase();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  if (namedColors.includes(val)) {
							 | 
						||
| 
								 | 
							
								    return exports.TYPES.COLOR;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  switch (val) {
							 | 
						||
| 
								 | 
							
								    // the following are deprecated in CSS3
							 | 
						||
| 
								 | 
							
								    case 'activeborder':
							 | 
						||
| 
								 | 
							
								    case 'activecaption':
							 | 
						||
| 
								 | 
							
								    case 'appworkspace':
							 | 
						||
| 
								 | 
							
								    case 'background':
							 | 
						||
| 
								 | 
							
								    case 'buttonface':
							 | 
						||
| 
								 | 
							
								    case 'buttonhighlight':
							 | 
						||
| 
								 | 
							
								    case 'buttonshadow':
							 | 
						||
| 
								 | 
							
								    case 'buttontext':
							 | 
						||
| 
								 | 
							
								    case 'captiontext':
							 | 
						||
| 
								 | 
							
								    case 'graytext':
							 | 
						||
| 
								 | 
							
								    case 'highlight':
							 | 
						||
| 
								 | 
							
								    case 'highlighttext':
							 | 
						||
| 
								 | 
							
								    case 'inactiveborder':
							 | 
						||
| 
								 | 
							
								    case 'inactivecaption':
							 | 
						||
| 
								 | 
							
								    case 'inactivecaptiontext':
							 | 
						||
| 
								 | 
							
								    case 'infobackground':
							 | 
						||
| 
								 | 
							
								    case 'infotext':
							 | 
						||
| 
								 | 
							
								    case 'menu':
							 | 
						||
| 
								 | 
							
								    case 'menutext':
							 | 
						||
| 
								 | 
							
								    case 'scrollbar':
							 | 
						||
| 
								 | 
							
								    case 'threeddarkshadow':
							 | 
						||
| 
								 | 
							
								    case 'threedface':
							 | 
						||
| 
								 | 
							
								    case 'threedhighlight':
							 | 
						||
| 
								 | 
							
								    case 'threedlightshadow':
							 | 
						||
| 
								 | 
							
								    case 'threedshadow':
							 | 
						||
| 
								 | 
							
								    case 'window':
							 | 
						||
| 
								 | 
							
								    case 'windowframe':
							 | 
						||
| 
								 | 
							
								    case 'windowtext':
							 | 
						||
| 
								 | 
							
								      return exports.TYPES.COLOR;
							 | 
						||
| 
								 | 
							
								    default:
							 | 
						||
| 
								 | 
							
								      return exports.TYPES.KEYWORD;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								exports.parseInteger = function parseInteger(val) {
							 | 
						||
| 
								 | 
							
								  var type = exports.valueType(val);
							 | 
						||
| 
								 | 
							
								  if (type === exports.TYPES.NULL_OR_EMPTY_STR) {
							 | 
						||
| 
								 | 
							
								    return val;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  if (type !== exports.TYPES.INTEGER) {
							 | 
						||
| 
								 | 
							
								    return undefined;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  return String(parseInt(val, 10));
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								exports.parseNumber = function parseNumber(val) {
							 | 
						||
| 
								 | 
							
								  var type = exports.valueType(val);
							 | 
						||
| 
								 | 
							
								  if (type === exports.TYPES.NULL_OR_EMPTY_STR) {
							 | 
						||
| 
								 | 
							
								    return val;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  if (type !== exports.TYPES.NUMBER && type !== exports.TYPES.INTEGER) {
							 | 
						||
| 
								 | 
							
								    return undefined;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  return String(parseFloat(val));
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								exports.parseLength = function parseLength(val) {
							 | 
						||
| 
								 | 
							
								  if (val === 0 || val === '0') {
							 | 
						||
| 
								 | 
							
								    return '0px';
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  var type = exports.valueType(val);
							 | 
						||
| 
								 | 
							
								  if (type === exports.TYPES.NULL_OR_EMPTY_STR) {
							 | 
						||
| 
								 | 
							
								    return val;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  if (type !== exports.TYPES.LENGTH) {
							 | 
						||
| 
								 | 
							
								    return undefined;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  return val;
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								exports.parsePercent = function parsePercent(val) {
							 | 
						||
| 
								 | 
							
								  if (val === 0 || val === '0') {
							 | 
						||
| 
								 | 
							
								    return '0%';
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  var type = exports.valueType(val);
							 | 
						||
| 
								 | 
							
								  if (type === exports.TYPES.NULL_OR_EMPTY_STR) {
							 | 
						||
| 
								 | 
							
								    return val;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  if (type !== exports.TYPES.PERCENT) {
							 | 
						||
| 
								 | 
							
								    return undefined;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  return val;
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// either a length or a percent
							 | 
						||
| 
								 | 
							
								exports.parseMeasurement = function parseMeasurement(val) {
							 | 
						||
| 
								 | 
							
								  var length = exports.parseLength(val);
							 | 
						||
| 
								 | 
							
								  if (length !== undefined) {
							 | 
						||
| 
								 | 
							
								    return length;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  return exports.parsePercent(val);
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								exports.parseUrl = function parseUrl(val) {
							 | 
						||
| 
								 | 
							
								  var type = exports.valueType(val);
							 | 
						||
| 
								 | 
							
								  if (type === exports.TYPES.NULL_OR_EMPTY_STR) {
							 | 
						||
| 
								 | 
							
								    return val;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  var res = urlRegEx.exec(val);
							 | 
						||
| 
								 | 
							
								  // does it match the regex?
							 | 
						||
| 
								 | 
							
								  if (!res) {
							 | 
						||
| 
								 | 
							
								    return undefined;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  var str = res[1];
							 | 
						||
| 
								 | 
							
								  // if it starts with single or double quotes, does it end with the same?
							 | 
						||
| 
								 | 
							
								  if ((str[0] === '"' || str[0] === "'") && str[0] !== str[str.length - 1]) {
							 | 
						||
| 
								 | 
							
								    return undefined;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  if (str[0] === '"' || str[0] === "'") {
							 | 
						||
| 
								 | 
							
								    str = str.substr(1, str.length - 2);
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  var i;
							 | 
						||
| 
								 | 
							
								  for (i = 0; i < str.length; i++) {
							 | 
						||
| 
								 | 
							
								    switch (str[i]) {
							 | 
						||
| 
								 | 
							
								      case '(':
							 | 
						||
| 
								 | 
							
								      case ')':
							 | 
						||
| 
								 | 
							
								      case ' ':
							 | 
						||
| 
								 | 
							
								      case '\t':
							 | 
						||
| 
								 | 
							
								      case '\n':
							 | 
						||
| 
								 | 
							
								      case "'":
							 | 
						||
| 
								 | 
							
								      case '"':
							 | 
						||
| 
								 | 
							
								        return undefined;
							 | 
						||
| 
								 | 
							
								      case '\\':
							 | 
						||
| 
								 | 
							
								        i++;
							 | 
						||
| 
								 | 
							
								        break;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  return 'url(' + str + ')';
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								exports.parseString = function parseString(val) {
							 | 
						||
| 
								 | 
							
								  var type = exports.valueType(val);
							 | 
						||
| 
								 | 
							
								  if (type === exports.TYPES.NULL_OR_EMPTY_STR) {
							 | 
						||
| 
								 | 
							
								    return val;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  if (type !== exports.TYPES.STRING) {
							 | 
						||
| 
								 | 
							
								    return undefined;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  var i;
							 | 
						||
| 
								 | 
							
								  for (i = 1; i < val.length - 1; i++) {
							 | 
						||
| 
								 | 
							
								    switch (val[i]) {
							 | 
						||
| 
								 | 
							
								      case val[0]:
							 | 
						||
| 
								 | 
							
								        return undefined;
							 | 
						||
| 
								 | 
							
								      case '\\':
							 | 
						||
| 
								 | 
							
								        i++;
							 | 
						||
| 
								 | 
							
								        while (i < val.length - 1 && /[0-9A-Fa-f]/.test(val[i])) {
							 | 
						||
| 
								 | 
							
								          i++;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								        break;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  if (i >= val.length) {
							 | 
						||
| 
								 | 
							
								    return undefined;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  return val;
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								exports.parseColor = function parseColor(val) {
							 | 
						||
| 
								 | 
							
								  var type = exports.valueType(val);
							 | 
						||
| 
								 | 
							
								  if (type === exports.TYPES.NULL_OR_EMPTY_STR) {
							 | 
						||
| 
								 | 
							
								    return val;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  var red,
							 | 
						||
| 
								 | 
							
								    green,
							 | 
						||
| 
								 | 
							
								    blue,
							 | 
						||
| 
								 | 
							
								    hue,
							 | 
						||
| 
								 | 
							
								    saturation,
							 | 
						||
| 
								 | 
							
								    lightness,
							 | 
						||
| 
								 | 
							
								    alpha = 1;
							 | 
						||
| 
								 | 
							
								  var parts;
							 | 
						||
| 
								 | 
							
								  var res = colorRegEx1.exec(val);
							 | 
						||
| 
								 | 
							
								  // is it #aaa or #ababab
							 | 
						||
| 
								 | 
							
								  if (res) {
							 | 
						||
| 
								 | 
							
								    var hex = val.substr(1);
							 | 
						||
| 
								 | 
							
								    if (hex.length === 3) {
							 | 
						||
| 
								 | 
							
								      hex = hex[0] + hex[0] + hex[1] + hex[1] + hex[2] + hex[2];
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    red = parseInt(hex.substr(0, 2), 16);
							 | 
						||
| 
								 | 
							
								    green = parseInt(hex.substr(2, 2), 16);
							 | 
						||
| 
								 | 
							
								    blue = parseInt(hex.substr(4, 2), 16);
							 | 
						||
| 
								 | 
							
								    return 'rgb(' + red + ', ' + green + ', ' + blue + ')';
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  res = colorRegEx2.exec(val);
							 | 
						||
| 
								 | 
							
								  if (res) {
							 | 
						||
| 
								 | 
							
								    parts = res[1].split(/\s*,\s*/);
							 | 
						||
| 
								 | 
							
								    if (parts.length !== 3) {
							 | 
						||
| 
								 | 
							
								      return undefined;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    if (parts.every(percentRegEx.test.bind(percentRegEx))) {
							 | 
						||
| 
								 | 
							
								      red = Math.floor((parseFloat(parts[0].slice(0, -1)) * 255) / 100);
							 | 
						||
| 
								 | 
							
								      green = Math.floor((parseFloat(parts[1].slice(0, -1)) * 255) / 100);
							 | 
						||
| 
								 | 
							
								      blue = Math.floor((parseFloat(parts[2].slice(0, -1)) * 255) / 100);
							 | 
						||
| 
								 | 
							
								    } else if (parts.every(integerRegEx.test.bind(integerRegEx))) {
							 | 
						||
| 
								 | 
							
								      red = parseInt(parts[0], 10);
							 | 
						||
| 
								 | 
							
								      green = parseInt(parts[1], 10);
							 | 
						||
| 
								 | 
							
								      blue = parseInt(parts[2], 10);
							 | 
						||
| 
								 | 
							
								    } else {
							 | 
						||
| 
								 | 
							
								      return undefined;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    red = Math.min(255, Math.max(0, red));
							 | 
						||
| 
								 | 
							
								    green = Math.min(255, Math.max(0, green));
							 | 
						||
| 
								 | 
							
								    blue = Math.min(255, Math.max(0, blue));
							 | 
						||
| 
								 | 
							
								    return 'rgb(' + red + ', ' + green + ', ' + blue + ')';
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  res = colorRegEx3.exec(val);
							 | 
						||
| 
								 | 
							
								  if (res) {
							 | 
						||
| 
								 | 
							
								    parts = res[1].split(/\s*,\s*/);
							 | 
						||
| 
								 | 
							
								    if (parts.length !== 4) {
							 | 
						||
| 
								 | 
							
								      return undefined;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    if (parts.slice(0, 3).every(percentRegEx.test.bind(percentRegEx))) {
							 | 
						||
| 
								 | 
							
								      red = Math.floor((parseFloat(parts[0].slice(0, -1)) * 255) / 100);
							 | 
						||
| 
								 | 
							
								      green = Math.floor((parseFloat(parts[1].slice(0, -1)) * 255) / 100);
							 | 
						||
| 
								 | 
							
								      blue = Math.floor((parseFloat(parts[2].slice(0, -1)) * 255) / 100);
							 | 
						||
| 
								 | 
							
								      alpha = parseFloat(parts[3]);
							 | 
						||
| 
								 | 
							
								    } else if (parts.slice(0, 3).every(integerRegEx.test.bind(integerRegEx))) {
							 | 
						||
| 
								 | 
							
								      red = parseInt(parts[0], 10);
							 | 
						||
| 
								 | 
							
								      green = parseInt(parts[1], 10);
							 | 
						||
| 
								 | 
							
								      blue = parseInt(parts[2], 10);
							 | 
						||
| 
								 | 
							
								      alpha = parseFloat(parts[3]);
							 | 
						||
| 
								 | 
							
								    } else {
							 | 
						||
| 
								 | 
							
								      return undefined;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    if (isNaN(alpha)) {
							 | 
						||
| 
								 | 
							
								      alpha = 1;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    red = Math.min(255, Math.max(0, red));
							 | 
						||
| 
								 | 
							
								    green = Math.min(255, Math.max(0, green));
							 | 
						||
| 
								 | 
							
								    blue = Math.min(255, Math.max(0, blue));
							 | 
						||
| 
								 | 
							
								    alpha = Math.min(1, Math.max(0, alpha));
							 | 
						||
| 
								 | 
							
								    if (alpha === 1) {
							 | 
						||
| 
								 | 
							
								      return 'rgb(' + red + ', ' + green + ', ' + blue + ')';
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    return 'rgba(' + red + ', ' + green + ', ' + blue + ', ' + alpha + ')';
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  res = colorRegEx4.exec(val);
							 | 
						||
| 
								 | 
							
								  if (res) {
							 | 
						||
| 
								 | 
							
								    const [, _hue, _saturation, _lightness, _alphaString = ''] = res;
							 | 
						||
| 
								 | 
							
								    const _alpha = parseFloat(_alphaString.replace(',', '').trim());
							 | 
						||
| 
								 | 
							
								    if (!_hue || !_saturation || !_lightness) {
							 | 
						||
| 
								 | 
							
								      return undefined;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    hue = parseFloat(_hue);
							 | 
						||
| 
								 | 
							
								    saturation = parseInt(_saturation, 10);
							 | 
						||
| 
								 | 
							
								    lightness = parseInt(_lightness, 10);
							 | 
						||
| 
								 | 
							
								    if (_alpha && numberRegEx.test(_alpha)) {
							 | 
						||
| 
								 | 
							
								      alpha = parseFloat(_alpha);
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    if (!_alphaString || alpha === 1) {
							 | 
						||
| 
								 | 
							
								      return 'hsl(' + hue + ', ' + saturation + '%, ' + lightness + '%)';
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    return 'hsla(' + hue + ', ' + saturation + '%, ' + lightness + '%, ' + alpha + ')';
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  if (type === exports.TYPES.COLOR) {
							 | 
						||
| 
								 | 
							
								    return val;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  return undefined;
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								exports.parseAngle = function parseAngle(val) {
							 | 
						||
| 
								 | 
							
								  var type = exports.valueType(val);
							 | 
						||
| 
								 | 
							
								  if (type === exports.TYPES.NULL_OR_EMPTY_STR) {
							 | 
						||
| 
								 | 
							
								    return val;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  if (type !== exports.TYPES.ANGLE) {
							 | 
						||
| 
								 | 
							
								    return undefined;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  var res = angleRegEx.exec(val);
							 | 
						||
| 
								 | 
							
								  var flt = parseFloat(res[1]);
							 | 
						||
| 
								 | 
							
								  if (res[2] === 'rad') {
							 | 
						||
| 
								 | 
							
								    flt *= 180 / Math.PI;
							 | 
						||
| 
								 | 
							
								  } else if (res[2] === 'grad') {
							 | 
						||
| 
								 | 
							
								    flt *= 360 / 400;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  while (flt < 0) {
							 | 
						||
| 
								 | 
							
								    flt += 360;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  while (flt > 360) {
							 | 
						||
| 
								 | 
							
								    flt -= 360;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  return flt + 'deg';
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								exports.parseKeyword = function parseKeyword(val, valid_keywords) {
							 | 
						||
| 
								 | 
							
								  var type = exports.valueType(val);
							 | 
						||
| 
								 | 
							
								  if (type === exports.TYPES.NULL_OR_EMPTY_STR) {
							 | 
						||
| 
								 | 
							
								    return val;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  if (type !== exports.TYPES.KEYWORD) {
							 | 
						||
| 
								 | 
							
								    return undefined;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  val = val.toString().toLowerCase();
							 | 
						||
| 
								 | 
							
								  var i;
							 | 
						||
| 
								 | 
							
								  for (i = 0; i < valid_keywords.length; i++) {
							 | 
						||
| 
								 | 
							
								    if (valid_keywords[i].toLowerCase() === val) {
							 | 
						||
| 
								 | 
							
								      return valid_keywords[i];
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  return undefined;
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// utility to translate from border-width to borderWidth
							 | 
						||
| 
								 | 
							
								var dashedToCamelCase = function(dashed) {
							 | 
						||
| 
								 | 
							
								  var i;
							 | 
						||
| 
								 | 
							
								  var camel = '';
							 | 
						||
| 
								 | 
							
								  var nextCap = false;
							 | 
						||
| 
								 | 
							
								  for (i = 0; i < dashed.length; i++) {
							 | 
						||
| 
								 | 
							
								    if (dashed[i] !== '-') {
							 | 
						||
| 
								 | 
							
								      camel += nextCap ? dashed[i].toUpperCase() : dashed[i];
							 | 
						||
| 
								 | 
							
								      nextCap = false;
							 | 
						||
| 
								 | 
							
								    } else {
							 | 
						||
| 
								 | 
							
								      nextCap = true;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  return camel;
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								exports.dashedToCamelCase = dashedToCamelCase;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								var is_space = /\s/;
							 | 
						||
| 
								 | 
							
								var opening_deliminators = ['"', "'", '('];
							 | 
						||
| 
								 | 
							
								var closing_deliminators = ['"', "'", ')'];
							 | 
						||
| 
								 | 
							
								// this splits on whitespace, but keeps quoted and parened parts together
							 | 
						||
| 
								 | 
							
								var getParts = function(str) {
							 | 
						||
| 
								 | 
							
								  var deliminator_stack = [];
							 | 
						||
| 
								 | 
							
								  var length = str.length;
							 | 
						||
| 
								 | 
							
								  var i;
							 | 
						||
| 
								 | 
							
								  var parts = [];
							 | 
						||
| 
								 | 
							
								  var current_part = '';
							 | 
						||
| 
								 | 
							
								  var opening_index;
							 | 
						||
| 
								 | 
							
								  var closing_index;
							 | 
						||
| 
								 | 
							
								  for (i = 0; i < length; i++) {
							 | 
						||
| 
								 | 
							
								    opening_index = opening_deliminators.indexOf(str[i]);
							 | 
						||
| 
								 | 
							
								    closing_index = closing_deliminators.indexOf(str[i]);
							 | 
						||
| 
								 | 
							
								    if (is_space.test(str[i])) {
							 | 
						||
| 
								 | 
							
								      if (deliminator_stack.length === 0) {
							 | 
						||
| 
								 | 
							
								        if (current_part !== '') {
							 | 
						||
| 
								 | 
							
								          parts.push(current_part);
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								        current_part = '';
							 | 
						||
| 
								 | 
							
								      } else {
							 | 
						||
| 
								 | 
							
								        current_part += str[i];
							 | 
						||
| 
								 | 
							
								      }
							 | 
						||
| 
								 | 
							
								    } else {
							 | 
						||
| 
								 | 
							
								      if (str[i] === '\\') {
							 | 
						||
| 
								 | 
							
								        i++;
							 | 
						||
| 
								 | 
							
								        current_part += str[i];
							 | 
						||
| 
								 | 
							
								      } else {
							 | 
						||
| 
								 | 
							
								        current_part += str[i];
							 | 
						||
| 
								 | 
							
								        if (
							 | 
						||
| 
								 | 
							
								          closing_index !== -1 &&
							 | 
						||
| 
								 | 
							
								          closing_index === deliminator_stack[deliminator_stack.length - 1]
							 | 
						||
| 
								 | 
							
								        ) {
							 | 
						||
| 
								 | 
							
								          deliminator_stack.pop();
							 | 
						||
| 
								 | 
							
								        } else if (opening_index !== -1) {
							 | 
						||
| 
								 | 
							
								          deliminator_stack.push(opening_index);
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								      }
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  if (current_part !== '') {
							 | 
						||
| 
								 | 
							
								    parts.push(current_part);
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  return parts;
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/*
							 | 
						||
| 
								 | 
							
								 * this either returns undefined meaning that it isn't valid
							 | 
						||
| 
								 | 
							
								 * or returns an object where the keys are dashed short
							 | 
						||
| 
								 | 
							
								 * hand properties and the values are the values to set
							 | 
						||
| 
								 | 
							
								 * on them
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								exports.shorthandParser = function parse(v, shorthand_for) {
							 | 
						||
| 
								 | 
							
								  var obj = {};
							 | 
						||
| 
								 | 
							
								  var type = exports.valueType(v);
							 | 
						||
| 
								 | 
							
								  if (type === exports.TYPES.NULL_OR_EMPTY_STR) {
							 | 
						||
| 
								 | 
							
								    Object.keys(shorthand_for).forEach(function(property) {
							 | 
						||
| 
								 | 
							
								      obj[property] = '';
							 | 
						||
| 
								 | 
							
								    });
							 | 
						||
| 
								 | 
							
								    return obj;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  if (typeof v === 'number') {
							 | 
						||
| 
								 | 
							
								    v = v.toString();
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  if (typeof v !== 'string') {
							 | 
						||
| 
								 | 
							
								    return undefined;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  if (v.toLowerCase() === 'inherit') {
							 | 
						||
| 
								 | 
							
								    return {};
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  var parts = getParts(v);
							 | 
						||
| 
								 | 
							
								  var valid = true;
							 | 
						||
| 
								 | 
							
								  parts.forEach(function(part, i) {
							 | 
						||
| 
								 | 
							
								    var part_valid = false;
							 | 
						||
| 
								 | 
							
								    Object.keys(shorthand_for).forEach(function(property) {
							 | 
						||
| 
								 | 
							
								      if (shorthand_for[property].isValid(part, i)) {
							 | 
						||
| 
								 | 
							
								        part_valid = true;
							 | 
						||
| 
								 | 
							
								        obj[property] = part;
							 | 
						||
| 
								 | 
							
								      }
							 | 
						||
| 
								 | 
							
								    });
							 | 
						||
| 
								 | 
							
								    valid = valid && part_valid;
							 | 
						||
| 
								 | 
							
								  });
							 | 
						||
| 
								 | 
							
								  if (!valid) {
							 | 
						||
| 
								 | 
							
								    return undefined;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  return obj;
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								exports.shorthandSetter = function(property, shorthand_for) {
							 | 
						||
| 
								 | 
							
								  return function(v) {
							 | 
						||
| 
								 | 
							
								    var obj = exports.shorthandParser(v, shorthand_for);
							 | 
						||
| 
								 | 
							
								    if (obj === undefined) {
							 | 
						||
| 
								 | 
							
								      return;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    //console.log('shorthandSetter for:', property, 'obj:', obj);
							 | 
						||
| 
								 | 
							
								    Object.keys(obj).forEach(function(subprop) {
							 | 
						||
| 
								 | 
							
								      // in case subprop is an implicit property, this will clear
							 | 
						||
| 
								 | 
							
								      // *its* subpropertiesX
							 | 
						||
| 
								 | 
							
								      var camel = dashedToCamelCase(subprop);
							 | 
						||
| 
								 | 
							
								      this[camel] = obj[subprop];
							 | 
						||
| 
								 | 
							
								      // in case it gets translated into something else (0 -> 0px)
							 | 
						||
| 
								 | 
							
								      obj[subprop] = this[camel];
							 | 
						||
| 
								 | 
							
								      this.removeProperty(subprop);
							 | 
						||
| 
								 | 
							
								      // don't add in empty properties
							 | 
						||
| 
								 | 
							
								      if (obj[subprop] !== '') {
							 | 
						||
| 
								 | 
							
								        this._values[subprop] = obj[subprop];
							 | 
						||
| 
								 | 
							
								      }
							 | 
						||
| 
								 | 
							
								    }, this);
							 | 
						||
| 
								 | 
							
								    Object.keys(shorthand_for).forEach(function(subprop) {
							 | 
						||
| 
								 | 
							
								      if (!obj.hasOwnProperty(subprop)) {
							 | 
						||
| 
								 | 
							
								        this.removeProperty(subprop);
							 | 
						||
| 
								 | 
							
								        delete this._values[subprop];
							 | 
						||
| 
								 | 
							
								      }
							 | 
						||
| 
								 | 
							
								    }, this);
							 | 
						||
| 
								 | 
							
								    // in case the value is something like 'none' that removes all values,
							 | 
						||
| 
								 | 
							
								    // check that the generated one is not empty, first remove the property
							 | 
						||
| 
								 | 
							
								    // if it already exists, then call the shorthandGetter, if it's an empty
							 | 
						||
| 
								 | 
							
								    // string, don't set the property
							 | 
						||
| 
								 | 
							
								    this.removeProperty(property);
							 | 
						||
| 
								 | 
							
								    var calculated = exports.shorthandGetter(property, shorthand_for).call(this);
							 | 
						||
| 
								 | 
							
								    if (calculated !== '') {
							 | 
						||
| 
								 | 
							
								      this._setProperty(property, calculated);
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								  };
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								exports.shorthandGetter = function(property, shorthand_for) {
							 | 
						||
| 
								 | 
							
								  return function() {
							 | 
						||
| 
								 | 
							
								    if (this._values[property] !== undefined) {
							 | 
						||
| 
								 | 
							
								      return this.getPropertyValue(property);
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    return Object.keys(shorthand_for)
							 | 
						||
| 
								 | 
							
								      .map(function(subprop) {
							 | 
						||
| 
								 | 
							
								        return this.getPropertyValue(subprop);
							 | 
						||
| 
								 | 
							
								      }, this)
							 | 
						||
| 
								 | 
							
								      .filter(function(value) {
							 | 
						||
| 
								 | 
							
								        return value !== '';
							 | 
						||
| 
								 | 
							
								      })
							 | 
						||
| 
								 | 
							
								      .join(' ');
							 | 
						||
| 
								 | 
							
								  };
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// isValid(){1,4} | inherit
							 | 
						||
| 
								 | 
							
								// if one, it applies to all
							 | 
						||
| 
								 | 
							
								// if two, the first applies to the top and bottom, and the second to left and right
							 | 
						||
| 
								 | 
							
								// if three, the first applies to the top, the second to left and right, the third bottom
							 | 
						||
| 
								 | 
							
								// if four, top, right, bottom, left
							 | 
						||
| 
								 | 
							
								exports.implicitSetter = function(property_before, property_after, isValid, parser) {
							 | 
						||
| 
								 | 
							
								  property_after = property_after || '';
							 | 
						||
| 
								 | 
							
								  if (property_after !== '') {
							 | 
						||
| 
								 | 
							
								    property_after = '-' + property_after;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  var part_names = ['top', 'right', 'bottom', 'left'];
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  return function(v) {
							 | 
						||
| 
								 | 
							
								    if (typeof v === 'number') {
							 | 
						||
| 
								 | 
							
								      v = v.toString();
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    if (typeof v !== 'string') {
							 | 
						||
| 
								 | 
							
								      return undefined;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    var parts;
							 | 
						||
| 
								 | 
							
								    if (v.toLowerCase() === 'inherit' || v === '') {
							 | 
						||
| 
								 | 
							
								      parts = [v];
							 | 
						||
| 
								 | 
							
								    } else {
							 | 
						||
| 
								 | 
							
								      parts = getParts(v);
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    if (parts.length < 1 || parts.length > 4) {
							 | 
						||
| 
								 | 
							
								      return undefined;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    if (!parts.every(isValid)) {
							 | 
						||
| 
								 | 
							
								      return undefined;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    parts = parts.map(function(part) {
							 | 
						||
| 
								 | 
							
								      return parser(part);
							 | 
						||
| 
								 | 
							
								    });
							 | 
						||
| 
								 | 
							
								    this._setProperty(property_before + property_after, parts.join(' '));
							 | 
						||
| 
								 | 
							
								    if (parts.length === 1) {
							 | 
						||
| 
								 | 
							
								      parts[1] = parts[0];
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    if (parts.length === 2) {
							 | 
						||
| 
								 | 
							
								      parts[2] = parts[0];
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    if (parts.length === 3) {
							 | 
						||
| 
								 | 
							
								      parts[3] = parts[1];
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    for (var i = 0; i < 4; i++) {
							 | 
						||
| 
								 | 
							
								      var property = property_before + '-' + part_names[i] + property_after;
							 | 
						||
| 
								 | 
							
								      this.removeProperty(property);
							 | 
						||
| 
								 | 
							
								      if (parts[i] !== '') {
							 | 
						||
| 
								 | 
							
								        this._values[property] = parts[i];
							 | 
						||
| 
								 | 
							
								      }
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    return v;
							 | 
						||
| 
								 | 
							
								  };
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								//
							 | 
						||
| 
								 | 
							
								//  Companion to implicitSetter, but for the individual parts.
							 | 
						||
| 
								 | 
							
								//  This sets the individual value, and checks to see if all four
							 | 
						||
| 
								 | 
							
								//  sub-parts are set.  If so, it sets the shorthand version and removes
							 | 
						||
| 
								 | 
							
								//  the individual parts from the cssText.
							 | 
						||
| 
								 | 
							
								//
							 | 
						||
| 
								 | 
							
								exports.subImplicitSetter = function(prefix, part, isValid, parser) {
							 | 
						||
| 
								 | 
							
								  var property = prefix + '-' + part;
							 | 
						||
| 
								 | 
							
								  var subparts = [prefix + '-top', prefix + '-right', prefix + '-bottom', prefix + '-left'];
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  return function(v) {
							 | 
						||
| 
								 | 
							
								    if (typeof v === 'number') {
							 | 
						||
| 
								 | 
							
								      v = v.toString();
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    if (typeof v !== 'string') {
							 | 
						||
| 
								 | 
							
								      return undefined;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    if (!isValid(v)) {
							 | 
						||
| 
								 | 
							
								      return undefined;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    v = parser(v);
							 | 
						||
| 
								 | 
							
								    this._setProperty(property, v);
							 | 
						||
| 
								 | 
							
								    var parts = [];
							 | 
						||
| 
								 | 
							
								    for (var i = 0; i < 4; i++) {
							 | 
						||
| 
								 | 
							
								      if (this._values[subparts[i]] == null || this._values[subparts[i]] === '') {
							 | 
						||
| 
								 | 
							
								        break;
							 | 
						||
| 
								 | 
							
								      }
							 | 
						||
| 
								 | 
							
								      parts.push(this._values[subparts[i]]);
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    if (parts.length === 4) {
							 | 
						||
| 
								 | 
							
								      for (i = 0; i < 4; i++) {
							 | 
						||
| 
								 | 
							
								        this.removeProperty(subparts[i]);
							 | 
						||
| 
								 | 
							
								        this._values[subparts[i]] = parts[i];
							 | 
						||
| 
								 | 
							
								      }
							 | 
						||
| 
								 | 
							
								      this._setProperty(prefix, parts.join(' '));
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    return v;
							 | 
						||
| 
								 | 
							
								  };
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								var camel_to_dashed = /[A-Z]/g;
							 | 
						||
| 
								 | 
							
								var first_segment = /^\([^-]\)-/;
							 | 
						||
| 
								 | 
							
								var vendor_prefixes = ['o', 'moz', 'ms', 'webkit'];
							 | 
						||
| 
								 | 
							
								exports.camelToDashed = function(camel_case) {
							 | 
						||
| 
								 | 
							
								  var match;
							 | 
						||
| 
								 | 
							
								  var dashed = camel_case.replace(camel_to_dashed, '-$&').toLowerCase();
							 | 
						||
| 
								 | 
							
								  match = dashed.match(first_segment);
							 | 
						||
| 
								 | 
							
								  if (match && vendor_prefixes.indexOf(match[1]) !== -1) {
							 | 
						||
| 
								 | 
							
								    dashed = '-' + dashed;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  return dashed;
							 | 
						||
| 
								 | 
							
								};
							 |