146 lines
3.9 KiB
TypeScript
146 lines
3.9 KiB
TypeScript
|
// Copyright 2018 Signal Messenger, LLC
|
||
|
// SPDX-License-Identifier: AGPL-3.0-only
|
||
|
|
||
|
import type { IntlShape } from 'react-intl';
|
||
|
import { createIntl, createIntlCache } from 'react-intl';
|
||
|
import type { LocaleMessageType, LocaleMessagesType } from '../types/I18N';
|
||
|
import type {
|
||
|
LocalizerType,
|
||
|
ICUStringMessageParamsByKeyType,
|
||
|
} from '../types/Util';
|
||
|
import { strictAssert } from './assert';
|
||
|
import * as log from '../logging/log';
|
||
|
import * as Errors from '../types/errors';
|
||
|
import { Environment, getEnvironment } from '../environment';
|
||
|
import { bidiIsolate } from './unicodeBidi';
|
||
|
|
||
|
export function isLocaleMessageType(
|
||
|
value: unknown
|
||
|
): value is LocaleMessageType {
|
||
|
return (
|
||
|
typeof value === 'object' &&
|
||
|
value != null &&
|
||
|
Object.hasOwn(value, 'messageformat')
|
||
|
);
|
||
|
}
|
||
|
|
||
|
export type SetupI18nOptionsType = Readonly<{
|
||
|
renderEmojify: (parts: ReadonlyArray<unknown>) => JSX.Element | void;
|
||
|
}>;
|
||
|
|
||
|
export function createCachedIntl(
|
||
|
locale: string,
|
||
|
icuMessages: Record<string, string>,
|
||
|
{ renderEmojify }: SetupI18nOptionsType
|
||
|
): IntlShape {
|
||
|
const intlCache = createIntlCache();
|
||
|
const intl = createIntl(
|
||
|
{
|
||
|
locale: locale.replace('_', '-'), // normalize supported locales to browser format
|
||
|
messages: icuMessages,
|
||
|
defaultRichTextElements: {
|
||
|
emojify: renderEmojify,
|
||
|
},
|
||
|
onError(error) {
|
||
|
log.error('intl.onError', Errors.toLogFormat(error));
|
||
|
},
|
||
|
onWarn(warning) {
|
||
|
if (
|
||
|
getEnvironment() === Environment.Test &&
|
||
|
warning.includes(
|
||
|
// This warning is very noisy during tests
|
||
|
'"defaultRichTextElements" was specified but "message" was not pre-compiled.'
|
||
|
)
|
||
|
) {
|
||
|
return;
|
||
|
}
|
||
|
log.warn('intl.onWarn', warning);
|
||
|
},
|
||
|
},
|
||
|
intlCache
|
||
|
);
|
||
|
return intl;
|
||
|
}
|
||
|
|
||
|
function normalizeSubstitutions<
|
||
|
Substitutions extends Record<string, string | number | Date> | undefined
|
||
|
>(substitutions?: Substitutions): Substitutions | undefined {
|
||
|
if (!substitutions) {
|
||
|
return;
|
||
|
}
|
||
|
const normalized: Record<string, string | number | Date> = {};
|
||
|
const entries = Object.entries(substitutions);
|
||
|
if (entries.length === 0) {
|
||
|
return;
|
||
|
}
|
||
|
for (const [key, value] of entries) {
|
||
|
if (typeof value === 'string') {
|
||
|
normalized[key] = bidiIsolate(value);
|
||
|
} else {
|
||
|
normalized[key] = value;
|
||
|
}
|
||
|
}
|
||
|
return normalized as Substitutions;
|
||
|
}
|
||
|
|
||
|
function filterLegacyMessages(
|
||
|
messages: LocaleMessagesType
|
||
|
): Record<string, string> {
|
||
|
const icuMessages: Record<string, string> = {};
|
||
|
|
||
|
for (const [key, value] of Object.entries(messages)) {
|
||
|
if (isLocaleMessageType(value) && value.messageformat != null) {
|
||
|
icuMessages[key] = value.messageformat;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return icuMessages;
|
||
|
}
|
||
|
|
||
|
export function setupI18n(
|
||
|
locale: string,
|
||
|
messages: LocaleMessagesType,
|
||
|
{ renderEmojify }: SetupI18nOptionsType
|
||
|
): LocalizerType {
|
||
|
if (!locale) {
|
||
|
throw new Error('i18n: locale parameter is required');
|
||
|
}
|
||
|
if (!messages) {
|
||
|
throw new Error('i18n: messages parameter is required');
|
||
|
}
|
||
|
|
||
|
const intl = createCachedIntl(locale, filterLegacyMessages(messages), {
|
||
|
renderEmojify,
|
||
|
});
|
||
|
|
||
|
const localizer: LocalizerType = (<
|
||
|
Key extends keyof ICUStringMessageParamsByKeyType
|
||
|
>(
|
||
|
key: Key,
|
||
|
substitutions: ICUStringMessageParamsByKeyType[Key]
|
||
|
) => {
|
||
|
const result = intl.formatMessage(
|
||
|
{ id: key },
|
||
|
normalizeSubstitutions(substitutions)
|
||
|
);
|
||
|
|
||
|
strictAssert(result !== key, `i18n: missing translation for "${key}"`);
|
||
|
|
||
|
return result;
|
||
|
}) as LocalizerType;
|
||
|
|
||
|
localizer.getIntl = () => {
|
||
|
return intl;
|
||
|
};
|
||
|
localizer.getLocale = () => locale;
|
||
|
localizer.getLocaleMessages = () => messages;
|
||
|
localizer.getLocaleDirection = () => {
|
||
|
return window.SignalContext.getResolvedMessagesLocaleDirection();
|
||
|
};
|
||
|
localizer.getHourCyclePreference = () => {
|
||
|
return window.SignalContext.getHourCyclePreference();
|
||
|
};
|
||
|
|
||
|
return localizer;
|
||
|
}
|