feat: add custom tags for i18n

pull/3206/head
Ryan Miller 8 months ago
parent 79f3f35977
commit e2952322ba

@ -28,14 +28,14 @@
"notificationsSlowModeDescription": "{app_name} will occasionally check for new messages in the background.",
"attachmentsNotification": "{emoji} Attachment",
"notificationsAndroidSystem": "{message_count} new messages in {conversation_count} conversations",
"emojiReactsHoverTwoName": "{name} & {other_name} reacted with {emoji}",
"emojiReactsHoverTwoName": "{name} & {other_name} reacted with <emoji />",
"callsCalledYou": "{name} called you",
"disappearingMessagesLegacy": "{name} is using an outdated client. Disappearing messages may not work as expected.",
"attachmentsMedia": "{name} on {date_time}",
"emojiReactsHoverName": "{name} reacted with {emoji}",
"emojiReactsHoverName": "{name} reacted with <emoji />",
"notificationsIosGroup": "{name} to {conversation_name}",
"emojiReactsHoverTwoNameMultiple": "{name}, {other_name} and {count} others reacted with {emoji}",
"emojiReactsHoverTwoNameOne": "{name}, {other_name} and 1 other reacted with {emoji}",
"emojiReactsHoverTwoNameMultiple": "{name}, {other_name} and {count} others reacted with <emoji />",
"emojiReactsHoverTwoNameOne": "{name}, {other_name} and 1 other reacted with <emoji />",
"disappearingMessagesUpdated": "<b>{admin_name}</b> updated disappearing message settings.",
"groupMemberMoreNew": "<b>{name}</b> and <b>{count} others</b> joined the group.",
"groupMemberLeftMore": "<b>{name}</b> and <b>{count} others</b> left the group.",
@ -704,7 +704,7 @@
"membersInviteShareDescription": "Would you like to share group message history with <b>{name}</b>?",
"yes": "Yes",
"you": "You",
"emojiReactsHoverYouName": "You & {name} reacted with {emoji}",
"emojiReactsHoverYouName": "You & {name} reacted with <emoji />",
"communityJoinedAlready": "You are already a member of this community.",
"groupOnlyAdmin": "You are the only admin in &lt;b&gt;{group_name}&lt;/b&gt;.&lt;/br&gt;&lt;/br&gt;Group members and settings cannot be changed without an admin.",
"callsYouCalled": "You called {name}",
@ -726,11 +726,11 @@
"callsYouMissedCallPermissions": "You missed a call from <b>{name}</b> because you haven't enabled <b>Voice and Video Calls</b> in Privacy Settings.",
"lockAppEnablePasscode": "You must enable a passcode in your iOS Settings in order to use Screen Lock.",
"settingsRestartDescription": "You must restart {app_name} to apply your new settings.",
"emojiReactsHoverYou": "You reacted with {emoji}",
"emojiReactsHoverYou": "You reacted with <emoji />",
"groupRemovedYou": "You were removed from <b>{group_name}</b>.",
"messageRequestPendingDescription": "You will be able to send voice messages and attachments once the recipient has approved this message request.",
"linkPreviewsSendModalDescription": "You will not have full metadata protection when sending link previews.",
"emojiReactsHoverYouNameMultiple": "You, {name} & {count} others reacted with {emoji}",
"emojiReactsHoverYouNameMultiple": "You, {name} & {count} others reacted with <emoji />",
"emojiReactsHoverYouNameOne": "You, {name} & 1 other reacted with <emoji />",
"notificationsFastModeDescriptionIos": "You'll be notified of new messages reliably and immediately using Apple's notification Servers.",
"notificationsFastModeDescriptionAndroid": "You'll be notified of new messages reliably and immediately using Google's notification Servers.",

@ -32,6 +32,7 @@ window.sessionFeatureFlags = {
useTestNet: isTestNet(),
integrationTestEnv: isTestIntegration(),
useClosedGroupV3: false,
replaceLocalizedStringsWithKeys: false,
debug: {
debugLogging: !_.isEmpty(process.env.SESSION_DEBUG),
debugLibsessionDumps: !_.isEmpty(process.env.SESSION_DEBUG_LIBSESSION_DUMPS),

@ -20,6 +20,13 @@ const formattingTagRegex = new RegExp(
'g'
);
const supportedCustomTags = ['emoji'];
const customTagRegex = new RegExp(
`<(?:${supportedFormattingTags.join('|')})>.*?</(?:${supportedCustomTags.join('|')})>`,
'g'
);
const StyledHtmlRenderer = styled.span<{ darkMode: boolean }>`
span {
color: ${props => (props.darkMode ? 'var(--primary-color)' : 'var(--text-primary-color)')};
@ -50,8 +57,11 @@ export const I18n = <T extends LocalizerToken>(props: I18nProps<T>) => {
...([props.token, i18nArgs] as GetMessageArgs<T>)
);
const containsFormattingTag = i18nString.match(formattingTagRegex);
const containsCustomTag = i18nString.match(customTagRegex);
/** If the string contains a relevant formatting tag, render it as HTML */
if (i18nString.match(formattingTagRegex)) {
if (containsFormattingTag || containsCustomTag) {
return (
<StyledHtmlRenderer darkMode={darkMode}>
<SessionHtmlRenderer tag={props.as} html={i18nString} />

@ -15,7 +15,7 @@ export const customTag = {
),
};
export const SessionCustomTagRenderer = <Tag extends keyof typeof customTag>({
export const SessionCustomTag = <Tag extends keyof typeof customTag>({
tag,
props,
}: {
@ -25,4 +25,6 @@ export const SessionCustomTagRenderer = <Tag extends keyof typeof customTag>({
return customTag[tag](props);
};
SessionCustomTagRenderer({ tag: 'emoji', props: { emoji: '' } });
export const SessionCustomTagRenderer = ({ str }: { str: string }) => {
const splitString = str.split();
};

@ -92,32 +92,29 @@ const generateReactionString = (
) => {
const name = contacts[0];
const other_name = contacts[1];
const emoji = '';
switch (numberOfReactors) {
case 1:
return isYou
? window.i18n('emojiReactsHoverYou', { emoji })
: window.i18n('emojiReactsHoverName', { name, emoji });
? window.i18n('emojiReactsHoverYou')
: window.i18n('emojiReactsHoverName', { name });
case 2:
return isYou
? window.i18n('emojiReactsHoverYouName', { name, emoji })
: window.i18n('emojiReactsHoverTwoName', { name, other_name, emoji });
? window.i18n('emojiReactsHoverYouName', { name })
: window.i18n('emojiReactsHoverTwoName', { name, other_name });
case 3:
return isYou
? window.i18n('emojiReactsHoverYouNameOne', { name, emoji })
: window.i18n('emojiReactsHoverTwoNameOne', { name, other_name, emoji });
? window.i18n('emojiReactsHoverYouNameOne', { name })
: window.i18n('emojiReactsHoverTwoNameOne', { name, other_name });
default:
return isYou
? window.i18n('emojiReactsHoverYouNameMultiple', {
name,
emoji,
count: numberOfReactors - 2,
})
: window.i18n('emojiReactsHoverTwoNameMultiple', {
name,
other_name,
emoji,
count: numberOfReactors - 2,
});
}

@ -16,9 +16,10 @@ import {
PluralString,
} from '../types/Localizer';
import { LOCALE_DEFAULTS } from '../session/constants';
import { updateLocale } from '../state/ducks/dictionary';
export function loadDictionary(locale: Locale) {
return import(`../../_locales/${locale}/messages.json`) as Promise<Dictionary>;
return import(`../../_locales/${locale}/messages.json`) as Promise<LocalizerDictionary>;
}
const timeLocaleMap = {
@ -144,10 +145,17 @@ export const setupi18n = (locale: Locale, dictionary: LocalizerDictionary) => {
if (!locale) {
throw new Error('i18n: locale parameter is required');
}
if (!dictionary) {
throw new Error('i18n: messages parameter is required');
}
if (window.inboxStore) {
window.inboxStore.dispatch(updateLocale(locale));
window.log.info('Loaded dictionary dispatch');
}
window.log.info('i18n setup');
/**
* Retrieves a localized message string, substituting variables where necessary.
*
@ -165,12 +173,25 @@ export const setupi18n = (locale: Locale, dictionary: LocalizerDictionary) => {
...[token, args]: GetMessageArgs<T>
): R {
try {
const inboxStore = window.inboxStore;
const {
inboxStore,
sessionFeatureFlags: { replaceLocalizedStringsWithKeys },
} = window;
const localizedDictionary =
if (replaceLocalizedStringsWithKeys) {
return token as R;
}
const storedDictionary =
inboxStore && 'getState' in inboxStore && typeof inboxStore.getState === 'function'
? (inboxStore.getState().dictionary.dictionary as LocalizerDictionary)
: dictionary;
: undefined;
if (!storedDictionary) {
i18nLog(`i18n: Stored dictionary not found, using setup dictionary as fallback`);
}
const localizedDictionary = storedDictionary ?? dictionary;
let localizedString = localizedDictionary[token] as R;

1
ts/window.d.ts vendored

@ -51,6 +51,7 @@ declare global {
useTestNet: boolean;
useClosedGroupV3: boolean;
integrationTestEnv: boolean;
replaceLocalizedStringsWithKeys: boolean;
debug: {
debugLogging: boolean;
debugLibsessionDumps: boolean;

Loading…
Cancel
Save