2023-06-14 23:26:05 +00:00
|
|
|
// Copyright 2018 Signal Messenger, LLC
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
|
|
|
import React from 'react';
|
|
|
|
import type { IntlShape } from 'react-intl';
|
|
|
|
import { createIntl, createIntlCache } from 'react-intl';
|
|
|
|
import type { LocaleMessageType, LocaleMessagesType } from '../types/I18N';
|
2024-02-29 00:42:43 +00:00
|
|
|
import type { LocalizerType, ReplacementValuesType } from '../types/Util';
|
2023-06-14 23:26:05 +00:00
|
|
|
import { strictAssert } from './assert';
|
|
|
|
import { Emojify } from '../components/conversation/Emojify';
|
2023-08-01 16:41:28 +00:00
|
|
|
import * as log from '../logging/log';
|
|
|
|
import * as Errors from '../types/errors';
|
|
|
|
import { Environment, getEnvironment } from '../environment';
|
2024-02-29 00:42:43 +00:00
|
|
|
import { bidiIsolate } from './unicodeBidi';
|
2023-06-14 23:26:05 +00:00
|
|
|
|
|
|
|
export function isLocaleMessageType(
|
|
|
|
value: unknown
|
|
|
|
): value is LocaleMessageType {
|
|
|
|
return (
|
|
|
|
typeof value === 'object' &&
|
|
|
|
value != null &&
|
|
|
|
Object.hasOwn(value, 'messageformat')
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2023-06-15 00:57:04 +00:00
|
|
|
export function renderEmojify(parts: ReadonlyArray<unknown>): JSX.Element {
|
|
|
|
strictAssert(parts.length === 1, '<emojify> must contain only one child');
|
2023-06-14 23:26:05 +00:00
|
|
|
const text = parts[0];
|
2023-06-15 00:57:04 +00:00
|
|
|
strictAssert(typeof text === 'string', '<emojify> must contain only text');
|
2023-06-14 23:26:05 +00:00
|
|
|
return <Emojify text={text} />;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function createCachedIntl(
|
|
|
|
locale: string,
|
|
|
|
icuMessages: Record<string, string>
|
|
|
|
): IntlShape {
|
|
|
|
const intlCache = createIntlCache();
|
|
|
|
const intl = createIntl(
|
|
|
|
{
|
|
|
|
locale: locale.replace('_', '-'), // normalize supported locales to browser format
|
|
|
|
messages: icuMessages,
|
|
|
|
defaultRichTextElements: {
|
2023-06-15 00:57:04 +00:00
|
|
|
emojify: renderEmojify,
|
2023-06-14 23:26:05 +00:00
|
|
|
},
|
2023-08-01 16:41:28 +00:00
|
|
|
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);
|
|
|
|
},
|
2023-06-14 23:26:05 +00:00
|
|
|
},
|
|
|
|
intlCache
|
|
|
|
);
|
|
|
|
return intl;
|
|
|
|
}
|
|
|
|
|
2024-02-29 00:42:43 +00:00
|
|
|
function normalizeSubstitutions(
|
|
|
|
substitutions?: ReplacementValuesType
|
|
|
|
): ReplacementValuesType | undefined {
|
|
|
|
if (!substitutions) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const normalized: ReplacementValuesType = {};
|
2024-02-29 22:01:12 +00:00
|
|
|
const keys = Object.keys(substitutions);
|
|
|
|
if (keys.length === 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
for (let i = 0; i < keys.length; i += 1) {
|
|
|
|
const key = keys[i];
|
|
|
|
const value = substitutions[key];
|
2024-02-29 00:42:43 +00:00
|
|
|
if (typeof value === 'string') {
|
|
|
|
normalized[key] = bidiIsolate(value);
|
|
|
|
} else {
|
|
|
|
normalized[key] = value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return normalized;
|
|
|
|
}
|
|
|
|
|
2023-06-14 23:26:05 +00:00
|
|
|
export function setupI18n(
|
|
|
|
locale: string,
|
|
|
|
messages: LocaleMessagesType
|
|
|
|
): 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));
|
|
|
|
|
|
|
|
const localizer: LocalizerType = (key, substitutions) => {
|
2024-02-29 00:42:43 +00:00
|
|
|
const result = intl.formatMessage(
|
|
|
|
{ id: key },
|
|
|
|
normalizeSubstitutions(substitutions)
|
|
|
|
);
|
2023-06-14 23:26:05 +00:00
|
|
|
|
|
|
|
strictAssert(result !== key, `i18n: missing translation for "${key}"`);
|
|
|
|
|
|
|
|
return result;
|
|
|
|
};
|
|
|
|
|
|
|
|
localizer.getIntl = () => {
|
|
|
|
return intl;
|
|
|
|
};
|
|
|
|
localizer.getLocale = () => locale;
|
|
|
|
localizer.getLocaleMessages = () => messages;
|
|
|
|
localizer.getLocaleDirection = () => {
|
2023-08-07 20:28:09 +00:00
|
|
|
return window.SignalContext.getResolvedMessagesLocaleDirection();
|
2023-06-14 23:26:05 +00:00
|
|
|
};
|
2023-07-31 16:23:19 +00:00
|
|
|
localizer.getHourCyclePreference = () => {
|
2023-08-07 20:28:09 +00:00
|
|
|
return window.SignalContext.getHourCyclePreference();
|
2023-07-31 16:23:19 +00:00
|
|
|
};
|
2023-06-14 23:26:05 +00:00
|
|
|
|
|
|
|
return localizer;
|
|
|
|
}
|