Skip to content

Commit

Permalink
feat: added support for inline manifest icons
Browse files Browse the repository at this point in the history
  • Loading branch information
swernerx committed Aug 2, 2024
1 parent ada127e commit 6618030
Show file tree
Hide file tree
Showing 4 changed files with 46 additions and 15 deletions.
13 changes: 9 additions & 4 deletions src/cli.ts
Original file line number Diff line number Diff line change
Expand Up @@ -11,29 +11,34 @@ function splitList(value: string): number[] {
program
.argument("<fileOrFolder>", "File or folder to process")
.option(
"--png-quality <quality>",
"-p, --png-quality <quality>",
"Set PNG quality",
Number,
DEFAULTS.pngQuality
)
.option(
"--manifest-icon-sizes <sizes>",
"-m, --manifest-icon-sizes <sizes>",
"Set manifest icon sizes (comma-separated)",
splitList,
DEFAULTS.manifestIconSizes
)
.option(
"--apple-icon-sizes <sizes>",
"-a, --apple-icon-sizes <sizes>",
"Set Apple icon sizes (comma-separated)",
splitList,
DEFAULTS.appleIconSizes
)
.option(
"--fav-icon-sizes <sizes>",
"-s, --fav-icon-sizes <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
Expand Down
39 changes: 32 additions & 7 deletions src/factory/manifest.ts
Original file line number Diff line number Diff line change
@@ -1,12 +1,34 @@
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"

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<string> {
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,
Expand All @@ -16,19 +38,14 @@ export async function generateWebManifest(
const icons: WebManifestIcon[] = []
const manifestData: WebManifest = {
name: pkg?.packageJson.description,
start_url: ".",
start_url: "/",
scope: "/",
display: "browser",
icons
}

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)
Expand All @@ -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`
Expand Down
3 changes: 2 additions & 1 deletion src/options.ts
Original file line number Diff line number Diff line change
Expand Up @@ -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
6 changes: 3 additions & 3 deletions test/sebastian-software.webmanifest
Original file line number Diff line number Diff line change
@@ -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"
}
Expand Down

0 comments on commit 6618030

Please sign in to comment.