1
0
mirror of https://github.com/simple-icons/simple-icons.git synced 2025-07-02 22:16:54 +02:00
Files
simple-icons/scripts/utils.js

181 lines
4.6 KiB
JavaScript
Raw Normal View History

// @ts-check
2024-06-06 14:40:35 +02:00
/**
* @file Internal utilities.
*
* Here resides all the functionality that does not qualifies to reside
* in the SDK because is not publicly exposed.
*/
import fs from 'node:fs/promises';
2024-03-25 01:38:18 +08:00
import path from 'node:path';
import {collator, getIconSlug, getIconsDataPath, titleToSlug} from '../sdk.mjs';
2024-06-06 14:40:35 +02:00
/**
* @typedef {import("../sdk.js").IconData} IconData
* @typedef {import("../sdk.js").DuplicateAlias} DuplicateAlias
2024-06-06 14:40:35 +02:00
*/
/**
* Get JSON schema data.
2024-06-06 14:40:35 +02:00
* @returns {Promise<any>} JSON schema data.
*/
export const getJsonSchemaData = async () =>
2024-12-21 22:49:25 +08:00
JSON.parse(
await fs.readFile(
path.resolve(import.meta.dirname, '..', '.jsonschema.json'),
'utf8',
),
2024-12-21 22:49:25 +08:00
);
/**
* Write icons data to _data/simple-icons.json.
* @param {IconData[]} iconsData Icons data array.
* @param {boolean} [minify] Whether to minify the JSON output.
*/
export const writeIconsData = async (iconsData, minify = false) => {
2024-12-21 22:49:25 +08:00
await fs.writeFile(
getIconsDataPath(),
2024-12-21 22:49:25 +08:00
`${JSON.stringify(iconsData, null, minify ? 0 : '\t')}\n`,
'utf8',
);
};
/**
* Get SPDX license IDs from `spdx-license-ids` package.
* @returns {Promise<string[]>} Set of SPDX license IDs.
*/
export const getSpdxLicenseIds = async () =>
2024-12-21 22:49:25 +08:00
JSON.parse(
await fs.readFile(
path.resolve(
import.meta.dirname,
'..',
2024-12-21 22:49:25 +08:00
'node_modules',
'spdx-license-ids',
'index.json',
),
'utf8',
),
);
/**
* The compare function for sorting icons in *_data/simple-icons.json*.
* @param {IconData} a Icon A.
* @param {IconData} b Icon B.
* @returns {number} Comparison result.
*/
2025-05-28 02:36:59 +08:00
export const sortIconsCompare = (a, b) =>
a.title === b.title
? collator.compare(getIconSlug(a), getIconSlug(b))
2024-12-21 22:49:25 +08:00
: collator.compare(a.title, b.title);
2025-02-23 01:22:52 +08:00
/**
* The compare function for sorting icon duplicate aliases in *_data/simple-icons.json*.
* @param {DuplicateAlias} a Duplicate alias A.
* @param {DuplicateAlias} b Duplicate alias B.
* @returns {number} Comparison result.
*/
2025-05-28 02:36:59 +08:00
const sortDuplicatesCompare = (a, b) =>
a.title === b.title
? collator.compare(titleToSlug(a.title), titleToSlug(b.title))
: collator.compare(a.title, b.title);
/**
* Sort icon data or duplicate alias object.
* @template {IconData | DuplicateAlias} T Either icon data or duplicate alias.
* @param {T} icon The icon data or duplicate alias as it appears in *_data/simple-icons.json*.
* @returns {T} The sorted icon data or duplicate alias.
2025-02-23 01:22:52 +08:00
*/
const sortIconOrDuplicate = (icon) => {
2025-02-23 01:22:52 +08:00
const keyOrder = [
'title',
'slug',
'hex',
'source',
'guidelines',
'license',
'aliases',
// This is not appears in icon data but it's in the alias object.
'loc',
];
/** @type {T} */
const sortedIcon = Object.assign(
Object.fromEntries(
Object.entries(icon).sort(
([key1], [key2]) => keyOrder.indexOf(key1) - keyOrder.indexOf(key2),
),
2025-02-23 01:22:52 +08:00
),
);
2025-02-23 01:22:52 +08:00
return sortedIcon;
};
/**
* Sort license object.
* @param {IconData['license']} license The license object as it appears in *_data/simple-icons.json*.
* @returns {IconData['license']} The sorted license object.
2025-02-23 01:22:52 +08:00
*/
const sortLicense = (license) => {
if (!license) return undefined;
const keyOrder = ['type', 'url'];
/** @type {IconData['license']} */
const sortedLicense = Object.assign(
Object.fromEntries(
Object.entries(license).sort(
([key1], [key2]) => keyOrder.indexOf(key1) - keyOrder.indexOf(key2),
),
2025-02-23 01:22:52 +08:00
),
);
2025-02-23 01:22:52 +08:00
return sortedLicense;
};
/**
* Sort object key alphabetically.
* @param {IconData['aliases']} object The aliases object as it appears in *_data/simple-icons.json*.
* @returns {{[_: string]: string} | undefined} The sorted aliases object.
2025-02-23 01:22:52 +08:00
*/
const sortAlphabetically = (object) => {
if (!object) return undefined;
const sorted = Object.assign(
Object.fromEntries(
Object.entries(object).sort(([key1], [key2]) => (key1 > key2 ? 1 : -1)),
),
2025-02-23 01:22:52 +08:00
);
return sorted;
};
/**
* Sort icons data.
* @param {IconData[]} iconsData The icons data as it appears in *_data/simple-icons.json*.
* @returns {IconData[]} The sorted icons data.
*/
export const formatIconData = (iconsData) => {
2025-05-21 18:23:40 +08:00
const iconsDataCopy = structuredClone(iconsData);
2025-05-28 02:36:59 +08:00
const icons = iconsDataCopy.map((icon) =>
sortIconOrDuplicate({
2025-02-23 01:22:52 +08:00
...icon,
license: sortLicense(icon.license),
aliases: icon.aliases
? sortAlphabetically({
aka: icon.aliases.aka?.sort(collator.compare),
dup: icon.aliases.dup
? icon.aliases.dup.sort(sortDuplicatesCompare).map((d) =>
sortIconOrDuplicate({
...d,
loc: sortAlphabetically(d.loc),
}),
)
2025-02-23 01:22:52 +08:00
: undefined,
loc: sortAlphabetically(icon.aliases.loc),
old: icon.aliases.old?.sort(collator.compare),
})
: undefined,
2025-05-28 02:36:59 +08:00
}),
);
icons.sort(sortIconsCompare);
return icons;
2025-02-23 01:22:52 +08:00
};