- web/: Vue 3 + Vite + UnoCSS + Pinia, dark tactical theme (amber/#0d1117) - AppNav, ListingCard, SearchView with filters/sort, composables (useSnipeMode, useKonamiCode, useMotion), Pinia search store - Steal shimmer, auction countdown, Snipe Mode easter egg all native in Vue - docker/web/: nginx + multi-stage Dockerfile (node build → nginx serve) - compose.yml: api (8510) + web (8509) services - Dockerfile CMD updated to uvicorn for upcoming FastAPI layer - Clean build: 0 TS errors, 380 modules
32 lines
No EOL
833 B
JavaScript
32 lines
No EOL
833 B
JavaScript
import { camelize, pascalize, snakelize } from "../misc/strings.js";
|
|
import { promises } from "fs";
|
|
|
|
/**
|
|
* @returns A {@link CustomIconLoader} for loading icons from a directory
|
|
*/
|
|
function FileSystemIconLoader(dir, transform) {
|
|
return async (name) => {
|
|
const paths = [
|
|
`${dir}/${name}.svg`,
|
|
`${dir}/${camelize(name)}.svg`,
|
|
`${dir}/${pascalize(name)}.svg`,
|
|
`${dir}/${snakelize(name)}.svg`
|
|
];
|
|
let stat;
|
|
for (const path of paths) {
|
|
try {
|
|
stat = await promises.lstat(path);
|
|
} catch (err) {
|
|
continue;
|
|
}
|
|
if (stat.isFile()) {
|
|
let svg = await promises.readFile(path, "utf-8");
|
|
const cleanupIdx = svg.indexOf("<svg");
|
|
if (cleanupIdx > 0) svg = svg.slice(cleanupIdx);
|
|
return typeof transform === "function" ? await transform(svg) : svg;
|
|
}
|
|
}
|
|
};
|
|
}
|
|
|
|
export { FileSystemIconLoader }; |