|
|
'use strict';
|
|
|
|
|
|
Object.defineProperty(exports, "__esModule", {
|
|
|
value: true
|
|
|
});
|
|
|
|
|
|
exports.default = function (nodes, opts) {
|
|
|
let family = [];
|
|
|
let last = null;
|
|
|
let i, max;
|
|
|
|
|
|
nodes.forEach((node, index, arr) => {
|
|
|
if (node.type === 'string' || node.type === 'function') {
|
|
|
family.push(node);
|
|
|
} else if (node.type === 'word') {
|
|
|
if (!last) {
|
|
|
last = { type: 'word', value: '' };
|
|
|
family.push(last);
|
|
|
}
|
|
|
|
|
|
last.value += node.value;
|
|
|
} else if (node.type === 'space') {
|
|
|
if (last && index !== arr.length - 1) {
|
|
|
last.value += ' ';
|
|
|
}
|
|
|
} else {
|
|
|
last = null;
|
|
|
}
|
|
|
});
|
|
|
|
|
|
family = family.map(node => {
|
|
|
if (node.type === 'string') {
|
|
|
const isKeyword = regexKeyword.test(node.value);
|
|
|
|
|
|
if (!opts.removeQuotes || isKeyword || /[0-9]/.test(node.value.slice(0, 1))) {
|
|
|
return (0, _postcssValueParser.stringify)(node);
|
|
|
}
|
|
|
|
|
|
let escaped = escapeIdentifierSequence(node.value);
|
|
|
|
|
|
if (escaped.length < node.value.length + 2) {
|
|
|
return escaped;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
return (0, _postcssValueParser.stringify)(node);
|
|
|
});
|
|
|
|
|
|
if (opts.removeAfterKeyword) {
|
|
|
for (i = 0, max = family.length; i < max; i += 1) {
|
|
|
if (~genericFontFamilykeywords.indexOf(family[i].toLowerCase())) {
|
|
|
family = family.slice(0, i + 1);
|
|
|
break;
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
if (opts.removeDuplicates) {
|
|
|
family = uniqs(family);
|
|
|
}
|
|
|
|
|
|
return [{
|
|
|
type: 'word',
|
|
|
value: family.join()
|
|
|
}];
|
|
|
};
|
|
|
|
|
|
var _postcssValueParser = require('postcss-value-parser');
|
|
|
|
|
|
var _uniqs = require('./uniqs');
|
|
|
|
|
|
var _uniqs2 = _interopRequireDefault(_uniqs);
|
|
|
|
|
|
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
|
|
|
|
const uniqs = (0, _uniqs2.default)('monospace');
|
|
|
const globalKeywords = ['inherit', 'initial', 'unset'];
|
|
|
const genericFontFamilykeywords = ['sans-serif', 'serif', 'fantasy', 'cursive', 'monospace', 'system-ui'];
|
|
|
|
|
|
function makeArray(value, length) {
|
|
|
let array = [];
|
|
|
while (length--) {
|
|
|
array[length] = value;
|
|
|
}
|
|
|
return array;
|
|
|
}
|
|
|
|
|
|
const regexSimpleEscapeCharacters = /[ !"#$%&'()*+,.\/;<=>?@\[\\\]^`{|}~]/;
|
|
|
|
|
|
function escape(string, escapeForString) {
|
|
|
let counter = 0;
|
|
|
let character = null;
|
|
|
let charCode = null;
|
|
|
let value = null;
|
|
|
let output = '';
|
|
|
|
|
|
while (counter < string.length) {
|
|
|
character = string.charAt(counter++);
|
|
|
charCode = character.charCodeAt();
|
|
|
|
|
|
// \r is already tokenized away at this point
|
|
|
// `:` can be escaped as `\:`, but that fails in IE < 8
|
|
|
if (!escapeForString && /[\t\n\v\f:]/.test(character)) {
|
|
|
value = '\\' + charCode.toString(16) + ' ';
|
|
|
} else if (!escapeForString && regexSimpleEscapeCharacters.test(character)) {
|
|
|
value = '\\' + character;
|
|
|
} else {
|
|
|
value = character;
|
|
|
}
|
|
|
|
|
|
output += value;
|
|
|
}
|
|
|
|
|
|
if (!escapeForString) {
|
|
|
if (/^-[-\d]/.test(output)) {
|
|
|
output = '\\-' + output.slice(1);
|
|
|
}
|
|
|
|
|
|
const firstChar = string.charAt(0);
|
|
|
|
|
|
if (/\d/.test(firstChar)) {
|
|
|
output = '\\3' + firstChar + ' ' + output.slice(1);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
return output;
|
|
|
}
|
|
|
|
|
|
const regexKeyword = new RegExp(genericFontFamilykeywords.concat(globalKeywords).join('|'), 'i');
|
|
|
const regexInvalidIdentifier = /^(-?\d|--)/;
|
|
|
const regexSpaceAtStart = /^\x20/;
|
|
|
const regexWhitespace = /[\t\n\f\r\x20]/g;
|
|
|
const regexIdentifierCharacter = /^[a-zA-Z\d\xa0-\uffff_-]+$/;
|
|
|
const regexConsecutiveSpaces = /(\\(?:[a-fA-F0-9]{1,6}\x20|\x20))?(\x20{2,})/g;
|
|
|
const regexTrailingEscape = /\\[a-fA-F0-9]{0,6}\x20$/;
|
|
|
const regexTrailingSpace = /\x20$/;
|
|
|
|
|
|
function escapeIdentifierSequence(string) {
|
|
|
let identifiers = string.split(regexWhitespace);
|
|
|
let index = 0;
|
|
|
let result = [];
|
|
|
let escapeResult;
|
|
|
|
|
|
while (index < identifiers.length) {
|
|
|
let subString = identifiers[index++];
|
|
|
|
|
|
if (subString === '') {
|
|
|
result.push(subString);
|
|
|
continue;
|
|
|
}
|
|
|
|
|
|
escapeResult = escape(subString, false);
|
|
|
|
|
|
if (regexIdentifierCharacter.test(subString)) {
|
|
|
// the font family name part consists of allowed characters exclusively
|
|
|
if (regexInvalidIdentifier.test(subString)) {
|
|
|
// the font family name part starts with two hyphens, a digit, or a
|
|
|
// hyphen followed by a digit
|
|
|
if (index === 1) {
|
|
|
// if this is the first item
|
|
|
result.push(escapeResult);
|
|
|
} else {
|
|
|
// if it’s not the first item, we can simply escape the space
|
|
|
// between the two identifiers to merge them into a single
|
|
|
// identifier rather than escaping the start characters of the
|
|
|
// second identifier
|
|
|
result[index - 2] += '\\';
|
|
|
result.push(escape(subString, true));
|
|
|
}
|
|
|
} else {
|
|
|
// the font family name part doesn’t start with two hyphens, a digit,
|
|
|
// or a hyphen followed by a digit
|
|
|
result.push(escapeResult);
|
|
|
}
|
|
|
} else {
|
|
|
// the font family name part contains invalid identifier characters
|
|
|
result.push(escapeResult);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
result = result.join(' ').replace(regexConsecutiveSpaces, ($0, $1, $2) => {
|
|
|
const spaceCount = $2.length;
|
|
|
const escapesNeeded = Math.floor(spaceCount / 2);
|
|
|
const array = makeArray('\\ ', escapesNeeded);
|
|
|
|
|
|
if (spaceCount % 2) {
|
|
|
array[escapesNeeded - 1] += '\\ ';
|
|
|
}
|
|
|
|
|
|
return ($1 || '') + ' ' + array.join(' ');
|
|
|
});
|
|
|
|
|
|
// Escape trailing spaces unless they’re already part of an escape
|
|
|
if (regexTrailingSpace.test(result) && !regexTrailingEscape.test(result)) {
|
|
|
result = result.replace(regexTrailingSpace, '\\ ');
|
|
|
}
|
|
|
|
|
|
if (regexSpaceAtStart.test(result)) {
|
|
|
result = '\\ ' + result.slice(1);
|
|
|
}
|
|
|
|
|
|
return result;
|
|
|
}
|
|
|
|
|
|
;
|
|
|
module.exports = exports['default']; |