2024-03-12 17:32:11 +00:00
|
|
|
// 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,
|
2024-05-22 16:24:27 +00:00
|
|
|
LocalizerOptions,
|
2024-03-12 17:32:11 +00:00
|
|
|
} from '../types/Util';
|
|
|
|
import { strictAssert } from './assert';
|
|
|
|
import * as log from '../logging/log';
|
|
|
|
import * as Errors from '../types/errors';
|
|
|
|
import { Environment, getEnvironment } from '../environment';
|
2024-07-15 23:15:18 +00:00
|
|
|
import { bidiIsolate, bidiStrip } from './unicodeBidi';
|
2024-03-12 17:32:11 +00:00
|
|
|
|
|
|
|
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
|
2024-07-15 23:15:18 +00:00
|
|
|
>(
|
|
|
|
substitutions?: Substitutions,
|
|
|
|
options?: LocalizerOptions
|
|
|
|
): Substitutions | undefined {
|
2024-03-12 17:32:11 +00:00
|
|
|
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') {
|
2024-07-15 23:15:18 +00:00
|
|
|
if (options?.bidi === 'strip') {
|
|
|
|
normalized[key] = bidiStrip(value);
|
|
|
|
} else {
|
|
|
|
normalized[key] = bidiIsolate(value);
|
|
|
|
}
|
2024-03-12 17:32:11 +00:00
|
|
|
} 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,
|
2024-05-22 16:24:27 +00:00
|
|
|
substitutions: ICUStringMessageParamsByKeyType[Key],
|
|
|
|
options?: LocalizerOptions
|
2024-03-12 17:32:11 +00:00
|
|
|
) => {
|
|
|
|
const result = intl.formatMessage(
|
|
|
|
{ id: key },
|
2024-07-15 23:15:18 +00:00
|
|
|
normalizeSubstitutions(substitutions, options)
|
2024-03-12 17:32:11 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|