1
0
mirror of https://github.com/kremalicious/portfolio.git synced 2025-01-05 11:25:00 +01:00
portfolio/scripts/favicon.ts

95 lines
2.8 KiB
TypeScript
Raw Normal View History

import fs from 'fs'
2023-02-05 23:52:47 +01:00
import path from 'path'
2022-11-16 20:04:06 +01:00
import sharp from 'sharp'
import ico from 'sharp-ico'
2022-11-16 20:04:06 +01:00
const imagesSourcePath = path.resolve(path.join(process.cwd(), 'src', 'images'))
const faviconSource = `${imagesSourcePath}/favicon-512.png`
const faviconSourceSvg = `${imagesSourcePath}/favicon.svg`
2022-11-16 20:04:06 +01:00
const outputWebRoot = path.resolve(path.join(process.cwd(), 'public'))
const outputManifest = path.resolve(
path.join(process.cwd(), 'public', 'manifest')
)
// All the sizes and meta we'll need
// https://evilmartians.com/chronicles/how-to-favicon-in-2021-six-files-that-fit-most-needs
const sizes = [32, 180, 192, 512]
const outputMeta = `
2022-11-16 23:36:38 +01:00
<link rel="icon" href="/favicon.ico" sizes="any" />
<link rel="icon" href="/favicon.svg" type="image/svg+xml" />
<link rel="apple-touch-icon" href="/apple-touch-icon.png" />
<link rel="manifest" href="/manifest/manifest.webmanifest" />
2022-11-16 20:04:06 +01:00
`
function createManifest(iconsizes: number[]) {
const manifest = {
name: 'matthias kretschmann',
2022-11-16 21:36:25 +01:00
short_name: 'mk',
display: 'standalone',
2022-11-16 22:46:50 +01:00
start_url: '/',
2022-11-16 20:04:06 +01:00
icons: iconsizes.map((size) => ({
src: `/manifest/favicon-${size}.png`,
type: 'image/png',
sizes: `${size}x${size}`
}))
}
2022-11-16 20:04:06 +01:00
fs.writeFileSync(
`${outputManifest}/manifest.webmanifest`,
JSON.stringify(manifest)
)
}
2022-11-16 21:36:25 +01:00
function nuke() {
fs.rmSync(outputManifest, { recursive: true, force: true })
fs.rmSync(`${outputWebRoot}/apple-touch-icon.png`, { force: true })
fs.rmSync(`${outputWebRoot}/favicon.ico`, { force: true })
fs.rmSync(`${outputWebRoot}/favicon.svg`, { force: true })
fs.mkdirSync(outputManifest, { recursive: true })
}
async function buildFavicons() {
try {
2022-11-16 20:04:06 +01:00
// Nuke all & create output folder first
2022-11-16 21:36:25 +01:00
nuke()
2022-11-16 20:04:06 +01:00
// copy over the svg, as it's handcrafted
fs.copyFileSync(faviconSourceSvg, `${outputWebRoot}/favicon.svg`)
2022-11-16 20:04:06 +01:00
// generate all the rest
await Promise.all(
sizes.map(async (size) => {
let destination = `${outputManifest}/favicon-${size}.png`
if (size === 180) destination = `${outputWebRoot}/apple-touch-icon.png`
2022-11-16 21:36:25 +01:00
// 32px size only used for favicon.ico
2022-11-16 20:04:06 +01:00
if (size === 32) {
await ico.sharpsToIco(
[sharp(faviconSource)],
`${outputWebRoot}/favicon.ico`,
{ sizes: [32, 24, 16], resizeOptions: {} }
)
fs.rmSync(destination, { force: true })
} else {
await sharp(faviconSource).resize(size, size).toFile(destination)
}
2022-11-16 20:04:06 +01:00
})
)
2022-11-16 20:04:06 +01:00
// write out manifest
createManifest([192, 512])
2022-11-16 20:04:06 +01:00
console.log(`
-----------------------------
Favicon generation complete!
-----------------------------
Add this to src/components/Meta/Favicon.tsx:
${outputMeta}
`)
} catch (error) {
console.error(error.message)
}
}
buildFavicons()