diff --git a/ts/util/i18n/functions/getMessage.ts b/ts/util/i18n/functions/getMessage.ts index 574d4d49a..6c7159c7d 100644 --- a/ts/util/i18n/functions/getMessage.ts +++ b/ts/util/i18n/functions/getMessage.ts @@ -13,6 +13,7 @@ import { formatMessageWithArgs } from './formatMessageWithArgs'; import { getRawMessage } from './getRawMessage'; import { inEnglish } from './inEnglish'; import { stripped } from './stripped'; +import { localizeFromOld, type StringArgsRecord } from '../localizedString'; /** * Checks if a string contains a dynamic variable. @@ -46,14 +47,15 @@ function getMessageDefault ): R | T { try { - const rawMessage = getRawMessage(...([token, args] as GetMessageArgs)); - - /** If a localized string does not have any arguments to substitute it is returned with no changes. */ - if (!isStringWithArgs(rawMessage)) { - return rawMessage; - } - - return formatMessageWithArgs(rawMessage, args as ArgsRecord); + return localizeFromOld(token as T, args as StringArgsRecord).toString() as T | R; + // const rawMessage = getRawMessage(...([token, args] as GetMessageArgs)); + // + // /** If a localized string does not have any arguments to substitute it is returned with no changes. */ + // if (!isStringWithArgs(rawMessage)) { + // return rawMessage; + // } + // + // return formatMessageWithArgs(rawMessage, args as ArgsRecord); } catch (error) { i18nLog(error.message); return token as R; diff --git a/ts/util/i18n/localizedString.ts b/ts/util/i18n/localizedString.ts new file mode 100644 index 000000000..ee5abfcca --- /dev/null +++ b/ts/util/i18n/localizedString.ts @@ -0,0 +1,297 @@ +import { getFallbackDictionary, getTranslationDictionary } from './translationDictionaries'; +import { en } from '../../localization/locales'; +import { getLocale } from './shared'; +import { LOCALE_DEFAULTS } from '../../localization/constants'; +import { deSanitizeHtmlTags, sanitizeArgs } from '../../components/basic/I18n'; +import type { LocalizerDictionary } from '../../types/Localizer'; + +type PluralKey = 'count'; + +type ArgString = `${string}{${string}}${string}`; +type RawString = ArgString | string; + +type PluralString = `{${PluralKey}, plural, one [${RawString}] other [${RawString}]}`; + +// type LocalizedString = string; + +type GenericLocalizedDictionary = Record; + +type TokenString = keyof Dict extends string + ? keyof Dict + : never; + +// type GenericArgsRecord = Record; + +/** The dynamic arguments in a localized string */ +type StringArgs = + /** If a string follows the plural format use its plural variable name and recursively check for + * dynamic args inside all plural forms */ + T extends `{${infer PluralVar}, plural, one [${infer PluralOne}] other [${infer PluralOther}]}` + ? PluralVar | StringArgs | StringArgs + : /** If a string segment follows the variable form parse its variable name and recursively + * check for more dynamic args */ + T extends `${string}{${infer Var}}${infer Rest}` + ? Var | StringArgs + : never; + +export type StringArgsRecord = Record, string | number>; + +// TODO: move this to a test file +// +// const stringArgsTestStrings = { +// none: 'test', +// one: 'test{count}', +// two: 'test {count} second {another}', +// three: 'test {count} second {another} third {third}', +// four: 'test {count} second {another} third {third} fourth {fourth}', +// five: 'test {count} second {another} third {third} fourth {fourth} fifth {fifth}', +// twoConnected: 'test {count}{another}', +// threeConnected: '{count}{another}{third}', +// } as const; +// +// const stringArgsTestResults = { +// one: { count: 'count' }, +// two: { count: 'count', another: 'another' }, +// three: { count: 'count', another: 'another', third: 'third' }, +// four: { count: 'count', another: 'another', third: 'third', fourth: 'fourth' }, +// five: { count: 'count', another: 'another', third: 'third', fourth: 'fourth', fifth: 'fifth' }, +// twoConnected: { count: 'count', another: 'another' }, +// threeConnected: { count: 'count', another: 'another', third: 'third' }, +// } as const; +// +// let st0: Record, string>; +// const st1: Record, string> = stringArgsTestResults.one; +// const st2: Record, string> = stringArgsTestResults.two; +// const st3: Record< +// StringArgs, +// string +// > = stringArgsTestResults.three; +// const st4: Record< +// StringArgs, +// string +// > = stringArgsTestResults.four; +// const st5: Record< +// StringArgs, +// string +// > = stringArgsTestResults.five; +// const st6: Record< +// StringArgs, +// string +// > = stringArgsTestResults.twoConnected; +// const st7: Record< +// StringArgs, +// string +// > = stringArgsTestResults.threeConnected; +// +// const results = [st0, st1, st2, st3, st4, st5, st6, st7]; + +// Above is testing stuff + +function getPluralKey(string: PluralString): R { + const match = /{(\w+), plural, one \[.+\] other \[.+\]}/g.exec(string); + return match?.[1] as R; +} + +function getStringForCardinalRule( + localizedString: string, + cardinalRule: Intl.LDMLPluralRule +): string | undefined { + // TODO: investigate if this is the best way to handle regex like this + const cardinalPluralRegex: Record = { + zero: /zero \[(.*?)\]/g, + one: /one \[(.*?)\]/g, + two: /two \[(.*?)\]/g, + few: /few \[(.*?)\]/g, + many: /many \[(.*?)\]/g, + other: /other \[(.*?)\]/g, + }; + const regex = cardinalPluralRegex[cardinalRule]; + const match = regex.exec(localizedString); + return match?.[1] ?? undefined; +} + +const isPluralForm = (localizedString: string): localizedString is PluralString => + /{\w+, plural, one \[.+\] other \[.+\]}/g.test(localizedString); + +/** + * Checks if a string contains a dynamic variable. + * @param localizedString - The string to check. + * @returns `true` if the string contains a dynamic variable, otherwise `false`. + */ +const isStringWithArgs = (localizedString: string): localizedString is ArgString => + localizedString.includes('{'); + +/** + * Logs an i18n message to the console. + * @param message - The message to log. + */ +export function i18nLog(message: string) { + if (typeof window !== 'undefined') { + // eslint-disable-next-line no-console + (window?.log?.error ?? console.log)(`i18n: ${message}`); + } else { + // eslint-disable-next-line no-console + console.log(`i18n: ${message}`); + } +} + +const isReplaceLocalizedStringsWithKeysEnabled = () => + !!(typeof window !== 'undefined' && window?.sessionFeatureFlags?.replaceLocalizedStringsWithKeys); + +export class LocalizedStringBuilder< + Dict extends GenericLocalizedDictionary, + T extends TokenString, +> extends String { + private readonly token: T; + private args?: StringArgsRecord; + private isStripped = false; + private isEnglishForced = false; + + private readonly renderStringAsToken = isReplaceLocalizedStringsWithKeysEnabled(); + + constructor(token: T) { + super(token); + this.token = token; + } + + public toString(): string { + try { + if (this.renderStringAsToken) { + return this.token; + } + + const rawString = this.getRawString(); + const str = isStringWithArgs(rawString) ? this.formatStringWithArgs(rawString) : rawString; + + if (this.isStripped) { + return this.postProcessStrippedString(str); + } + + return str; + } catch (error) { + i18nLog(error); + return this.token; + } + } + + withArgs(args: StringArgsRecord): Omit { + this.args = args; + return this; + } + + forceEnglish(): Omit { + this.isEnglishForced = true; + return this; + } + + strip(): Omit { + const sanitizedArgs = this.args ? sanitizeArgs(this.args, '\u200B') : undefined; + if (sanitizedArgs) { + this.args = sanitizedArgs as StringArgsRecord; + } + this.isStripped = true; + + return this; + } + + private postProcessStrippedString(str: string): string { + const strippedString = str.replaceAll(/<[^>]*>/g, ''); + return deSanitizeHtmlTags(strippedString, '\u200B'); + } + + private getRawString(): RawString | TokenString { + try { + if (this.renderStringAsToken) { + return this.token; + } + + const dict: GenericLocalizedDictionary = this.isEnglishForced + ? en + : getTranslationDictionary(); + + let localizedString = dict[this.token]; + + if (!localizedString) { + i18nLog(`Attempted to get translation for nonexistent key: '${this.token}'`); + + localizedString = (getFallbackDictionary() as GenericLocalizedDictionary)[this.token]; + + if (!localizedString) { + i18nLog( + `Attempted to get translation for nonexistent key: '${this.token}' in fallback dictionary` + ); + return this.token; + } + } + + return isPluralForm(localizedString) + ? this.resolvePluralString(localizedString) + : localizedString; + } catch (error) { + i18nLog(error.message); + return this.token; + } + } + + private resolvePluralString(str: PluralString): string { + const pluralKey = getPluralKey(str); + + // This should not be possible, but we need to handle it in case it does happen + if (!pluralKey) { + i18nLog(`Attempted to get nonexistent pluralKey for plural form string '${str}'`); + return this.token; + } + + let num = this.args?.[pluralKey as keyof StringArgsRecord]; + + if (num === undefined) { + i18nLog(`Attempted to get plural count for missing argument '${pluralKey}'`); + num = 0; + } + + if (typeof num !== 'number') { + i18nLog(`Attempted to get plural count for argument '${pluralKey}' which is not a number`); + num = parseInt(num, 10); + if (Number.isNaN(num)) { + i18nLog( + `Attempted to get parsed plural count for argument '${pluralKey}' which is not a number` + ); + num = 0; + } + } + + const currentLocale = getLocale(); + const cardinalRule = new Intl.PluralRules(currentLocale).select(num); + + const pluralString = getStringForCardinalRule(str, cardinalRule); + + if (!pluralString) { + i18nLog(`Plural string not found for cardinal '${cardinalRule}': '${str}'`); + return this.token; + } + + return pluralString.replaceAll('#', `${num}`); + } + + private formatStringWithArgs(str: ArgString): string { + /** Find and replace the dynamic variables in a localized string and substitute the variables with the provided values */ + return str.replace(/\{(\w+)\}/g, (match, arg: string) => { + const matchedArg = this.args + ? this.args[arg as keyof StringArgsRecord].toString() + : undefined; + + return matchedArg ?? LOCALE_DEFAULTS[arg as keyof typeof LOCALE_DEFAULTS] ?? match; + }); + } +} + +export function localize>(token: T) { + return new LocalizedStringBuilder(token); +} + +export function localizeFromOld>( + token: T, + args: StringArgsRecord +) { + return new LocalizedStringBuilder(token).withArgs(args); +} diff --git a/ts/util/i18n/translationDictionaries.ts b/ts/util/i18n/translationDictionaries.ts index 72c5cb7a4..fca76da84 100644 --- a/ts/util/i18n/translationDictionaries.ts +++ b/ts/util/i18n/translationDictionaries.ts @@ -29,3 +29,7 @@ export function getTranslationDictionary(): LocalizerDictionary { i18nLog('getTranslationDictionary: dictionary not init yet. Using en.'); return en; } + +export function getFallbackDictionary(): LocalizerDictionary { + return en; +}