| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130 | 'use strict';const postcssValueParser = require('postcss-value-parser');const isNumbery = require('./isNumbery');const isStandardSyntaxValue = require('./isStandardSyntaxValue');const isValidFontSize = require('./isValidFontSize');const isVariable = require('./isVariable');const { assert } = require('./validateTypes');const {	basicKeywords,	fontFamilyKeywords,	fontShorthandKeywords,} = require('../reference/keywords');const nodeTypesToCheck = new Set(['word', 'string', 'space', 'div']);/** @typedef {import('postcss-value-parser').Node} Node *//** * * @param {Node} firstNode * @param {Node} secondNode * @param {string | null} charactersBetween * * @returns {Node} */function joinValueNodes(firstNode, secondNode, charactersBetween) {	firstNode.value = firstNode.value + charactersBetween + secondNode.value;	return firstNode;}/** * Get the font-families within a `font` shorthand property value. * * @param {string} value * @returns {Node[]} Collection font-family nodes */module.exports = function findFontFamily(value) {	/** @type {Node[]} */	const fontFamilies = [];	const valueNodes = postcssValueParser(value);	const { nodes: children } = valueNodes;	// Handle `inherit`, `initial` and etc	if (children.length === 1 && children[0] && basicKeywords.has(children[0].value.toLowerCase())) {		return [children[0]];	}	let needMergeNodesByValue = false;	/** @type {string | null} */	let mergeCharacters = null;	valueNodes.walk((valueNode, index, nodes) => {		if (valueNode.type === 'function') {			return false;		}		if (!nodeTypesToCheck.has(valueNode.type)) {			return;		}		const valueLowerCase = valueNode.value.toLowerCase();		// Ignore non standard syntax		if (!isStandardSyntaxValue(valueLowerCase)) {			return;		}		// Ignore variables		if (isVariable(valueLowerCase)) {			return;		}		// Ignore keywords for other font parts		if (fontShorthandKeywords.has(valueLowerCase) && !fontFamilyKeywords.has(valueLowerCase)) {			return;		}		// Ignore font-sizes		if (isValidFontSize(valueNode.value)) {			return;		}		const prevNode = nodes[index - 1];		const prevPrevNode = nodes[index - 2];		// Ignore anything come after a <font-size>/, because it's a line-height		if (prevNode && prevNode.value === '/' && prevPrevNode && isValidFontSize(prevPrevNode.value)) {			return;		}		// Ignore number values		if (isNumbery(valueLowerCase)) {			return;		}		// Detect when a space or comma is dividing a list of font-families, and save the joining character.		if (			(valueNode.type === 'space' || (valueNode.type === 'div' && valueNode.value !== ',')) &&			fontFamilies.length !== 0		) {			needMergeNodesByValue = true;			mergeCharacters = valueNode.value;			return;		}		if (valueNode.type === 'space' || valueNode.type === 'div') {			return;		}		const fontFamily = valueNode;		if (needMergeNodesByValue) {			const lastFontFamily = fontFamilies[fontFamilies.length - 1];			assert(lastFontFamily);			joinValueNodes(lastFontFamily, fontFamily, mergeCharacters);			needMergeNodesByValue = false;			mergeCharacters = null;		} else {			fontFamilies.push(fontFamily);		}	});	return fontFamilies;};
 |