-
Notifications
You must be signed in to change notification settings - Fork 3
/
prebuild.mjs
127 lines (115 loc) · 4.41 KB
/
prebuild.mjs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
import { spawn } from "child_process";
import { lstatSync, readFileSync, writeFileSync } from "fs";
import { readdir } from "fs/promises";
import { dirname, posix, resolve, sep, join } from "path";
import { exit } from "process";
import { fileURLToPath } from "url";
const __dirname = dirname(fileURLToPath(import.meta.url));
const pkgFile = JSON.parse(readFileSync(resolve(__dirname, "package.json"), "utf8"));
const outputFile = resolve(__dirname, "src", "js", "icons", "list.ts");
// Create a list of icons
const iconsDir = resolve(__dirname, "node_modules", "@fluentui", "svg-icons", "icons") + sep;
// https://stackoverflow.com/a/71166133/8304380
const walk = async (dirPath) => Promise.all(
await readdir(dirPath, { withFileTypes: true }).then((entries) => entries.map((entry) => {
const childPath = join(dirPath, entry.name);
return entry.isDirectory() ? walk(childPath) : childPath;
})),
);
console.time("Create icons listing");
// Get the version of svg-icons we currently have a listing of
// Skip creating a new listing if same
const fileMeta = readFileSync(outputFile, "utf8").split("\n")[0].substring(2).trim().split(",");
const iconsVersion = pkgFile.dependencies["@fluentui/svg-icons"];
if (fileMeta[0] === iconsVersion && fileMeta[1] === pkgFile.version) {
console.log("Icon list up to date");
exit(0);
} else {
// Make a new listing + create a tar of the icons (without blocking)
const createTarScript = resolve(__dirname, "prebuild.create-tar.mjs");
const child = spawn("node", [createTarScript], { detached: true, stdio: "ignore" });
child.unref();
console.log("Creating icons tarball in a new detached process");
}
// Object of first icon segment => [icon names], used for grouping in output file
let iconsList = {};
const iconFilesRaw = await walk(iconsDir);
// Make the path use the same directory separators
// Filter out directory names
// Only keep the file name
// Some icons are in a directory, so convert \ to /
// Remove file extensions
// Sort alphabetically
// Group icons
iconFilesRaw.flat(Number.POSITIVE_INFINITY)
.map(icon => resolve(icon))
.filter(icon => lstatSync(icon).isFile())
.map(icon => icon.replace(iconsDir, ""))
.map(icon => icon.split(sep).join(posix.sep))
.map(icon => icon.replace(".svg", ""))
.sort((a, b) => a.localeCompare(b))
.map(icon => {
const groupId = icon.split(/[_|/-]+/)[0];
if (!(groupId in iconsList))
iconsList[groupId] = [];
iconsList[groupId].push(icon);
});
// Template literal with quotes instead of JSON.stringify: https://linkify.cz/template-stringify-bench
// Convert to a grouped array (without []) of icons
let lastGroup = "";
const iconsString = Object.entries(iconsList)
.map((group, i) => {
const isFirst = i === 0;
const isLast = i + 1 === Object.keys(iconsList).length;
const groupId = group[0];
// Make the icon names a valid string and join them
const val = group[1].map(icon => `"${icon}"`).join(",");
let prefix = "", suffix = "";
if (groupId !== lastGroup) {
lastGroup = groupId;
// Make it start on a new line
if (isFirst) prefix = "\n\t";
// A comma because val doesn't have it
suffix = ",\n";
// Don't insert \t at the end of array
if (!isLast) suffix += "\t";
}
// Combine all of it together
return `${prefix}${val}${suffix}`;
})
.join("");
// Create group union types of all icons
const iconGroupsType = Object.entries(iconsList)
.map(group => {
const name = group[0];
const values = group[1].map(icon => `"${icon}"`).join("|");
return `type ${name}_t = ${values};`;
})
.join("\n");
// Make a simple union type of all groups
// Grouped by intial
let lastGroupInitial = "";
const groupUnionType = Object.keys(iconsList)
.map(groupId => {
const initial = groupId.charAt(0);
if (initial !== lastGroupInitial) {
lastGroupInitial = initial;
groupId = `\n\t${groupId}`;
}
return `${groupId}_t`;
})
.join("|");
const content = `//${iconsVersion},${pkgFile.version}
/*
Generated file, DO NOT EDIT (changes will be lost anyway)
*/
import { CDN_URL } from "../fluent";
// Convert ["icon_name", ...] to {icon_name: "icon_name", ...}
const arrToObj = (arr: _Icons[]) => Object.fromEntries(arr.map(e => [e, e]));
export const _iconsUrl = \`\${CDN_URL}/icons\` as const;
export const _icons = arrToObj([${iconsString}]);
export type _Icons = ${groupUnionType};
${iconGroupsType}
`.replaceAll("\n", "\r\n"); // ESLint wants us to use windows line breaks
writeFileSync(outputFile, content);
console.timeEnd("Create icons listing");