123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206 |
- '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'];
|