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/build/package.js

206 lines
7.1 KiB
JavaScript
Raw Permalink Normal View History

2024-03-25 01:38:18 +08:00
#!/usr/bin/env node
// @ts-check
/**
2024-06-06 14:40:35 +02:00
* @file
2023-04-19 15:23:13 +02:00
* Simple Icons package build script.
*/
2024-06-06 14:40:35 +02:00
/**
* @typedef {import('../../types.js').License} License
*/
2024-03-25 01:38:18 +08:00
import {promises as fs} from 'node:fs';
import path from 'node:path';
2024-09-17 09:46:48 +08:00
import {format} from 'node:util';
2024-03-25 01:38:18 +08:00
import {transform as esbuildTransform} from 'esbuild';
import {
2024-12-21 22:49:25 +08:00
getIconSlug,
getIconsData,
slugToVariableName,
svgToPath,
titleToHtmlFriendly,
2023-04-19 15:23:13 +02:00
} from '../../sdk.mjs';
import {sortIconsCompare} from '../utils.js';
const UTF8 = 'utf8';
const rootDirectory = path.resolve(import.meta.dirname, '..', '..');
2024-03-25 01:38:18 +08:00
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');
const templatesDirectory = path.resolve(import.meta.dirname, 'templates');
2024-03-25 01:38:18 +08:00
const iconObjectTemplateFile = path.resolve(
2024-12-21 22:49:25 +08:00
templatesDirectory,
'icon-object.js.template',
2024-03-25 01:38:18 +08: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('../../types.d.ts').IconData} IconDataAndObject
*/
2024-06-06 14:40:35 +02:00
const icons = await getIconsData();
const iconObjectTemplate = await fs.readFile(iconObjectTemplateFile, UTF8);
/**
* Escape a string for use in a JavaScript string.
* @param {string} value The value to escape.
* @returns {string} The escaped value.
2024-06-06 14:40:35 +02:00
*/
2025-05-28 02:36:59 +08:00
const escape = (value) => value.replaceAll(/(?<!\\)'/g, String.raw`\'`);
2024-06-06 14:40:35 +02:00
/**
* Converts a license object to a URL if the URL is not defined.
* @param {License} license The license object or URL.
* @returns {License} The license object with a URL.
2024-06-06 14:40:35 +02:00
*/
const licenseToString = (license) => {
2024-12-21 22:49:25 +08:00
if (license.url === undefined) {
license.url = `https://spdx.org/licenses/${license.type}`;
}
2024-06-06 14:40:35 +02:00
2024-12-21 22:49:25 +08:00
return license;
2024-06-06 14:40:35 +02:00
};
/**
* Converts an icon object to a JavaScript object.
* @param {IconDataAndObject} icon The icon object.
* @returns {string} The JavaScript object.
*/
2025-05-28 02:36:59 +08:00
const iconDataAndObjectToJsRepr = (icon) =>
format(
2024-12-21 22:49:25 +08:00
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(licenseToString(icon.license))},`,
);
2024-06-06 14:40:35 +02:00
/**
* Write JavaScript content to a file.
*
* ESBuild by default uses `ascii` encoding for JavaScript files, so the titles of icons
* are encoded using escape sequences (eg. "Aerom\xE9xico" instead of "Aeroméxico").
* See {@link https://esbuild.github.io/api/#charset}.
* Although this adds a minimal size overhead, it is needed to ensure that our distributed
* JavaScript files are compatible with all JavaScript environments. Especially, browsers
* that are not using `<meta charset="utf-8">` in their HTML. As we support browsers
* without meta charset in SVG `<title>` elements, we need to ensure the same for scripts.
* @param {string} filepath The path to the file to write.
* @param {string} rawJavaScript The raw JavaScript content to write to the file.
* @param {'cjs'} [format] The format of the resulting JavaScript file.
2024-06-06 14:40:35 +02:00
*/
const writeJs = async (filepath, rawJavaScript, format = undefined) => {
/** @type {import('esbuild').TransformOptions} */
const options = {minify: true, charset: 'ascii', format};
2024-12-21 22:49:25 +08:00
const {code} = await esbuildTransform(rawJavaScript, options);
// ESBuild adds a trailing newline to the end of the file
await fs.writeFile(filepath, code.trimEnd());
2024-06-06 14:40:35 +02:00
};
/**
* Write TypeScript content to a file.
* @param {string} filepath The path to the file to write.
* @param {string} rawTypeScript The raw TypeScript content to write to the file.
2024-06-06 14:40:35 +02:00
*/
const writeTs = async (filepath, rawTypeScript) => {
2024-12-21 22:49:25 +08:00
await fs.writeFile(filepath, rawTypeScript);
2024-06-06 14:40:35 +02:00
};
/**
* Build icons intermediate instances.
* @returns {Promise<{
* icon: IconDataAndObject,
* iconObjectRepr: string,
* iconExportName: string
* }[]>} Merged icon data and object instances.
*/
const buildIcons = async () =>
Promise.all(
icons.map(async (iconData) => {
const slug = getIconSlug(iconData);
const svgFilepath = path.resolve(iconsDirectory, `${slug}.svg`);
const svg = await fs.readFile(svgFilepath, UTF8);
/** @type {IconDataAndObject} */
const icon = {};
Object.assign(icon, iconData);
icon.svg = svg;
icon.path = svgToPath(svg);
icon.slug = slug;
const iconObjectRepr = iconDataAndObjectToJsRepr(icon);
const iconExportName = slugToVariableName(slug);
return {icon, iconObjectRepr, iconExportName};
2024-12-21 22:49:25 +08:00
}),
);
const build = async () => {
const builtIcons = await buildIcons();
2024-12-21 22:49:25 +08:00
const iconsBarrelDts = [];
const iconsBarrelJs = [];
const iconsBarrelMjs = [];
builtIcons.sort((a, b) => sortIconsCompare(a.icon, b.icon));
for (const {iconObjectRepr, iconExportName} of builtIcons) {
2024-12-21 22:49:25 +08:00
iconsBarrelDts.push(`export const ${iconExportName}:I;`);
iconsBarrelJs.push(`${iconExportName}:${iconObjectRepr},`);
iconsBarrelMjs.push(`export const ${iconExportName}=${iconObjectRepr}`);
2024-12-21 22:49:25 +08:00
}
// Constants used in templates to reduce package size
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>';`;
// Write our file containing the exports of all icons in CommonJS ...
const rawIndexJs = `${constantsString}module.exports={${iconsBarrelJs.join(
'',
)}};`;
await writeJs(indexJsFile, rawIndexJs);
// ... and ESM
const rawIndexMjs = constantsString + iconsBarrelMjs.join('');
await writeJs(indexMjsFile, rawIndexMjs);
// ... and create a type declaration file
const rawIndexDts = `import {SimpleIcon} from "./types";export {SimpleIcon};type I=SimpleIcon;${iconsBarrelDts.join(
'',
)}`;
await writeTs(indexDtsFile, rawIndexDts);
// Create a CommonJS SDK file
await writeJs(sdkJsFile, await fs.readFile(sdkMjsFile, UTF8), 'cjs');
// Build deprecated `simple-icons/icons` entrypoint.
// TODO: This must be removed at v17.
const deprecatedMessage =
`⚠️ The entrypoint 'simple-icons/icons' is deprecated and` +
` will be removed in version 17.0.0`;
const jsDeprecationMessage =
`${deprecatedMessage}. Please, import icons from 'simple-icons'` +
` using \`require('simple-icons')\` instead of \`require('simple-icons/icons')\`.`;
const iconsIndexJs =
`console.warn("${jsDeprecationMessage}");` +
`module.exports=require('./index.js');`;
const iconsIndexJsFile = path.resolve(rootDirectory, 'index-icons.js');
await writeJs(iconsIndexJsFile, iconsIndexJs);
const mjsDeprecationMessage =
`${deprecatedMessage}. Please, import icons from 'simple-icons'` +
` using \`import ... from 'simple-icons'\` instead of \`import ... from 'simple-icons/icons'\`.`;
const iconsIndexMjs =
`console.warn("${mjsDeprecationMessage}");` +
`export * from './index.mjs';`;
const iconsIndexMjsFile = path.resolve(rootDirectory, 'index-icons.mjs');
await writeJs(iconsIndexMjsFile, iconsIndexMjs);
};
export bundled icons from one entry point and add types (#6767) * export all icons from a single file (#6189) * fix: revert formatting, add exports to package.json * feat: generate icons.js and add relevant exports field * add minifyAndWrite Co-authored-by: Eric Cornelissen <ericornelissen@gmail.com> * fix: minifyAndWrite build * add type: commonjs Co-authored-by: Eric Cornelissen <ericornelissen@gmail.com> * simplify exports Co-authored-by: Eric Cornelissen <ericornelissen@gmail.com> * add "require" in exports * place objects directly in barrel file * write exports minified Co-authored-by: Eric Cornelissen <ericornelissen@gmail.com> * fix formatting Co-authored-by: Eric Cornelissen <ericornelissen@gmail.com> * refactor slugToVariableName code into a function * fix slugToVariableName * change prefix to "si" * move slugToVariableName to local helper functions * unignore icons.js and icons.mjs Co-authored-by: Eric Cornelissen <ericornelissen@gmail.com> * feat: add types (#6580) * feat: add types * fix linting error * export default from types/index.d.ts * minify * revert formatting changes * revert formatting change * change paths from types/index.d.ts to index.d.ts * mark icons.get as deprecated * move type alias to another file * update readme * update readme.md Co-authored-by: Eric Cornelissen <ericornelissen@gmail.com> * update typescript usage section Co-authored-by: Eric Cornelissen <ericornelissen@gmail.com> * fix conflicts * Apply suggestions from code review Co-authored-by: Eric Cornelissen <ericornelissen@gmail.com> * add writeTs function Co-authored-by: Eric Cornelissen <ericornelissen@gmail.com>
2021-10-28 16:16:34 -07:00
2024-03-25 01:38:18 +08:00
await build();