Skip to content

Commit e299cc3

Browse files
enhance(i18n): i18nRedirectorにdefaultLocaleのページのmetaを反映
1 parent deaaf47 commit e299cc3

File tree

3 files changed

+73
-22
lines changed

3 files changed

+73
-22
lines changed

.vitepress/config/index.ts

+22-3
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@ import { shared } from './shared';
55
import { ja } from './ja';
66
import { en } from './en';
77

8-
import { genI18nRedirector } from '../scripts/gen-i18n-redirector';
8+
import { createGenI18nRedirector } from '../scripts/gen-i18n-redirector';
99

1010
const locales = {
1111
ja: { label: '日本語', ...ja },
@@ -16,14 +16,24 @@ export const mainLocale = 'ja' as const satisfies keyof typeof locales;
1616

1717
export const baseUrl = 'https://aiscript-dev.github.io';
1818

19+
const { registerRouteMeta, genI18nRedirector } = createGenI18nRedirector();
20+
1921
export default defineConfig({
2022
...shared,
2123
locales,
2224
transformHead(context) {
23-
const head: HeadConfig[] = [];
25+
const head: HeadConfig[] = [ ...context.head ];
2426
if (!context.pageData.isNotFound) {
2527
const localesRegex = new RegExp(`^/(${Object.keys(locales).join('|')})`);
26-
const canonical = '/' + context.page.replace(/\.md$/, '.html').replace(/\/index\.html$/, '/');
28+
const canonical = '/' + context.page.replace(/index\.(md|html)$/, '').replace(/\.md$/, context.siteConfig.cleanUrls ? '' : '.html');
29+
const defaultLocalePath = canonical.replace(localesRegex, `/${mainLocale}`);
30+
31+
if (canonical.startsWith(`/${mainLocale}`)) {
32+
registerRouteMeta(canonical.slice(1), {
33+
title: context.title,
34+
description: context.description,
35+
});
36+
}
2737

2838
for (const locale of Object.keys(locales)) {
2939
const localePath = canonical.replace(localesRegex, `/${locale}`);
@@ -37,6 +47,15 @@ export default defineConfig({
3747
},
3848
]);
3949
}
50+
51+
head.push([
52+
'link',
53+
{
54+
rel: 'alternate',
55+
hreflang: 'x-default',
56+
href: baseUrl + defaultLocalePath,
57+
},
58+
]);
4059

4160
head.push([
4261
'link',

.vitepress/config/shared.ts

+1
Original file line numberDiff line numberDiff line change
@@ -28,6 +28,7 @@ export const shared = defineConfig({
2828
head: [
2929
['link', { rel: 'icon', href: '/favicon.ico' }],
3030
['link', { rel: 'apple-touch-icon', sizes: '180x180', href: '/icons/apple-touch-icon.png' }],
31+
['meta', { property: 'og:site_name', content: 'AiScript' }],
3132
],
3233

3334
markdown: {
+50-19
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,13 @@
1+
// @ts-expect-error Node
12
import fs from 'fs';
23
import { mainLocale, baseUrl } from '../config';
34
import type { SiteConfig } from 'vitepress';
45

6+
type RouteMeta = {
7+
title: string;
8+
description?: string;
9+
};
10+
511
async function createFile(path: string, content: string) {
612
const dir = path.replace(/\/[^/]+$/, '');
713
await fs.promises.writeFile(path, content).catch((err) => {
@@ -11,36 +17,61 @@ async function createFile(path: string, content: string) {
1117
});
1218
}
1319

14-
export async function genI18nRedirector(siteConfig: SiteConfig) {
15-
const routes = siteConfig.pages
16-
.filter((page) => page.startsWith(`${mainLocale}/`))
17-
.map((page) => page.replace(new RegExp(`^${mainLocale}\/`), '').replace(/\.md$/, '.html'));
20+
export function createGenI18nRedirector() {
21+
const routeMeta = new Map<string, RouteMeta>();
22+
23+
function registerRouteMeta(route: string, meta: RouteMeta) {
24+
routeMeta.set(route, meta);
25+
}
26+
27+
async function genI18nRedirector(siteConfig: SiteConfig) {
28+
const genStartedAt = performance.now();
29+
30+
const routes = siteConfig.pages
31+
.filter((page) => page.startsWith(`${mainLocale}/`));
32+
33+
const promises = routes.map(async (route) => {
34+
const routePath = route.replace(new RegExp(`^${mainLocale}\/`), '');
35+
const routePathForWrite = routePath.replace(/\.md$/, '.html');
36+
const routePathForRender = routePath.replace(/index\.(md|html)$/, '').replace(/\.md$/, siteConfig.cleanUrls ? '' : '.html');
1837

19-
const promises = routes.map((route) => {
20-
const localeNames = Object.keys(siteConfig.site.locales);
21-
const routeForRender = route.replace(/index\.html$/, '');
22-
const linkAlternate = localeNames.map((name) => `<link rel="alternate" hreflang="${siteConfig.site.locales[name].lang || name}" href="${baseUrl}/${name}/${routeForRender}">`).join('\n ');
23-
const fallbackLinks = localeNames.map((name) => `<a href="${baseUrl}/${name}/${routeForRender}">${siteConfig.site.locales[name].label}</a>`).join(', ');
24-
const content = `<!DOCTYPE html>
25-
<html lang="${siteConfig.site.locales[mainLocale].lang || 'ja-JP'}">
38+
let title = 'Redirecting...';
39+
let description: string | null = null;
40+
if (routeMeta.has(`${mainLocale}/${routePathForRender}`)) {
41+
title = routeMeta.get(`${mainLocale}/${routePathForRender}`)?.title ?? title;
42+
description = routeMeta.get(`${mainLocale}/${routePathForRender}`)?.description ?? null;
43+
}
44+
45+
const localeNames = Object.keys(siteConfig.site.locales);
46+
const linkAlternate = localeNames.map((name) => `<link rel="alternate" hreflang="${siteConfig.site.locales[name].lang || name}" href="${baseUrl}/${name}/${routePathForRender}">`).join('\n ');
47+
const fallbackLinks = localeNames.map((name) => `<a href="${baseUrl}/${name}/${routePathForRender}">${siteConfig.site.locales[name].label}</a>`).join(', ');
48+
const content = `<!DOCTYPE html>
49+
<html>
2650
<head>
2751
<meta charset="UTF-8">
2852
<meta name="viewport" content="width=device-width, initial-scale=1.0">
29-
<title>Redirecting...</title>
53+
<title>${title}</title>${description ? `\n <meta name="description" content="${description}">\n` : ''}
3054
${linkAlternate}
31-
<link rel="alternate" hreflang="x-default" href="${baseUrl}/${mainLocale}/${routeForRender}">
32-
<link rel="canonical" href="${baseUrl}/${mainLocale}/${routeForRender}">
55+
<link rel="alternate" hreflang="x-default" href="${baseUrl}/${mainLocale}/${routePathForRender}">
56+
<link rel="canonical" href="${baseUrl}/${mainLocale}/${routePathForRender}">
3357
<script type="text/javascript">const s = ${JSON.stringify(localeNames)}; const d = localStorage.getItem('ais:locale'); if (d) { location.replace('/' + d + location.pathname + location.search + location.hash); } else if (s.includes(navigator.language.split("-")[0])) { location.replace('/' + navigator.language.split("-")[0] + location.pathname + location.search + location.hash); } else { location.replace('/ja' + location.pathname + location.search + location.hash); }</script>
3458
</head>
3559
<body>
3660
<noscript>${fallbackLinks}</noscript>
3761
</body>
3862
</html>
3963
`;
40-
return createFile(`${siteConfig.outDir}/${route}`, content);
41-
});
42-
43-
await Promise.allSettled(promises);
64+
await createFile(`${siteConfig.outDir}/${routePathForWrite}`, content);
65+
});
66+
67+
await Promise.allSettled(promises);
68+
69+
const genFinishedAt = performance.now();
70+
console.log(`I18n redirector generated in ${Math.round((genFinishedAt - genStartedAt) / 10) / 100}s`);
71+
}
4472

45-
console.log('I18n redirector generated');
73+
return {
74+
registerRouteMeta,
75+
genI18nRedirector,
76+
};
4677
}

0 commit comments

Comments
 (0)