/* * SPDX-FileCopyrightText: syuilo and misskey-project * SPDX-License-Identifier: AGPL-3.0-only */ import type { ILocale, ParameterizedString } from '../../../locales/index.js'; type FlattenKeys = keyof { [K in keyof T as T[K] extends ILocale ? FlattenKeys extends infer C extends string ? `${K & string}.${C}` : never : T[K] extends TPrediction ? K : never]: T[K]; }; type ParametersOf> = TKey extends `${infer K}.${infer C}` // @ts-expect-error -- C は明らかに FlattenKeys になるが、型システムはここでは TKey がドット区切りであることのコンテキストを持たないので、型システムに合法にて示すことはできない。 ? ParametersOf : TKey extends keyof T ? T[TKey] extends ParameterizedString ? P : never : never; type Tsx = { readonly [K in keyof T as T[K] extends string ? never : K]: T[K] extends ParameterizedString ? (arg: { readonly [_ in P]: string | number }) => string // @ts-expect-error -- 証明省略 : Tsx; }; export class I18n { private tsxCache?: Tsx; private devMode: boolean; constructor(public locale: T, devMode = false) { this.devMode = devMode; //#region BIND this.t = this.t.bind(this); //#endregion } public get ts(): T { if (this.devMode) { class Handler implements ProxyHandler { get(target: TTarget, p: string | symbol): unknown { const value = target[p as keyof TTarget]; if (typeof value === 'object') { return new Proxy(value, new Handler()); } if (typeof value === 'string') { const parameters = Array.from(value.matchAll(/\{(\w+)\}/g), ([, parameter]) => parameter); if (parameters.length) { console.error(`Missing locale parameters: ${parameters.join(', ')} at ${String(p)}`); } return value; } console.error(`Unexpected locale key: ${String(p)}`); return p; } } return new Proxy(this.locale, new Handler()); } return this.locale; } public get tsx(): Tsx { if (this.devMode) { if (this.tsxCache) { return this.tsxCache; } class Handler implements ProxyHandler { get(target: TTarget, p: string | symbol): unknown { const value = target[p as keyof TTarget]; if (typeof value === 'object') { return new Proxy(value, new Handler()); } if (typeof value === 'string') { const quasis: string[] = []; const expressions: string[] = []; let cursor = 0; while (~cursor) { const start = value.indexOf('{', cursor); if (!~start) { quasis.push(value.slice(cursor)); break; } quasis.push(value.slice(cursor, start)); const end = value.indexOf('}', start); expressions.push(value.slice(start + 1, end)); cursor = end + 1; } if (!expressions.length) { console.error(`Unexpected locale key: ${String(p)}`); return () => value; } return (arg) => { let str = quasis[0]; for (let i = 0; i < expressions.length; i++) { if (!Object.hasOwn(arg, expressions[i])) { console.error(`Missing locale parameters: ${expressions[i]} at ${String(p)}`); } str += arg[expressions[i]] + quasis[i + 1]; } return str; }; } console.error(`Unexpected locale key: ${String(p)}`); return p; } } return this.tsxCache = new Proxy(this.locale, new Handler()) as unknown as Tsx; } if (this.tsxCache) { return this.tsxCache; } function build(target: ILocale): Tsx { const result = {} as Tsx; for (const k in target) { if (!Object.hasOwn(target, k)) { continue; } const value = target[k as keyof typeof target]; if (typeof value === 'object') { result[k] = build(value as ILocale); } else if (typeof value === 'string') { const quasis: string[] = []; const expressions: string[] = []; let cursor = 0; while (~cursor) { const start = value.indexOf('{', cursor); if (!~start) { quasis.push(value.slice(cursor)); break; } quasis.push(value.slice(cursor, start)); const end = value.indexOf('}', start); expressions.push(value.slice(start + 1, end)); cursor = end + 1; } if (!expressions.length) { continue; } result[k] = (arg) => { let str = quasis[0]; for (let i = 0; i < expressions.length; i++) { str += arg[expressions[i]] + quasis[i + 1]; } return str; }; } } return result; } return this.tsxCache = build(this.locale); } /** * @deprecated なるべくこのメソッド使うよりも ts 直接参照の方が vue のキャッシュ効いてパフォーマンスが良いかも */ public t>(key: TKey): string; /** * @deprecated なるべくこのメソッド使うよりも tsx 直接参照の方が vue のキャッシュ効いてパフォーマンスが良いかも */ public t>(key: TKey, args: { readonly [_ in ParametersOf]: string | number }): string; public t(key: string, args?: { readonly [_: string]: string | number }) { let str: string | ParameterizedString | ILocale = this.locale; for (const k of key.split('.')) { str = str[k]; if (this.devMode) { if (typeof str === 'undefined') { console.error(`Unexpected locale key: ${key}`); return key; } } } if (args) { if (this.devMode) { const missing = Array.from((str as string).matchAll(/\{(\w+)\}/g), ([, parameter]) => parameter).filter(parameter => !Object.hasOwn(args, parameter)); if (missing.length) { console.error(`Missing locale parameters: ${missing.join(', ')} at ${key}`); } } for (const [k, v] of Object.entries(args)) { const search = `{${k}}`; if (this.devMode) { if (!(str as string).includes(search)) { console.error(`Unexpected locale parameter: ${k} at ${key}`); } } str = (str as string).replace(search, v.toString()); } } return str; } } if (import.meta.vitest) { const { describe, expect, it } = import.meta.vitest; describe('i18n', () => { it('t', () => { const i18n = new I18n({ foo: 'foo', bar: { baz: 'baz', qux: 'qux {0}' as unknown as ParameterizedString<'0'>, quux: 'quux {0} {1}' as unknown as ParameterizedString<'0' | '1'>, }, }); expect(i18n.t('foo')).toBe('foo'); expect(i18n.t('bar.baz')).toBe('baz'); expect(i18n.tsx.bar.qux({ 0: 'hoge' })).toBe('qux hoge'); expect(i18n.tsx.bar.quux({ 0: 'hoge', 1: 'fuga' })).toBe('quux hoge fuga'); }); it('ts', () => { const i18n = new I18n({ foo: 'foo', bar: { baz: 'baz', qux: 'qux {0}' as unknown as ParameterizedString<'0'>, quux: 'quux {0} {1}' as unknown as ParameterizedString<'0' | '1'>, }, }); expect(i18n.ts.foo).toBe('foo'); expect(i18n.ts.bar.baz).toBe('baz'); }); it('tsx', () => { const i18n = new I18n({ foo: 'foo', bar: { baz: 'baz', qux: 'qux {0}' as unknown as ParameterizedString<'0'>, quux: 'quux {0} {1}' as unknown as ParameterizedString<'0' | '1'>, }, }); expect(i18n.tsx.bar.qux({ 0: 'hoge' })).toBe('qux hoge'); expect(i18n.tsx.bar.quux({ 0: 'hoge', 1: 'fuga' })).toBe('quux hoge fuga'); }); }); }