1
0
mirror of https://github.com/simple-icons/simple-icons.git synced 2025-01-05 01:20:39 +02:00
simple-icons/svglint.config.mjs

1072 lines
37 KiB
JavaScript
Raw Normal View History

2024-06-06 14:40:35 +02:00
/**
* @file
* Linting rules for SVGLint to check SVG icons.
*/
import fs from 'node:fs/promises';
import path from 'node:path';
2024-06-06 14:40:35 +02:00
import {svgPathBbox} from 'svg-path-bbox';
2024-03-24 19:38:18 +02:00
import parsePath from 'svg-path-segments';
import svgpath from 'svgpath';
import {
SVG_PATH_REGEX,
getDirnameFromImportMeta,
getIconsData,
htmlFriendlyToTitle,
2023-04-19 15:23:13 +02:00
} from './sdk.mjs';
const __dirname = getDirnameFromImportMeta(import.meta.url);
const htmlNamedEntitiesFile = path.join(
__dirname,
'node_modules',
'named-html-entities-json',
'index.json',
);
const icons = await getIconsData();
const htmlNamedEntities = JSON.parse(
await fs.readFile(htmlNamedEntitiesFile, 'utf8'),
);
const svgRegexp =
2024-03-24 19:38:18 +02:00
/^<svg( \S*=".*"){3}><title>.*<\/title><path d=".*"\/><\/svg>$/;
const negativeZerosRegexp = /-0(?=[^.]|[\s\d\w]|$)/g;
const iconSize = 24;
const iconTargetCenter = iconSize / 2;
const iconFloatPrecision = 3;
const iconMaxFloatPrecision = 5;
const iconTolerance = 0.001;
2024-06-06 14:40:35 +02:00
/**
* Remove leading zeros from a number as a string.
* @param {number | string} numberOrString The number or string to remove leading zeros from.
* @returns {string} The number as a string without leading zeros.
*/
const removeLeadingZeros = (numberOrString) => {
2024-03-24 19:38:18 +02:00
// Convert 0.03 to '.03'
2024-06-06 14:40:35 +02:00
return numberOrString.toString().replace(/^(-?)(0)(\.?.+)/, '$1$3');
};
/**
* Given three points, returns if the middle one (x2, y2) is collinear
* to the line formed by the two limit points.
2024-06-06 14:40:35 +02:00
* @param {number} x1 The x coordinate of the first point.
* @param {number} y1 The y coordinate of the first point.
* @param {number} x2 The x coordinate of the second point.
* @param {number} y2 The y coordinate of the second point.
* @param {number} x3 The x coordinate of the third point.
* @param {number} y3 The y coordinate of the third point.
* @returns {boolean} Whether the middle point is collinear to the line.
*/
2024-03-24 19:38:18 +02:00
// eslint-disable-next-line max-params
const collinear = (x1, y1, x2, y2, x3, y3) => {
return x1 * (y2 - y3) + x2 * (y3 - y1) + x3 * (y1 - y2) === 0;
};
/**
* Returns the number of digits after the decimal point.
2024-06-06 14:40:35 +02:00
* @param {number} number_ The number to count the decimals of.
* @returns {number} The number of digits after the decimal point.
*/
2024-03-24 19:38:18 +02:00
const countDecimals = (number_) => {
if (number_ && number_ % 1) {
const [base, op, trail] = number_.toExponential().split(/e([+-])/);
const elen = Number.parseInt(trail, 10);
const index = base.indexOf('.');
return index === -1
? elen
2024-03-24 19:38:18 +02:00
: base.length - index - 1 + (op === '+' ? -elen : elen);
}
2024-03-24 19:38:18 +02:00
return 0;
};
/**
* Get the index at which the first path value of an SVG starts.
2024-06-06 14:40:35 +02:00
* @param {string} svgFileContent The raw SVG as text.
* @returns {number} The index at which the path value starts.
*/
const getPathDIndex = (svgFileContent) => {
const pathDStart = '<path d="';
return svgFileContent.indexOf(pathDStart) + pathDStart.length;
};
/**
* Get the index at which the text of the first `<title></title>` tag starts.
2024-06-06 14:40:35 +02:00
* @param {string} svgFileContent The raw SVG as text.
* @returns {number} The index at which the title text starts.
*/
const getTitleTextIndex = (svgFileContent) => {
const titleStart = '<title>';
return svgFileContent.indexOf(titleStart) + titleStart.length;
};
2024-06-06 14:40:35 +02:00
/**
* Shorten a string with ellipsis if it exceeds 20 characters.
* @param {string} string_ The string to shorten.
* @returns {string} The shortened string.
*/
2024-03-24 19:38:18 +02:00
const maybeShortenedWithEllipsis = (string_) => {
return string_.length > 20 ? `${string_.slice(0, 20)}...` : string_;
};
/**
* Check if a string is a number.
* @param {string} string_ The string to check.
* @returns {boolean} Whether the string is a number.
*/
const isNumber = (string_) =>
[...string_].every((character) => '0123456789'.includes(character));
/**
* Memoize a function which accepts a single argument.
* A second argument can be passed to be used as key.
2024-06-06 14:40:35 +02:00
* @param {(arg0: any) => any} function_ The function to memoize.
* @returns {(arg0: any) => any} The memoized function.
*/
2024-03-24 19:38:18 +02:00
const memoize = (function_) => {
2024-06-06 14:40:35 +02:00
/** @type {{ [key: string]: any }} */
const results = {};
2024-06-06 14:40:35 +02:00
/**
* Memoized function.
* @param {any} argument The argument to memoize.
* @returns {any} The result of the memoized function.
*/
return (argument) => {
results[argument] ||= function_(argument);
2024-03-24 19:38:18 +02:00
2024-06-06 14:40:35 +02:00
return results[argument];
};
};
2024-06-06 14:40:35 +02:00
/** @typedef {import('cheerio').Cheerio<import('domhandler').Document>} Cheerio */
/** @type {($icon: Cheerio) => string} */
const getIconPath = memoize(($icon) => $icon.find('path').attr('d'));
/** @type {(iconPath: string) => import('svg-path-segments').Segment[]} */
const getIconPathSegments = memoize((iconPath) => parsePath(iconPath));
2024-06-06 14:40:35 +02:00
/** @type {(iconPath: string) => import('svg-path-bbox').BBox} */
const getIconPathBbox = memoize((iconPath) => svgPathBbox(iconPath));
2024-06-06 14:40:35 +02:00
/** @type {import('svglint').Config} */
const config = {
rules: {
elm: {
svg: 1,
'svg > title': 1,
'svg > path': 1,
'*': false,
},
attr: [
{
2024-03-24 19:38:18 +02:00
// Ensure that the SVG element has the appropriate attributes
// alphabetically ordered
role: 'img',
viewBox: `0 0 ${iconSize} ${iconSize}`,
xmlns: 'http://www.w3.org/2000/svg',
'rule::selector': 'svg',
'rule::whitelist': true,
'rule::order': true,
},
{
2024-03-24 19:38:18 +02:00
// Ensure that the title element has the appropriate attribute
'rule::selector': 'svg > title',
'rule::whitelist': true,
},
{
2024-03-24 19:38:18 +02:00
// Ensure that the path element only has the 'd' attribute
// (no style, opacity, etc.)
d: SVG_PATH_REGEX,
'rule::selector': 'svg > path',
'rule::whitelist': true,
},
],
custom: [
2024-06-06 14:40:35 +02:00
// eslint-disable-next-line complexity
(reporter, $, ast) => {
reporter.name = 'icon-title';
2024-03-24 19:38:18 +02:00
const iconTitleText = $.find('title').text();
const xmlNamedEntitiesCodepoints = [38, 60, 62];
const xmlNamedEntities = ['amp', 'lt', 'gt'];
let _validCodepointsRepr = true;
2024-03-24 19:38:18 +02:00
// Avoid character codepoints as hexadecimal representation
const hexadecimalCodepoints = [
...iconTitleText.matchAll(/&#x([A-Fa-f\d]+);/g),
];
if (hexadecimalCodepoints.length > 0) {
_validCodepointsRepr = false;
for (const match of hexadecimalCodepoints) {
const charHexReprIndex =
getTitleTextIndex(ast.source) + match.index + 1;
const charDec = Number.parseInt(match[1], 16);
let charRepr;
if (xmlNamedEntitiesCodepoints.includes(charDec)) {
charRepr = `&${
xmlNamedEntities[xmlNamedEntitiesCodepoints.indexOf(charDec)]
};`;
} else if (charDec < 128) {
charRepr = String.fromCodePoint(charDec);
} else {
charRepr = `&#${charDec};`;
}
reporter.error(
'Hexadecimal representation of encoded character' +
` "${match[0]}" found at index ${charHexReprIndex}:` +
` replace it with "${charRepr}".`,
);
}
}
2024-03-24 19:38:18 +02:00
// Avoid character codepoints as named entities
const namedEntitiesCodepoints = [
...iconTitleText.matchAll(/&([A-Za-z\d]+);/g),
];
if (namedEntitiesCodepoints.length > 0) {
for (const match of namedEntitiesCodepoints) {
const namedEntiyReprIndex =
getTitleTextIndex(ast.source) + match.index + 1;
if (!xmlNamedEntities.includes(match[1].toLowerCase())) {
_validCodepointsRepr = false;
const namedEntityJsRepr = htmlNamedEntities[match[1]];
let replacement;
if (
namedEntityJsRepr === undefined ||
2024-03-24 19:38:18 +02:00
namedEntityJsRepr.length !== 1
) {
replacement = 'its decimal or literal representation';
} else {
const namedEntityDec = namedEntityJsRepr.codePointAt(0);
2024-03-24 19:38:18 +02:00
replacement =
namedEntityDec < 128
? `"${namedEntityJsRepr}"`
: `"&#${namedEntityDec};"`;
}
reporter.error(
'Named entity representation of encoded character' +
` "${match[0]}" found at index ${namedEntiyReprIndex}.` +
` Replace it with ${replacement}.`,
);
}
}
}
if (_validCodepointsRepr) {
2024-03-24 19:38:18 +02:00
// Compare encoded title with original title and report error if not equal
const encodingMatches = [
...iconTitleText.matchAll(/&(#(\d+)|(amp|quot|lt|gt));/g),
];
const encodedBuf = [];
const indexesToIgnore = [];
for (const match of encodingMatches) {
for (let r = match.index; r < match.index + match[0].length; r++) {
indexesToIgnore.push(r);
}
}
for (let i = iconTitleText.length - 1; i >= 0; i--) {
if (indexesToIgnore.includes(i)) {
encodedBuf.unshift(iconTitleText[i]);
} else {
2024-03-24 19:38:18 +02:00
// Encode all non ascii characters plus "'&<> (XML named entities)
2024-06-06 14:40:35 +02:00
/** @type {number} */
// @ts-ignore
2024-03-24 19:38:18 +02:00
const charDecimalCode = iconTitleText.codePointAt(i);
if (charDecimalCode > 127) {
encodedBuf.unshift(`&#${charDecimalCode};`);
} else if (xmlNamedEntitiesCodepoints.includes(charDecimalCode)) {
encodedBuf.unshift(
`&${
xmlNamedEntities[
xmlNamedEntitiesCodepoints.indexOf(charDecimalCode)
]
};`,
);
} else {
encodedBuf.unshift(iconTitleText[i]);
}
}
}
2024-03-24 19:38:18 +02:00
const encodedIconTitleText = encodedBuf.join('');
if (encodedIconTitleText !== iconTitleText) {
_validCodepointsRepr = false;
reporter.error(
`Unencoded unicode characters found in title "${iconTitleText}":` +
` rewrite it as "${encodedIconTitleText}".`,
);
}
2024-03-24 19:38:18 +02:00
// Check if there are some other encoded characters in decimal notation
// which shouldn't be encoded
const numberMatches = encodingMatches.filter(
(m) => m[2] !== undefined && isNumber(m[2]),
);
for (const match of numberMatches) {
2024-03-24 19:38:18 +02:00
const decimalNumber = Number.parseInt(match[2], 10);
if (decimalNumber > 127) {
continue;
}
2024-03-24 19:38:18 +02:00
_validCodepointsRepr = false;
const decimalCodepointCharIndex =
getTitleTextIndex(ast.source) + match.index + 1;
2024-03-24 19:38:18 +02:00
let replacement;
if (xmlNamedEntitiesCodepoints.includes(decimalNumber)) {
replacement = `"&${
xmlNamedEntities[
xmlNamedEntitiesCodepoints.indexOf(decimalNumber)
]
};"`;
} else {
replacement = String.fromCodePoint(decimalNumber);
2024-03-24 19:38:18 +02:00
replacement = replacement === '"' ? `'"'` : `"${replacement}"`;
}
reporter.error(
`Unnecessary encoded character "${match[0]}" found` +
` at index ${decimalCodepointCharIndex}:` +
` replace it with ${replacement}.`,
);
}
if (_validCodepointsRepr) {
const iconName = htmlFriendlyToTitle(iconTitleText);
const iconExists = icons.some((icon) => icon.title === iconName);
if (!iconExists) {
reporter.error(
`No icon with title "${iconName}" found in simple-icons.json`,
);
}
}
}
},
2024-06-06 14:40:35 +02:00
(reporter, $) => {
reporter.name = 'icon-size';
2024-06-06 14:40:35 +02:00
const iconPath = getIconPath($);
const [minX, minY, maxX, maxY] = getIconPathBbox(iconPath);
2024-03-24 19:38:18 +02:00
const width = Number((maxX - minX).toFixed(iconFloatPrecision));
const height = Number((maxY - minY).toFixed(iconFloatPrecision));
if (width === 0 && height === 0) {
reporter.error(
'Path bounds were reported as 0 x 0; check if the path is valid',
);
} else if (width !== iconSize && height !== iconSize) {
reporter.error(
`Size of <path> must be exactly ${iconSize} in one dimension;` +
` the size is currently ${width} x ${height}`,
);
}
},
2024-06-06 14:40:35 +02:00
(reporter, $, ast) => {
reporter.name = 'icon-precision';
2024-06-06 14:40:35 +02:00
const iconPath = getIconPath($);
const segments = getIconPathSegments(iconPath);
for (const segment of segments) {
2024-06-06 14:40:35 +02:00
/** @type {number[]} */
// @ts-ignore
const numberParameters = segment.params.slice(1);
const precisionMax = Math.max(
2024-03-24 19:38:18 +02:00
// eslint-disable-next-line unicorn/no-array-callback-reference
2024-06-06 14:40:35 +02:00
...numberParameters.map(countDecimals),
);
if (precisionMax > iconMaxFloatPrecision) {
2024-03-24 19:38:18 +02:00
let errorMessage =
`found ${precisionMax} decimals in segment` +
2024-03-24 19:38:18 +02:00
` "${iconPath.slice(segment.start, segment.end)}"`;
if (segment.chain !== undefined) {
const readableChain = maybeShortenedWithEllipsis(
iconPath.slice(segment.chain.start, segment.chain.end),
);
2024-03-24 19:38:18 +02:00
errorMessage += ` of chain "${readableChain}"`;
}
2024-03-24 19:38:18 +02:00
errorMessage += ` at index ${
segment.start + getPathDIndex(ast.source)
}`;
reporter.error(
'Maximum precision should not be greater than' +
2024-03-24 19:38:18 +02:00
` ${iconMaxFloatPrecision}; ${errorMessage}`,
);
}
}
},
2024-06-06 14:40:35 +02:00
(reporter, $, ast) => {
reporter.name = 'ineffective-segments';
2024-06-06 14:40:35 +02:00
const iconPath = getIconPath($);
const segments = getIconPathSegments(iconPath);
2024-06-06 14:40:35 +02:00
/** @type {import('svg-path-segments').Segment[]} */
// TODO: svgpath does not includes the `segments` property on the interface,
2024-06-06 14:40:35 +02:00
// see https://github.com/fontello/svgpath/pull/67/files
//
/** @typedef {[string, ...number[]]} Segment */
/** @type {Segment[]} */
const absSegments =
// @ts-ignore
svgpath(iconPath).abs().unshort().segments;
const lowerMovementCommands = ['m', 'l'];
const lowerDirectionCommands = ['h', 'v'];
const lowerCurveCommand = 'c';
const lowerShorthandCurveCommand = 's';
const lowerCurveCommands = [
lowerCurveCommand,
lowerShorthandCurveCommand,
];
const upperMovementCommands = ['M', 'L'];
const upperHorDirectionCommand = 'H';
2024-03-24 19:38:18 +02:00
const upperVersionDirectionCommand = 'V';
const upperDirectionCommands = [
upperHorDirectionCommand,
2024-03-24 19:38:18 +02:00
upperVersionDirectionCommand,
];
const upperCurveCommand = 'C';
const upperShorthandCurveCommand = 'S';
const upperCurveCommands = [
upperCurveCommand,
upperShorthandCurveCommand,
];
const curveCommands = [...lowerCurveCommands, ...upperCurveCommands];
2024-03-24 19:38:18 +02:00
const commands = new Set([
...lowerMovementCommands,
...lowerDirectionCommands,
...upperMovementCommands,
...upperDirectionCommands,
...curveCommands,
2024-03-24 19:38:18 +02:00
]);
2024-06-06 14:40:35 +02:00
/**
* Check if a segment is ineffective.
* @param {import('svg-path-segments').Segment} segment The segment to check.
* @param {number} index The index of the segment in the path.
* @param {boolean} previousSegmentIsZ Whether the previous segment is a Z command.
* @returns {boolean} Whether the segment is ineffective.
*/
// eslint-disable-next-line complexity
const isInvalidSegment = (segment, index, previousSegmentIsZ) => {
const [command, x1Coord, y1Coord, ...rest] = segment.params;
2024-03-24 19:38:18 +02:00
if (commands.has(command)) {
// Relative directions (h or v) having a length of 0
if (lowerDirectionCommands.includes(command) && x1Coord === 0) {
return true;
}
2024-03-24 19:38:18 +02:00
// Relative movement (m or l) having a distance of 0
if (
index > 0 &&
lowerMovementCommands.includes(command) &&
x1Coord === 0 &&
y1Coord === 0
) {
// When the path is closed (z), the new segment can start with
// a relative placement (m) as if it were absolute (M)
return command.toLowerCase() === 'm' ? !previousSegmentIsZ : true;
}
2024-03-24 19:38:18 +02:00
if (
lowerCurveCommands.includes(command) &&
x1Coord === 0 &&
y1Coord === 0
) {
const [x2Coord, y2Coord] = rest;
if (
// Relative shorthand curve (s) having a control point of 0
command === lowerShorthandCurveCommand ||
// Relative bézier curve (c) having control points of 0
(command === lowerCurveCommand &&
x2Coord === 0 &&
y2Coord === 0)
) {
return true;
}
}
2024-03-24 19:38:18 +02:00
if (index > 0) {
2024-03-24 19:38:18 +02:00
let [yPreviousCoord, xPreviousCoord] = [
...absSegments[index - 1],
].reverse();
// If the previous command was a direction one,
// we need to iterate back until we find the missing coordinates
// @ts-ignore
2024-03-24 19:38:18 +02:00
if (upperDirectionCommands.includes(xPreviousCoord)) {
// @ts-ignore
2024-03-24 19:38:18 +02:00
xPreviousCoord = undefined;
// @ts-ignore
2024-03-24 19:38:18 +02:00
yPreviousCoord = undefined;
let index_ = index;
while (
2024-03-24 19:38:18 +02:00
--index_ > 0 &&
(xPreviousCoord === undefined || yPreviousCoord === undefined)
) {
2024-03-24 19:38:18 +02:00
let [yPreviousCoordDeep, xPreviousCoordDeep] = [
...absSegments[index_],
].reverse();
2024-06-06 14:40:35 +02:00
// If the previous command was a horizontal movement,
// we need to consider the single coordinate as x
2024-03-24 19:38:18 +02:00
if (upperHorDirectionCommand === xPreviousCoordDeep) {
xPreviousCoordDeep = yPreviousCoordDeep;
// @ts-ignore
2024-03-24 19:38:18 +02:00
yPreviousCoordDeep = undefined;
}
2024-03-24 19:38:18 +02:00
// If the previous command was a vertical movement,
// we need to consider the single coordinate as y
2024-03-24 19:38:18 +02:00
if (upperVersionDirectionCommand === xPreviousCoordDeep) {
// @ts-ignore
2024-03-24 19:38:18 +02:00
xPreviousCoordDeep = undefined;
}
2024-03-24 19:38:18 +02:00
if (
2024-03-24 19:38:18 +02:00
xPreviousCoord === undefined &&
xPreviousCoordDeep !== undefined
) {
2024-03-24 19:38:18 +02:00
xPreviousCoord = xPreviousCoordDeep;
}
2024-03-24 19:38:18 +02:00
if (
2024-03-24 19:38:18 +02:00
yPreviousCoord === undefined &&
yPreviousCoordDeep !== undefined
) {
2024-03-24 19:38:18 +02:00
yPreviousCoord = yPreviousCoordDeep;
}
}
}
if (upperCurveCommands.includes(command)) {
const [x2Coord, y2Coord, xCoord, yCoord] = rest;
// Absolute shorthand curve (S) having
// the same coordinate as the previous segment
// and a control point equal to the ending point
if (
upperShorthandCurveCommand === command &&
2024-03-24 19:38:18 +02:00
x1Coord === xPreviousCoord &&
y1Coord === yPreviousCoord &&
x1Coord === x2Coord &&
y1Coord === y2Coord
) {
return true;
}
2024-03-24 19:38:18 +02:00
// Absolute bézier curve (C) having
// the same coordinate as the previous segment
// and last control point equal to the ending point
if (
upperCurveCommand === command &&
2024-03-24 19:38:18 +02:00
x1Coord === xPreviousCoord &&
y1Coord === yPreviousCoord &&
x2Coord === xCoord &&
y2Coord === yCoord
) {
return true;
}
}
return (
// Absolute horizontal direction (H) having
// the same x coordinate as the previous segment
(upperHorDirectionCommand === command &&
2024-03-24 19:38:18 +02:00
x1Coord === xPreviousCoord) ||
// Absolute vertical direction (V) having
// the same y coordinate as the previous segment
2024-03-24 19:38:18 +02:00
(upperVersionDirectionCommand === command &&
x1Coord === yPreviousCoord) ||
// Absolute movement (M or L) having the same
// coordinate as the previous segment
(upperMovementCommands.includes(command) &&
2024-03-24 19:38:18 +02:00
x1Coord === xPreviousCoord &&
y1Coord === yPreviousCoord)
);
}
}
2024-06-06 14:40:35 +02:00
return false;
};
for (let index = 0; index < segments.length; index++) {
const segment = segments[index];
const previousSegmentIsZ =
index > 0 && segments[index - 1].params[0].toLowerCase() === 'z';
2024-06-06 14:40:35 +02:00
if (isInvalidSegment(segment, index, previousSegmentIsZ)) {
2024-03-24 19:38:18 +02:00
const [command, _x1, _y1, ...rest] = segment.params;
2024-03-24 19:38:18 +02:00
let errorMessage = `Ineffective segment "${iconPath.slice(
segment.start,
segment.end,
)}" found`;
let resolutionTip = 'should be removed';
if (curveCommands.includes(command)) {
const [x2, y2, x, y] = rest;
if (
command === lowerShorthandCurveCommand &&
(x2 !== 0 || y2 !== 0)
) {
resolutionTip = `should be "l${removeLeadingZeros(
x2,
)} ${removeLeadingZeros(y2)}" or removed`;
}
2024-03-24 19:38:18 +02:00
if (command === upperShorthandCurveCommand) {
resolutionTip = `should be "L${removeLeadingZeros(
x2,
)} ${removeLeadingZeros(y2)}" or removed`;
}
2024-03-24 19:38:18 +02:00
if (command === lowerCurveCommand && (x !== 0 || y !== 0)) {
resolutionTip = `should be "l${removeLeadingZeros(
x,
)} ${removeLeadingZeros(y)}" or removed`;
}
2024-03-24 19:38:18 +02:00
if (command === upperCurveCommand) {
resolutionTip = `should be "L${removeLeadingZeros(
x,
)} ${removeLeadingZeros(y)}" or removed`;
}
}
if (segment.chain !== undefined) {
const readableChain = maybeShortenedWithEllipsis(
iconPath.slice(segment.chain.start, segment.chain.end),
);
2024-03-24 19:38:18 +02:00
errorMessage += ` in chain "${readableChain}"`;
}
2024-03-24 19:38:18 +02:00
errorMessage += ` at index ${
segment.start + getPathDIndex(ast.source)
}`;
2024-03-24 19:38:18 +02:00
reporter.error(`${errorMessage} (${resolutionTip})`);
}
}
},
2024-06-06 14:40:35 +02:00
(reporter, $, ast) => {
reporter.name = 'collinear-segments';
/**
* Extracts collinear coordinates from SVG path straight lines
2024-06-06 14:40:35 +02:00
* (does not extracts collinear coordinates from curves).
* @param {string} iconPath The SVG path of the icon.
* @returns {import('svg-path-segments').Segment[]} The collinear segments.
*/
// eslint-disable-next-line complexity
const getCollinearSegments = (iconPath) => {
2024-03-24 19:38:18 +02:00
const segments = getIconPathSegments(iconPath);
const collinearSegments = [];
const straightLineCommands = 'HhVvLlMm';
2024-03-24 19:38:18 +02:00
let currentLine = [];
let currentAbsCoord = [undefined, undefined];
let startPoint;
let _inStraightLine = false;
let _nextInStraightLine = false;
let _resetStartPoint = false;
for (let s = 0; s < segments.length; s++) {
2024-03-24 19:38:18 +02:00
const seg = segments[s];
const parms = seg.params;
const cmd = parms[0];
2024-06-06 14:40:35 +02:00
const nextCmd =
s + 1 < segments.length ? segments[s + 1].params[0] : null;
switch (cmd) {
// Next switch cases have been ordered by frequency
// of occurrence in the SVG paths of the icons
2024-03-24 19:38:18 +02:00
case 'M': {
2024-06-06 14:40:35 +02:00
// @ts-ignore
2024-03-24 19:38:18 +02:00
currentAbsCoord[0] = parms[1];
2024-06-06 14:40:35 +02:00
// @ts-ignore
2024-03-24 19:38:18 +02:00
currentAbsCoord[1] = parms[2];
// SVG 1.1:
// If a moveto is followed by multiple pairs of coordinates,
// the subsequent pairs are treated as implicit lineto commands.
if (seg.chain === undefined || seg.chain.start === seg.start) {
startPoint = undefined;
}
2024-03-24 19:38:18 +02:00
break;
2024-03-24 19:38:18 +02:00
}
case 'm': {
2024-06-06 14:40:35 +02:00
// @ts-ignore
2024-03-24 19:38:18 +02:00
currentAbsCoord[0] = (currentAbsCoord[0] || 0) + parms[1];
2024-06-06 14:40:35 +02:00
// @ts-ignore
2024-03-24 19:38:18 +02:00
currentAbsCoord[1] = (currentAbsCoord[1] || 0) + parms[2];
if (seg.chain === undefined || seg.chain.start === seg.start) {
startPoint = undefined;
}
2024-03-24 19:38:18 +02:00
break;
2024-03-24 19:38:18 +02:00
}
case 'H': {
2024-06-06 14:40:35 +02:00
// @ts-ignore
2024-03-24 19:38:18 +02:00
currentAbsCoord[0] = parms[1];
break;
2024-03-24 19:38:18 +02:00
}
case 'h': {
2024-06-06 14:40:35 +02:00
// @ts-ignore
2024-03-24 19:38:18 +02:00
currentAbsCoord[0] = (currentAbsCoord[0] || 0) + parms[1];
break;
2024-03-24 19:38:18 +02:00
}
case 'V': {
2024-06-06 14:40:35 +02:00
// @ts-ignore
2024-03-24 19:38:18 +02:00
currentAbsCoord[1] = parms[1];
break;
2024-03-24 19:38:18 +02:00
}
case 'v': {
2024-06-06 14:40:35 +02:00
// @ts-ignore
2024-03-24 19:38:18 +02:00
currentAbsCoord[1] = (currentAbsCoord[1] || 0) + parms[1];
break;
2024-03-24 19:38:18 +02:00
}
case 'L': {
2024-06-06 14:40:35 +02:00
// @ts-ignore
2024-03-24 19:38:18 +02:00
currentAbsCoord[0] = parms[1];
2024-06-06 14:40:35 +02:00
// @ts-ignore
2024-03-24 19:38:18 +02:00
currentAbsCoord[1] = parms[2];
break;
2024-03-24 19:38:18 +02:00
}
case 'l': {
2024-06-06 14:40:35 +02:00
// @ts-ignore
2024-03-24 19:38:18 +02:00
currentAbsCoord[0] = (currentAbsCoord[0] || 0) + parms[1];
2024-06-06 14:40:35 +02:00
// @ts-ignore
2024-03-24 19:38:18 +02:00
currentAbsCoord[1] = (currentAbsCoord[1] || 0) + parms[2];
break;
2024-03-24 19:38:18 +02:00
}
case 'Z':
2024-03-24 19:38:18 +02:00
case 'z': {
// TODO: Overlapping in Z should be handled in another rule
// @ts-ignore
2024-03-24 19:38:18 +02:00
currentAbsCoord = [startPoint[0], startPoint[1]];
_resetStartPoint = true;
break;
2024-03-24 19:38:18 +02:00
}
case 'C': {
2024-06-06 14:40:35 +02:00
// @ts-ignore
2024-03-24 19:38:18 +02:00
currentAbsCoord[0] = parms[5];
2024-06-06 14:40:35 +02:00
// @ts-ignore
2024-03-24 19:38:18 +02:00
currentAbsCoord[1] = parms[6];
break;
2024-03-24 19:38:18 +02:00
}
case 'c': {
2024-06-06 14:40:35 +02:00
// @ts-ignore
2024-03-24 19:38:18 +02:00
currentAbsCoord[0] = (currentAbsCoord[0] || 0) + parms[5];
2024-06-06 14:40:35 +02:00
// @ts-ignore
2024-03-24 19:38:18 +02:00
currentAbsCoord[1] = (currentAbsCoord[1] || 0) + parms[6];
break;
2024-03-24 19:38:18 +02:00
}
case 'A': {
2024-06-06 14:40:35 +02:00
// @ts-ignore
2024-03-24 19:38:18 +02:00
currentAbsCoord[0] = parms[6];
2024-06-06 14:40:35 +02:00
// @ts-ignore
2024-03-24 19:38:18 +02:00
currentAbsCoord[1] = parms[7];
break;
2024-03-24 19:38:18 +02:00
}
case 'a': {
2024-06-06 14:40:35 +02:00
// @ts-ignore
2024-03-24 19:38:18 +02:00
currentAbsCoord[0] = (currentAbsCoord[0] || 0) + parms[6];
2024-06-06 14:40:35 +02:00
// @ts-ignore
2024-03-24 19:38:18 +02:00
currentAbsCoord[1] = (currentAbsCoord[1] || 0) + parms[7];
break;
2024-03-24 19:38:18 +02:00
}
case 's': {
2024-06-06 14:40:35 +02:00
// @ts-ignore
2024-03-24 19:38:18 +02:00
currentAbsCoord[0] = (currentAbsCoord[0] || 0) + parms[1];
2024-06-06 14:40:35 +02:00
// @ts-ignore
2024-03-24 19:38:18 +02:00
currentAbsCoord[1] = (currentAbsCoord[1] || 0) + parms[2];
break;
2024-03-24 19:38:18 +02:00
}
case 'S': {
2024-06-06 14:40:35 +02:00
// @ts-ignore
2024-03-24 19:38:18 +02:00
currentAbsCoord[0] = parms[1];
2024-06-06 14:40:35 +02:00
// @ts-ignore
2024-03-24 19:38:18 +02:00
currentAbsCoord[1] = parms[2];
break;
2024-03-24 19:38:18 +02:00
}
case 't': {
2024-06-06 14:40:35 +02:00
// @ts-ignore
2024-03-24 19:38:18 +02:00
currentAbsCoord[0] = (currentAbsCoord[0] || 0) + parms[1];
2024-06-06 14:40:35 +02:00
// @ts-ignore
2024-03-24 19:38:18 +02:00
currentAbsCoord[1] = (currentAbsCoord[1] || 0) + parms[2];
break;
2024-03-24 19:38:18 +02:00
}
case 'T': {
2024-06-06 14:40:35 +02:00
// @ts-ignore
2024-03-24 19:38:18 +02:00
currentAbsCoord[0] = parms[1];
2024-06-06 14:40:35 +02:00
// @ts-ignore
2024-03-24 19:38:18 +02:00
currentAbsCoord[1] = parms[2];
break;
2024-03-24 19:38:18 +02:00
}
case 'Q': {
2024-06-06 14:40:35 +02:00
// @ts-ignore
2024-03-24 19:38:18 +02:00
currentAbsCoord[0] = parms[3];
2024-06-06 14:40:35 +02:00
// @ts-ignore
2024-03-24 19:38:18 +02:00
currentAbsCoord[1] = parms[4];
break;
2024-03-24 19:38:18 +02:00
}
case 'q': {
2024-06-06 14:40:35 +02:00
// @ts-ignore
2024-03-24 19:38:18 +02:00
currentAbsCoord[0] = (currentAbsCoord[0] || 0) + parms[3];
2024-06-06 14:40:35 +02:00
// @ts-ignore
2024-03-24 19:38:18 +02:00
currentAbsCoord[1] = (currentAbsCoord[1] || 0) + parms[4];
break;
2024-03-24 19:38:18 +02:00
}
default: {
throw new Error(`"${cmd}" command not handled`);
2024-03-24 19:38:18 +02:00
}
}
if (startPoint === undefined) {
2024-03-24 19:38:18 +02:00
startPoint = [currentAbsCoord[0], currentAbsCoord[1]];
} else if (_resetStartPoint) {
startPoint = undefined;
_resetStartPoint = false;
}
// @ts-ignore
_nextInStraightLine = straightLineCommands.includes(nextCmd);
2024-03-24 19:38:18 +02:00
const _exitingStraightLine =
_inStraightLine && !_nextInStraightLine;
_inStraightLine = straightLineCommands.includes(cmd);
if (_inStraightLine) {
2024-03-24 19:38:18 +02:00
currentLine.push([currentAbsCoord[0], currentAbsCoord[1]]);
} else {
if (_exitingStraightLine) {
if (straightLineCommands.includes(cmd)) {
2024-03-24 19:38:18 +02:00
currentLine.push([currentAbsCoord[0], currentAbsCoord[1]]);
}
2024-03-24 19:38:18 +02:00
// Get collinear coordinates
2024-03-24 19:38:18 +02:00
for (let p = 1; p < currentLine.length - 1; p++) {
const _collinearCoord = collinear(
// @ts-ignore
2024-03-24 19:38:18 +02:00
currentLine[p - 1][0],
currentLine[p - 1][1],
currentLine[p][0],
currentLine[p][1],
currentLine[p + 1][0],
currentLine[p + 1][1],
);
if (_collinearCoord) {
collinearSegments.push(
2024-03-24 19:38:18 +02:00
segments[s - currentLine.length + p + 1],
);
}
}
}
2024-03-24 19:38:18 +02:00
currentLine = [];
}
}
return collinearSegments;
};
2024-06-06 14:40:35 +02:00
const iconPath = getIconPath($);
2024-03-24 19:38:18 +02:00
const collinearSegments = getCollinearSegments(iconPath);
if (collinearSegments.length === 0) {
return;
}
2024-03-24 19:38:18 +02:00
const pathDIndex = getPathDIndex(ast.source);
for (const segment of collinearSegments) {
2024-03-24 19:38:18 +02:00
let errorMessage = `Collinear segment "${iconPath.slice(
segment.start,
segment.end,
)}" found`;
if (segment.chain !== undefined) {
2024-03-24 19:38:18 +02:00
const readableChain = maybeShortenedWithEllipsis(
iconPath.slice(segment.chain.start, segment.chain.end),
);
2024-03-24 19:38:18 +02:00
errorMessage += ` in chain "${readableChain}"`;
}
2024-03-24 19:38:18 +02:00
errorMessage += ` at index ${
segment.start + pathDIndex
} (should be removed)`;
2024-03-24 19:38:18 +02:00
reporter.error(errorMessage);
}
},
(reporter, $, ast) => {
reporter.name = 'extraneous';
if (!svgRegexp.test(ast.source)) {
if (ast.source.includes('\n') || ast.source.includes('\r')) {
reporter.error(
'Unexpected newline character(s) detected in SVG markup',
);
} else {
reporter.error(
'Unexpected character(s), most likely extraneous' +
' whitespace, detected in SVG markup',
);
}
}
},
2024-06-06 14:40:35 +02:00
(reporter, $, ast) => {
reporter.name = 'negative-zeros';
2024-06-06 14:40:35 +02:00
const iconPath = getIconPath($);
// Find negative zeros inside path
2024-03-24 19:38:18 +02:00
const negativeZeroMatches = [...iconPath.matchAll(negativeZerosRegexp)];
if (negativeZeroMatches.length > 0) {
// Calculate the index for each match in the file
const pathDIndex = getPathDIndex(ast.source);
for (const match of negativeZeroMatches) {
const negativeZeroFileIndex = match.index + pathDIndex;
const previousChar = ast.source[negativeZeroFileIndex - 1];
const replacement = '0123456789'.includes(previousChar)
? ' 0'
: '0';
reporter.error(
`Found "-0" at index ${negativeZeroFileIndex} (should` +
` be "${replacement}")`,
);
}
}
},
2024-06-06 14:40:35 +02:00
(reporter, $) => {
reporter.name = 'icon-centered';
2024-06-06 14:40:35 +02:00
const iconPath = getIconPath($);
const [minX, minY, maxX, maxY] = getIconPathBbox(iconPath);
2024-03-24 19:38:18 +02:00
const centerX = Number(((minX + maxX) / 2).toFixed(iconFloatPrecision));
const devianceX = centerX - iconTargetCenter;
2024-03-24 19:38:18 +02:00
const centerY = Number(((minY + maxY) / 2).toFixed(iconFloatPrecision));
const devianceY = centerY - iconTargetCenter;
if (
Math.abs(devianceX) > iconTolerance ||
Math.abs(devianceY) > iconTolerance
) {
reporter.error(
`<path> must be centered at (${iconTargetCenter}, ${iconTargetCenter});` +
` the center is currently (${centerX}, ${centerY})`,
);
}
},
2024-06-06 14:40:35 +02:00
(reporter, $, ast) => {
reporter.name = 'final-closepath';
2024-06-06 14:40:35 +02:00
const iconPath = getIconPath($);
const segments = getIconPathSegments(iconPath);
// Unnecessary characters after the final closepath
2024-06-06 14:40:35 +02:00
/** @type {import('svg-path-segments').Segment} */
// @ts-ignore
const lastSegment = segments.at(-1);
2024-06-06 14:40:35 +02:00
const endsWithZ = ['z', 'Z'].includes(lastSegment.params[0]);
if (endsWithZ && lastSegment.end - lastSegment.start > 1) {
const ending = iconPath.slice(lastSegment.start + 1);
const closepath = iconPath.at(lastSegment.start);
const pathDIndex = getPathDIndex(ast.source);
const index = pathDIndex + lastSegment.start + 2;
const errorMessage =
`Invalid character(s) "${ending}" after the final` +
` closepath command "${closepath}" at index ${index}` +
` (should be removed)`;
reporter.error(errorMessage);
}
},
2024-06-06 14:40:35 +02:00
(reporter, $, ast) => {
reporter.name = 'path-format';
2024-06-06 14:40:35 +02:00
const iconPath = getIconPath($);
if (!SVG_PATH_REGEX.test(iconPath)) {
2024-03-24 19:38:18 +02:00
const errorMessage = 'Invalid path format';
let reason;
if (!iconPath.startsWith('M') && !iconPath.startsWith('m')) {
2024-03-24 19:38:18 +02:00
// Doesn't start with moveto
reason =
'should start with "moveto" command ("M" or "m"),' +
2024-03-24 19:38:18 +02:00
` but starts with "${iconPath[0]}"`;
reporter.error(`${errorMessage}: ${reason}`);
}
2024-03-24 19:38:18 +02:00
const validPathCharacters = SVG_PATH_REGEX.source.replaceAll(
/[[\]+^$]/g,
'',
);
const invalidCharactersMsgs = [];
const pathDIndex = getPathDIndex(ast.source);
2024-03-24 19:38:18 +02:00
for (const [i, char] of Object.entries(iconPath)) {
if (!validPathCharacters.includes(char)) {
invalidCharactersMsgs.push(
2024-03-24 19:38:18 +02:00
`"${char}" at index ${pathDIndex + Number.parseInt(i, 10)}`,
);
}
}
2024-03-24 19:38:18 +02:00
// Contains invalid characters
if (invalidCharactersMsgs.length > 0) {
reason = `unexpected character${
invalidCharactersMsgs.length > 1 ? 's' : ''
} found (${invalidCharactersMsgs.join(', ')})`;
2024-03-24 19:38:18 +02:00
reporter.error(`${errorMessage}: ${reason}`);
}
}
},
(reporter, $, ast) => {
reporter.name = 'svg-format';
// Don't allow explicit '</path>' closing tag
if (ast.source.includes('</path>')) {
const reason =
`found a closing "path" tag at index ${ast.source.indexOf(
'</path>',
)}. The path should be self-closing,` +
' use "/>" instead of "></path>".';
reporter.error(`Invalid SVG content format: ${reason}`);
}
},
],
},
};
export default config;