2024-03-24 19:38:18 +02:00
|
|
|
#!/usr/bin/env node
|
2018-08-26 23:23:57 +02:00
|
|
|
/**
|
2024-06-06 14:40:35 +02:00
|
|
|
* @file
|
2023-04-19 15:23:13 +02:00
|
|
|
* Simple Icons package build script.
|
2018-08-26 23:23:57 +02:00
|
|
|
*/
|
|
|
|
|
2024-06-06 14:40:35 +02:00
|
|
|
/**
|
|
|
|
* @typedef {import('../../types.js').License} License
|
|
|
|
* @typedef {import('esbuild').TransformOptions} EsBuildTransformOptions
|
|
|
|
*/
|
|
|
|
|
2024-03-24 19:38:18 +02:00
|
|
|
import {promises as fs} from 'node:fs';
|
2021-12-25 16:22:56 +02:00
|
|
|
import path from 'node:path';
|
|
|
|
import util from 'node:util';
|
2024-03-24 19:38:18 +02:00
|
|
|
import {transform as esbuildTransform} from 'esbuild';
|
2021-12-25 16:22:56 +02:00
|
|
|
import {
|
2024-03-24 19:38:18 +02:00
|
|
|
collator,
|
|
|
|
getDirnameFromImportMeta,
|
2021-12-25 16:22:56 +02:00
|
|
|
getIconSlug,
|
2024-03-24 19:38:18 +02:00
|
|
|
getIconsData,
|
|
|
|
slugToVariableName,
|
2021-12-25 16:22:56 +02:00
|
|
|
svgToPath,
|
|
|
|
titleToHtmlFriendly,
|
2023-04-19 15:23:13 +02:00
|
|
|
} from '../../sdk.mjs';
|
2021-12-25 16:22:56 +02:00
|
|
|
|
|
|
|
const __dirname = getDirnameFromImportMeta(import.meta.url);
|
2019-07-14 21:05:38 +02:00
|
|
|
|
2021-10-25 21:13:10 +02:00
|
|
|
const UTF8 = 'utf8';
|
2019-08-15 13:23:35 +02:00
|
|
|
|
2024-03-24 19:38:18 +02:00
|
|
|
const rootDirectory = path.resolve(__dirname, '..', '..');
|
|
|
|
const iconsDirectory = path.resolve(rootDirectory, 'icons');
|
|
|
|
const indexJsFile = path.resolve(rootDirectory, 'index.js');
|
|
|
|
const indexMjsFile = path.resolve(rootDirectory, 'index.mjs');
|
|
|
|
const sdkJsFile = path.resolve(rootDirectory, 'sdk.js');
|
|
|
|
const sdkMjsFile = path.resolve(rootDirectory, 'sdk.mjs');
|
|
|
|
const indexDtsFile = path.resolve(rootDirectory, 'index.d.ts');
|
2019-08-15 13:23:35 +02:00
|
|
|
|
2024-03-24 19:38:18 +02:00
|
|
|
const templatesDirectory = path.resolve(__dirname, 'templates');
|
|
|
|
const iconObjectTemplateFile = path.resolve(
|
|
|
|
templatesDirectory,
|
|
|
|
'icon-object.js.template',
|
|
|
|
);
|
2019-08-15 13:23:35 +02:00
|
|
|
|
2024-06-06 21:24:29 +02:00
|
|
|
/**
|
|
|
|
* Merged type from icon data and icon JS object needed to build by reference
|
|
|
|
* to not decrease performance in the build process.
|
|
|
|
* @typedef {import('../../types.js').SimpleIcon & import('../../sdk.d.ts').IconData} IconDataAndObject
|
|
|
|
*/
|
|
|
|
|
|
|
|
/** @type {IconDataAndObject[]} */
|
|
|
|
// @ts-ignore
|
2024-06-06 14:40:35 +02:00
|
|
|
const icons = await getIconsData();
|
|
|
|
const iconObjectTemplate = await fs.readFile(iconObjectTemplateFile, UTF8);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {string} value The value to escape
|
|
|
|
* @returns {string} The escaped value
|
|
|
|
*/
|
|
|
|
const escape = (value) => {
|
|
|
|
return value.replaceAll(/(?<!\\)'/g, "\\'");
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {License} license The license object or URL
|
|
|
|
* @returns {License} The license object with a URL
|
|
|
|
*/
|
|
|
|
const licenseToObject = (license) => {
|
|
|
|
if (license.url === undefined) {
|
|
|
|
license.url = `https://spdx.org/licenses/${license.type}`;
|
|
|
|
}
|
|
|
|
|
|
|
|
return license;
|
|
|
|
};
|
|
|
|
|
2024-06-06 21:24:29 +02:00
|
|
|
/**
|
|
|
|
* Converts an icon object to a JavaScript object.
|
|
|
|
* @param {IconDataAndObject} icon The icon object
|
|
|
|
* @returns {string} The JavaScript object
|
|
|
|
*/
|
2024-06-06 14:40:35 +02:00
|
|
|
const iconToJsObject = (icon) => {
|
|
|
|
return util.format(
|
|
|
|
iconObjectTemplate,
|
|
|
|
escape(icon.title),
|
|
|
|
escape(icon.slug),
|
|
|
|
escape(titleToHtmlFriendly(icon.title)),
|
|
|
|
escape(icon.path),
|
|
|
|
escape(icon.source),
|
|
|
|
escape(icon.hex),
|
|
|
|
icon.guidelines ? `\n guidelines: '${escape(icon.guidelines)}',` : '',
|
|
|
|
icon.license === undefined
|
|
|
|
? ''
|
|
|
|
: `\n license: ${JSON.stringify(licenseToObject(icon.license))},`,
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {string} filepath The path to the file to write
|
|
|
|
* @param {string} rawJavaScript The raw JavaScript content to write to the file
|
|
|
|
* @param {EsBuildTransformOptions | null} options The options to pass to esbuild
|
|
|
|
*/
|
|
|
|
const writeJs = async (filepath, rawJavaScript, options = null) => {
|
|
|
|
options = options === null ? {minify: true} : options;
|
|
|
|
const {code} = await esbuildTransform(rawJavaScript, options);
|
|
|
|
await fs.writeFile(filepath, code);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {string} filepath The path to the file to write
|
|
|
|
* @param {string} rawTypeScript The raw TypeScript content to write to the file
|
|
|
|
*/
|
|
|
|
const writeTs = async (filepath, rawTypeScript) => {
|
|
|
|
await fs.writeFile(filepath, rawTypeScript);
|
|
|
|
};
|
|
|
|
|
2021-11-24 04:01:24 +02:00
|
|
|
const build = async () => {
|
2022-09-23 20:44:28 +02:00
|
|
|
const buildIcons = await Promise.all(
|
2021-12-25 16:22:56 +02:00
|
|
|
icons.map(async (icon) => {
|
2021-11-24 04:01:24 +02:00
|
|
|
const filename = getIconSlug(icon);
|
2024-03-24 19:38:18 +02:00
|
|
|
const svgFilepath = path.resolve(iconsDirectory, `${filename}.svg`);
|
2023-08-08 06:38:52 +02:00
|
|
|
icon.svg = await fs.readFile(svgFilepath, UTF8);
|
2021-11-24 04:01:24 +02:00
|
|
|
icon.path = svgToPath(icon.svg);
|
|
|
|
icon.slug = filename;
|
2024-06-06 14:40:35 +02:00
|
|
|
const iconObject = iconToJsObject(icon);
|
2021-11-24 04:01:24 +02:00
|
|
|
const iconExportName = slugToVariableName(icon.slug);
|
2024-03-24 19:38:18 +02:00
|
|
|
return {icon, iconObject, iconExportName};
|
2021-11-24 04:01:24 +02:00
|
|
|
}),
|
2021-11-23 22:33:37 +02:00
|
|
|
);
|
2021-10-29 01:16:34 +02:00
|
|
|
|
2022-09-23 20:44:28 +02:00
|
|
|
const iconsBarrelDts = [];
|
|
|
|
const iconsBarrelJs = [];
|
|
|
|
const iconsBarrelMjs = [];
|
|
|
|
|
2022-09-24 17:29:43 +02:00
|
|
|
buildIcons.sort((a, b) => collator.compare(a.icon.title, b.icon.title));
|
2024-03-24 19:38:18 +02:00
|
|
|
for (const {iconObject, iconExportName} of buildIcons) {
|
2022-09-23 20:44:28 +02:00
|
|
|
iconsBarrelDts.push(`export const ${iconExportName}:I;`);
|
|
|
|
iconsBarrelJs.push(`${iconExportName}:${iconObject},`);
|
|
|
|
iconsBarrelMjs.push(`export const ${iconExportName}=${iconObject}`);
|
2023-08-08 06:38:52 +02:00
|
|
|
}
|
2022-09-23 20:44:28 +02:00
|
|
|
|
2024-03-24 19:38:18 +02:00
|
|
|
// Constants used in templates to reduce package size
|
2022-05-27 15:09:33 +02:00
|
|
|
const constantsString = `const a='<svg role="img" viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg"><title>',b='</title><path d="',c='"/></svg>';`;
|
|
|
|
|
2024-03-24 19:38:18 +02:00
|
|
|
// Write our file containing the exports of all icons in CommonJS ...
|
2022-11-28 13:32:57 +02:00
|
|
|
const rawIndexJs = `${constantsString}module.exports={${iconsBarrelJs.join(
|
2022-05-27 15:09:33 +02:00
|
|
|
'',
|
|
|
|
)}};`;
|
2022-11-28 13:32:57 +02:00
|
|
|
await writeJs(indexJsFile, rawIndexJs);
|
2024-03-24 19:38:18 +02:00
|
|
|
// ... and ESM
|
2022-11-28 13:32:57 +02:00
|
|
|
const rawIndexMjs = constantsString + iconsBarrelMjs.join('');
|
|
|
|
await writeJs(indexMjsFile, rawIndexMjs);
|
2024-03-24 19:38:18 +02:00
|
|
|
// ... and create a type declaration file
|
2022-11-28 13:32:57 +02:00
|
|
|
const rawIndexDts = `import {SimpleIcon} from "./types";export {SimpleIcon};type I=SimpleIcon;${iconsBarrelDts.join(
|
2022-05-26 14:29:37 +02:00
|
|
|
'',
|
|
|
|
)}`;
|
2022-11-28 13:32:57 +02:00
|
|
|
await writeTs(indexDtsFile, rawIndexDts);
|
2023-04-19 15:23:13 +02:00
|
|
|
|
2024-03-24 19:38:18 +02:00
|
|
|
// Create a CommonJS SDK file
|
2023-04-19 15:23:13 +02:00
|
|
|
await writeJs(sdkJsFile, await fs.readFile(sdkMjsFile, UTF8), {
|
|
|
|
format: 'cjs',
|
|
|
|
});
|
2021-11-24 04:01:24 +02:00
|
|
|
};
|
2021-10-29 01:16:34 +02:00
|
|
|
|
2024-03-24 19:38:18 +02:00
|
|
|
await build();
|