2023-01-06 13:48:43 +01:00
|
|
|
import { readFile } from 'fs-extra'
|
|
|
|
import { resolve } from 'pathe'
|
|
|
|
import type { ManifestOptions } from 'vite-plugin-pwa'
|
|
|
|
import { getEnv } from '../../config/env'
|
|
|
|
import { i18n } from '../../config/i18n'
|
|
|
|
import type { LocaleObject } from '#i18n'
|
|
|
|
|
2023-01-14 21:58:52 +01:00
|
|
|
// We have to extend the ManifestOptions interface from 'vite-plugin-pwa'
|
|
|
|
// as that interface doesn't define the share_target field of Web App Manifests.
|
|
|
|
interface ExtendedManifestOptions extends ManifestOptions {
|
|
|
|
share_target: {
|
|
|
|
action: string
|
|
|
|
method: string
|
|
|
|
enctype: string
|
|
|
|
params: {
|
|
|
|
text: string
|
|
|
|
url: string
|
|
|
|
files: [{
|
|
|
|
name: string
|
|
|
|
accept: string[]
|
|
|
|
}]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export type LocalizedWebManifest = Record<string, Partial<ExtendedManifestOptions>>
|
2023-01-06 13:48:43 +01:00
|
|
|
|
|
|
|
export const pwaLocales = i18n.locales as LocaleObject[]
|
|
|
|
|
2023-01-14 21:58:52 +01:00
|
|
|
type WebManifestEntry = Pick<ExtendedManifestOptions, 'name' | 'short_name' | 'description'>
|
|
|
|
type RequiredWebManifestEntry = Required<WebManifestEntry & Pick<ExtendedManifestOptions, 'dir' | 'lang'>>
|
2023-01-06 13:48:43 +01:00
|
|
|
|
|
|
|
export const createI18n = async (): Promise<LocalizedWebManifest> => {
|
|
|
|
const { env } = await getEnv()
|
|
|
|
const envName = `${env === 'release' ? '' : `(${env})`}`
|
|
|
|
const { pwa } = await readI18nFile('en-US.json')
|
|
|
|
|
|
|
|
const defaultManifest: Required<WebManifestEntry> = pwa.webmanifest[env]
|
|
|
|
|
2023-01-20 16:25:28 +01:00
|
|
|
if (process.env.TAURI_PLATFORM) {
|
|
|
|
return {
|
|
|
|
'en-US': {
|
|
|
|
...defaultManifest,
|
|
|
|
lang: 'en-US',
|
|
|
|
dir: 'ltr',
|
|
|
|
scope: '/',
|
|
|
|
id: '/',
|
|
|
|
start_url: '/',
|
|
|
|
display: 'standalone',
|
|
|
|
background_color: '#ffffff',
|
|
|
|
theme_color: '#ffffff',
|
|
|
|
icons: [
|
|
|
|
{
|
|
|
|
src: 'pwa-192x192.png',
|
|
|
|
sizes: '192x192',
|
|
|
|
type: 'image/png',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
src: 'pwa-512x512.png',
|
|
|
|
sizes: '512x512',
|
|
|
|
type: 'image/png',
|
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-01-06 13:48:43 +01:00
|
|
|
const locales: RequiredWebManifestEntry[] = await Promise.all(
|
|
|
|
pwaLocales
|
|
|
|
.filter(l => l.code !== 'en-US')
|
|
|
|
.map(async ({ code, dir = 'ltr', file }) => {
|
|
|
|
// read locale file
|
|
|
|
const { pwa, app_name, app_desc_short } = await readI18nFile(file!)
|
|
|
|
const entry: WebManifestEntry = pwa?.webmanifest?.[env] ?? {}
|
|
|
|
if (!entry.name && app_name)
|
|
|
|
entry.name = dir === 'rtl' ? `${envName} ${app_name}` : `${app_name} ${envName}`
|
|
|
|
|
|
|
|
if (!entry.short_name && app_name)
|
|
|
|
entry.short_name = dir === 'rtl' ? `${envName} ${app_name}` : `${app_name} ${envName}`
|
|
|
|
|
|
|
|
if (!entry.description && app_desc_short)
|
|
|
|
entry.description = app_desc_short
|
|
|
|
|
|
|
|
return <RequiredWebManifestEntry>{
|
|
|
|
...defaultManifest,
|
|
|
|
...entry,
|
|
|
|
lang: code,
|
|
|
|
dir,
|
|
|
|
}
|
|
|
|
}),
|
|
|
|
)
|
|
|
|
locales.push({
|
|
|
|
...defaultManifest,
|
|
|
|
lang: 'en-US',
|
|
|
|
dir: 'ltr',
|
|
|
|
})
|
|
|
|
return locales.reduce((acc, { lang, dir, name, short_name, description }) => {
|
|
|
|
acc[lang] = {
|
|
|
|
scope: '/',
|
|
|
|
id: '/',
|
|
|
|
start_url: '/',
|
|
|
|
display: 'standalone',
|
|
|
|
lang,
|
|
|
|
name,
|
|
|
|
short_name,
|
|
|
|
description,
|
|
|
|
dir,
|
2023-01-06 22:04:17 +01:00
|
|
|
background_color: '#ffffff',
|
2023-01-06 13:48:43 +01:00
|
|
|
theme_color: '#ffffff',
|
|
|
|
icons: [
|
|
|
|
{
|
|
|
|
src: 'pwa-192x192.png',
|
|
|
|
sizes: '192x192',
|
|
|
|
type: 'image/png',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
src: 'pwa-512x512.png',
|
|
|
|
sizes: '512x512',
|
|
|
|
type: 'image/png',
|
|
|
|
},
|
|
|
|
],
|
2023-01-14 21:58:52 +01:00
|
|
|
share_target: {
|
|
|
|
action: '/web-share-target',
|
|
|
|
method: 'POST',
|
|
|
|
enctype: 'multipart/form-data',
|
|
|
|
params: {
|
|
|
|
text: 'text',
|
|
|
|
url: 'text',
|
|
|
|
files: [
|
|
|
|
{
|
|
|
|
name: 'files',
|
|
|
|
accept: ['image/*', 'video/*'],
|
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
},
|
2023-01-06 13:48:43 +01:00
|
|
|
}
|
2023-01-10 21:35:34 +01:00
|
|
|
acc[`${lang}-dark`] = {
|
|
|
|
scope: '/',
|
|
|
|
id: '/',
|
|
|
|
start_url: '/',
|
|
|
|
display: 'standalone',
|
|
|
|
lang,
|
|
|
|
name,
|
|
|
|
short_name,
|
|
|
|
description,
|
|
|
|
dir,
|
|
|
|
background_color: '#111111',
|
|
|
|
theme_color: '#111111',
|
|
|
|
icons: [
|
|
|
|
{
|
|
|
|
src: 'pwa-192x192.png',
|
|
|
|
sizes: '192x192',
|
|
|
|
type: 'image/png',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
src: 'pwa-512x512.png',
|
|
|
|
sizes: '512x512',
|
|
|
|
type: 'image/png',
|
|
|
|
},
|
|
|
|
],
|
2023-01-14 21:58:52 +01:00
|
|
|
share_target: {
|
|
|
|
action: '/web-share-target',
|
|
|
|
method: 'POST',
|
|
|
|
enctype: 'multipart/form-data',
|
|
|
|
params: {
|
|
|
|
text: 'text',
|
|
|
|
url: 'text',
|
|
|
|
files: [
|
|
|
|
{
|
|
|
|
name: 'files',
|
|
|
|
accept: ['image/*', 'video/*'],
|
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
},
|
2023-01-10 21:35:34 +01:00
|
|
|
}
|
2023-01-06 13:48:43 +01:00
|
|
|
|
|
|
|
return acc
|
|
|
|
}, {} as LocalizedWebManifest)
|
|
|
|
}
|
|
|
|
|
|
|
|
async function readI18nFile(file: string) {
|
|
|
|
return JSON.parse(Buffer.from(
|
|
|
|
await readFile(resolve(`./locales/${file}`), 'utf-8'),
|
|
|
|
).toString())
|
|
|
|
}
|