diff --git a/src/cli.ts b/src/cli.ts index 877ef47..b26daae 100644 --- a/src/cli.ts +++ b/src/cli.ts @@ -11,29 +11,34 @@ function splitList(value: string): number[] { program .argument("", "File or folder to process") .option( - "--png-quality ", + "-p, --png-quality ", "Set PNG quality", Number, DEFAULTS.pngQuality ) .option( - "--manifest-icon-sizes ", + "-m, --manifest-icon-sizes ", "Set manifest icon sizes (comma-separated)", splitList, DEFAULTS.manifestIconSizes ) .option( - "--apple-icon-sizes ", + "-a, --apple-icon-sizes ", "Set Apple icon sizes (comma-separated)", splitList, DEFAULTS.appleIconSizes ) .option( - "--fav-icon-sizes ", + "-s, --fav-icon-sizes ", "Set favicon sizes (comma-separated)", splitList, DEFAULTS.favIconSizes ) + .option( + "--no-inline-manifest-icons", + "Disable inlining manifest icons", + DEFAULTS.inlineManifestIcons + ) .parse(process.argv) // Parse and normalize the options diff --git a/src/factory/manifest.ts b/src/factory/manifest.ts index c795469..add686d 100644 --- a/src/factory/manifest.ts +++ b/src/factory/manifest.ts @@ -1,5 +1,5 @@ import { promises as fs } from "node:fs" -import { basename } from "node:path" +import { extname } from "node:path" import { readPackageUp } from "read-package-up" import sharp from "sharp" @@ -7,6 +7,28 @@ import sharp from "sharp" import type { Options } from "../options.js" import type { WebManifest, WebManifestIcon } from "../types.js" +/** + * Converts an image file to a base64 data URL string. + * @param filePath - The path to the image file. + * @returns A promise that resolves to the base64 data URL string. + */ +async function inlineIcon(filePath: string): Promise { + try { + const data = await fs.readFile(filePath) + const ext = extname(filePath).slice(1) // Get the file extension without the dot + const mimeType = `image/${ext}` + const base64Data = data.toString("base64") + const dataUrl = `data:${mimeType};base64,${base64Data}` + return dataUrl + } catch (error) { + const wrapped = + error instanceof Error + ? new Error(`Failed to read file: ${error.message}`) + : new Error("An unknown error occurred") + throw wrapped + } +} + export async function generateWebManifest( filePrefix: string, svgContent: string, @@ -16,7 +38,7 @@ export async function generateWebManifest( const icons: WebManifestIcon[] = [] const manifestData: WebManifest = { name: pkg?.packageJson.description, - start_url: ".", + start_url: "/", scope: "/", display: "browser", icons @@ -24,11 +46,6 @@ export async function generateWebManifest( for (const size of options.manifestIconSizes) { const pngFilePath = `${filePrefix}-pwa-${size}.png` - icons.push({ - src: basename(pngFilePath), - type: "image/png", - sizes: `${size}x${size}` - }) await sharp(Buffer.from(svgContent)) .resize(size, size) @@ -39,6 +56,14 @@ export async function generateWebManifest( quality: options.pngQuality }) .toFile(pngFilePath) + + icons.push({ + src: options.inlineManifestIcons + ? await inlineIcon(pngFilePath) + : basename(pngFilePath), + type: "image/png", + sizes: `${size}x${size}` + }) } const manifestPath = `${filePrefix}.webmanifest` diff --git a/src/options.ts b/src/options.ts index 13cfcde..2d2e462 100644 --- a/src/options.ts +++ b/src/options.ts @@ -2,7 +2,8 @@ export const DEFAULTS = { pngQuality: 95, manifestIconSizes: [192, 512], appleIconSizes: [152, 167, 180], - favIconSizes: [16, 32] + favIconSizes: [16, 32], + inlineManifestIcons: true } export type Options = typeof DEFAULTS diff --git a/test/sebastian-software.webmanifest b/test/sebastian-software.webmanifest index 19cdf51..147e344 100644 --- a/test/sebastian-software.webmanifest +++ b/test/sebastian-software.webmanifest @@ -1,16 +1,16 @@ { "name": "Effective Favicon Generator", - "start_url": ".", + "start_url": "/", "scope": "/", "display": "browser", "icons": [ { - "src": "sebastian-software-pwa-192.png", + "src": "", "type": "image/png", "sizes": "192x192" }, { - "src": "sebastian-software-pwa-512.png", + "src": "", "type": "image/png", "sizes": "512x512" }