fix: break down i18n file and remove translations from redux

pull/3206/head
Audric Ackermann 7 months ago
parent 5a7da00d00
commit 1789ac30c8

@ -4,7 +4,7 @@
const { ipcRenderer } = require('electron');
const url = require('url');
const os = require('os');
const { setupI18n } = require('./ts/util/i18n');
const { setupI18n } = require('./ts/util/i18n/i18n');
const config = url.parse(window.location.toString(), true).query;
const { locale } = config;
@ -12,8 +12,8 @@ const localeMessages = ipcRenderer.sendSync('locale-data');
window.theme = config.theme;
window.i18n = setupI18n({
initialLocale: locale,
initialDictionary: localeMessages,
locale,
translationDictionary: localeMessages,
});
window.getOSRelease = () =>

@ -6,7 +6,7 @@ const url = require('url');
const os = require('os');
const { setupI18n } = require('./ts/util/i18n');
const { setupI18n } = require('./ts/util/i18n/i18n');
const config = url.parse(window.location.toString(), true).query;
const { locale } = config;
@ -16,7 +16,7 @@ window._ = require('lodash');
window.getVersion = () => config.version;
window.theme = config.theme;
window.i18n = setupI18n({ initialLocale: locale, initialDictionary: localeMessages });
window.i18n = setupI18n({ locale, translationDictionary: localeMessages });
// got.js appears to need this to successfully submit debug logs to the cloud
window.nodeSetImmediate = setImmediate;

@ -3,7 +3,7 @@
const { ipcRenderer } = require('electron');
const url = require('url');
const { setupI18n } = require('./ts/util/i18n');
const { setupI18n } = require('./ts/util/i18n/i18n');
const config = url.parse(window.location.toString(), true).query;
const { locale } = config;
@ -12,7 +12,7 @@ const localeMessages = ipcRenderer.sendSync('locale-data');
// If the app is locked we can't access the database to check the theme.
window.theme = 'classic-dark';
window.primaryColor = 'green';
window.i18n = setupI18n({ initialLocale: locale, initialDictionary: localeMessages });
window.i18n = setupI18n({ locale, translationDictionary: localeMessages });
window.getEnvironment = () => config.environment;
window.getVersion = () => config.version;

@ -232,7 +232,7 @@ if (config.proxyUrl) {
window.nodeSetImmediate = setImmediate;
const data = require('./ts/data/dataInit');
const { setupI18n } = require('./ts/util/i18n');
const { setupI18n } = require('./ts/util/i18n/i18n');
window.Signal = data.initData();
const { getConversationController } = require('./ts/session/conversations/ConversationController');
@ -254,8 +254,8 @@ window.getSeedNodeList = () =>
'https://seed3.getsession.org:4443/',
];
const { locale: localFromEnv } = config;
window.i18n = setupI18n({ initialLocale: localFromEnv, initialDictionary: localeMessages });
const { locale } = config;
window.i18n = setupI18n({ locale, translationDictionary: localeMessages });
window.addEventListener('contextmenu', e => {
const editable = e && e.target.closest('textarea, input, [contenteditable="true"]');

@ -31,7 +31,6 @@ import { StateType } from '../state/reducer';
import { SessionMainPanel } from './SessionMainPanel';
import { SettingsKey } from '../data/settings-key';
import { initialDictionaryState } from '../state/ducks/dictionary';
import { getSettingsInitialState, updateAllOnStorageReady } from '../state/ducks/settings';
import { initialSogsRoomInfoState } from '../state/ducks/sogsRoomInfo';
import { useHasDeviceOutdatedSyncing } from '../state/selectors/settings';
@ -47,7 +46,6 @@ function makeLookup<T>(items: Array<T>, key: string): { [key: string]: T } {
return fromPairs(pairs);
}
const StyledGutter = styled.div`
width: var(--left-panel-width) !important;
transition: none;
@ -80,7 +78,6 @@ function createSessionInboxStore() {
call: initialCallState,
sogsRoomInfo: initialSogsRoomInfoState,
settings: getSettingsInitialState(),
dictionary: initialDictionaryState,
};
return createStore(initialState);

@ -3,8 +3,8 @@ import { useAppIsFocused } from '../hooks/useAppFocused';
import { getFocusedSettingsSection } from '../state/selectors/section';
import { SmartSessionConversation } from '../state/smart/SessionConversation';
import { useHTMLDirection } from '../util/i18n';
import { SessionSettingsView } from './settings/SessionSettings';
import { useHTMLDirection } from '../util/i18n/rtlSupport';
const FilteredSettingsView = SessionSettingsView as any;

@ -1,6 +1,6 @@
import { HTMLMotionProps, motion } from 'framer-motion';
import styled from 'styled-components';
import { HTMLDirection } from '../../util/i18n';
import { HTMLDirection } from '../../util/i18n/rtlSupport';
export interface FlexProps {
children?: any;

@ -51,10 +51,10 @@ import { NoMessageInConversation } from './SubtleNotification';
import { ConversationHeaderWithDetails } from './header/ConversationHeader';
import { isAudio } from '../../types/MIME';
import { HTMLDirection } from '../../util/i18n';
import { NoticeBanner } from '../NoticeBanner';
import { SessionSpinner } from '../loading';
import { RightPanel, StyledRightPanelContainer } from './right-panel/RightPanel';
import { HTMLDirection } from '../../util/i18n/rtlSupport';
const DEFAULT_JPEG_QUALITY = 0.85;

@ -2,7 +2,8 @@ import useInterval from 'react-use/lib/useInterval';
import useUpdate from 'react-use/lib/useUpdate';
import styled from 'styled-components';
import { CONVERSATION } from '../../session/constants';
import { formatFullDate, getConversationItemString } from '../../util/i18n';
import { getConversationItemString } from '../../util/i18n/formater/conversationItemTimestamp';
import { formatFullDate } from '../../util/i18n/formater/generics';
type Props = {
timestamp?: number;

@ -37,7 +37,6 @@ import {
StagedAttachmentImportedType,
StagedPreviewImportedType,
} from '../../../util/attachmentsUtil';
import { HTMLDirection } from '../../../util/i18n';
import { LinkPreviews } from '../../../util/linkPreviews';
import { CaptionEditor } from '../../CaptionEditor';
import { Flex } from '../../basic/Flex';
@ -58,6 +57,7 @@ import {
} from './CompositionButtons';
import { CompositionTextArea } from './CompositionTextArea';
import { cleanMentions, mentionsRegex } from './UserMentions';
import { HTMLDirection } from '../../../util/i18n/rtlSupport';
export interface ReplyingToMessageProps {
convoId: string;

@ -8,10 +8,10 @@ import {
useSelectedIsLeft,
useSelectedNicknameOrProfileNameOrShortenedPubkey,
} from '../../../state/selectors/selectedConversation';
import { HTMLDirection, useHTMLDirection } from '../../../util/i18n';
import { updateDraftForConversation } from '../SessionConversationDrafts';
import { renderEmojiQuickResultRow, searchEmojiForQuery } from './EmojiQuickResult';
import { renderUserMentionRow, styleForCompositionBoxSuggestions } from './UserMentions';
import { HTMLDirection, useHTMLDirection } from '../../../util/i18n/rtlSupport';
const sendMessageStyle = (dir?: HTMLDirection) => {
return {

@ -1,6 +1,6 @@
import { SuggestionDataItem } from 'react-mentions';
import { HTMLDirection } from '../../../util/i18n';
import { MemberListItem } from '../../MemberListItem';
import { HTMLDirection } from '../../../util/i18n/rtlSupport';
const listRTLStyle = { position: 'absolute', bottom: '0px', right: '100%' };

@ -5,7 +5,7 @@ import formatFileSize from 'filesize';
import { useSelectedConversationKey } from '../../../state/selectors/selectedConversation';
import { saveAttachmentToDisk } from '../../../util/attachmentsUtil';
import { MediaItemType } from '../../lightbox/LightboxGallery';
import { formatWithLocale } from '../../../util/i18n';
import { formatWithLocale } from '../../../util/i18n/formater/generics';
type Props = {
// Required

@ -21,13 +21,13 @@ import {
} from '../../../../state/selectors/conversations';
import { useSelectedIsPrivate } from '../../../../state/selectors/selectedConversation';
import { canDisplayImagePreview } from '../../../../types/Attachment';
import { formatFullDate } from '../../../../util/i18n';
import { MessageAttachment } from './MessageAttachment';
import { MessageAvatar } from './MessageAvatar';
import { MessageHighlighter } from './MessageHighlighter';
import { MessageLinkPreview } from './MessageLinkPreview';
import { MessageQuote } from './MessageQuote';
import { MessageText } from './MessageText';
import { formatFullDate } from '../../../../util/i18n/formater/generics';
export type MessageContentSelectorProps = Pick<
MessageRenderingProps,

@ -6,10 +6,10 @@ import useInterval from 'react-use/lib/useInterval';
import { useMessageExpirationPropsById } from '../../../../hooks/useParamSelector';
import { DURATION } from '../../../../session/constants';
import { nativeEmojiData } from '../../../../util/emoji';
import { formatAbbreviatedExpireDoubleTimer } from '../../../../util/i18n';
import { getRecentReactions } from '../../../../util/storage';
import { SpacerSM } from '../../../basic/Text';
import { SessionIcon, SessionIconButton } from '../../../icon';
import { formatAbbreviatedExpireDoubleTimer } from '../../../../util/i18n/formater/expirationTimer';
type Props = {
action: (...args: Array<any>) => void;

@ -1,6 +1,6 @@
import styled from 'styled-components';
import { DURATION } from '../../../../session/constants';
import { formatFullDate, formatRelativeWithLocale } from '../../../../util/i18n';
import { formatFullDate, formatRelativeWithLocale } from '../../../../util/i18n/formater/generics';
const DateBreakContainer = styled.div``;

@ -1,10 +1,10 @@
import styled from 'styled-components';
import { useRightOverlayMode } from '../../../hooks/useUI';
import { isRtlBody } from '../../../util/i18n';
import { Flex } from '../../basic/Flex';
import { OverlayRightPanelSettings } from './overlay/OverlayRightPanelSettings';
import { OverlayDisappearingMessages } from './overlay/disappearing-messages/OverlayDisappearingMessages';
import { OverlayMessageInfo } from './overlay/message-info/OverlayMessageInfo';
import { isRtlBody } from '../../../util/i18n/rtlSupport';
export const StyledRightPanelContainer = styled.div`
position: absolute;

@ -19,14 +19,15 @@ import {
import { isDevProd } from '../../../../../../shared/env_vars';
import { useSelectedConversationKey } from '../../../../../../state/selectors/selectedConversation';
import {
formatTimeDuration,
formatTimeDistanceToNow,
formatWithLocale,
} from '../../../../../../util/i18n';
import { Flex } from '../../../../../basic/Flex';
import { SpacerSM } from '../../../../../basic/Text';
import { CopyToClipboardIcon } from '../../../../../buttons';
import {
formatTimeDistanceToNow,
formatTimeDuration,
formatWithLocale,
} from '../../../../../../util/i18n/formater/generics';
export const MessageInfoLabel = styled.label<{ color?: string }>`
font-size: var(--font-size-lg);

@ -22,7 +22,7 @@ import { THEME_GLOBALS } from '../../themes/globals';
import { SessionWrapperModal } from '../SessionWrapperModal';
import { SessionIcon, SessionIconButton } from '../icon';
import { SessionSpinner } from '../loading';
import { getLocale } from '../../util/i18n';
import { getLocale } from '../../util/i18n/shared';
export type StatusLightType = {
glowStartDelay: number;

@ -4,10 +4,10 @@ import { motion } from 'framer-motion';
import { isEmpty, isEqual } from 'lodash';
import styled, { CSSProperties } from 'styled-components';
import { THEME_GLOBALS } from '../../themes/globals';
import { useHTMLDirection } from '../../util/i18n';
import { AnimatedFlex, Flex } from '../basic/Flex';
import { SpacerMD } from '../basic/Text';
import { SessionIconButton } from '../icon';
import { useHTMLDirection } from '../../util/i18n/rtlSupport';
type TextSizes = 'xs' | 'sm' | 'md' | 'lg' | 'xl';

@ -6,13 +6,13 @@ import { parseOpenGroupV2 } from '../../../session/apis/open_group_api/opengroup
import { sogsV3FetchPreviewBase64 } from '../../../session/apis/open_group_api/sogsv3/sogsV3FetchFile';
import { DefaultRoomsState, updateDefaultBase64RoomData } from '../../../state/ducks/defaultRooms';
import { StateType } from '../../../state/reducer';
import { useHTMLDirection } from '../../../util/i18n';
import { Avatar, AvatarSize } from '../../avatar/Avatar';
import { Flex } from '../../basic/Flex';
import { H4 } from '../../basic/Heading';
import { PillContainerHoverable, StyledPillContainerHoverable } from '../../basic/PillContainer';
import { SpacerXS } from '../../basic/Text';
import { SessionSpinner } from '../../loading';
import { useHTMLDirection } from '../../../util/i18n/rtlSupport';
export type JoinableRoomProps = {
completeUrl: string;

@ -30,7 +30,7 @@ import { Notifications } from '../util/notifications';
import { Registration } from '../util/registration';
import { Storage, isSignInByLinking } from '../util/storage';
import { getOppositeTheme, isThemeMismatched } from '../util/theme';
import { getLocale } from '../util/i18n';
import { getLocale } from '../util/i18n/shared';
// Globally disable drag and drop
document.body.addEventListener(

@ -1,5 +1,6 @@
import { isCI, isDevProd } from '../../shared/env_vars';
import { formatAbbreviatedExpireTimer, formatTimeDuration } from '../../util/i18n';
import { formatAbbreviatedExpireTimer } from '../../util/i18n/formater/expirationTimer';
import { formatTimeDuration } from '../../util/i18n/formater/generics';
import { DURATION_SECONDS } from '../constants';
type TimerOptionsEntry = { name: string; value: number };

@ -1,36 +0,0 @@
import { createSlice, PayloadAction } from '@reduxjs/toolkit';
import { Dictionary, en } from '../../localization/locales';
import { loadDictionary, Locale } from '../../util/i18n';
export type DictionaryState = {
dictionary: Dictionary;
locale: string;
};
export const initialDictionaryState = {
dictionary: en,
locale: 'en',
};
const dictionarySlice = createSlice({
name: 'dictionary',
initialState: initialDictionaryState,
reducers: {
updateLocale(state: DictionaryState, action: PayloadAction<Locale>) {
// eslint-disable-next-line more/no-then
loadDictionary(action.payload)
.then(dictionary => {
state.dictionary = dictionary;
state.locale = action.payload;
})
.catch(e => {
window.log.error('Failed to load dictionary', e);
});
},
},
});
// destructures
const { actions, reducer } = dictionarySlice;
export const { updateLocale } = actions;
export const defaultDictionaryReducer = reducer;

@ -11,7 +11,6 @@ import { reducer as theme } from './ducks/theme';
import { reducer as user, UserStateType } from './ducks/user';
import { PrimaryColorStateType, ThemeStateType } from '../themes/constants/colors';
import { defaultDictionaryReducer, DictionaryState } from './ducks/dictionary';
import { modalReducer as modals, ModalState } from './ducks/modalDialog';
import { defaultOnionReducer as onionPaths, OnionState } from './ducks/onion';
import { settingsReducer, SettingsState } from './ducks/settings';
@ -36,7 +35,6 @@ export type StateType = {
call: CallStateType;
sogsRoomInfo: SogsRoomInfoState;
settings: SettingsState;
dictionary: DictionaryState;
};
const reducers = {
@ -54,7 +52,6 @@ const reducers = {
call,
sogsRoomInfo: ReduxSogsRoomInfos.sogsRoomInfoReducer,
settings: settingsReducer,
dictionary: defaultDictionaryReducer,
};
// Making this work would require that our reducer signature supported AnyAction, not

@ -1,6 +1,5 @@
import { connect } from 'react-redux';
import { SessionConversation } from '../../components/conversation/SessionConversation';
import { HTMLDirection } from '../../util/i18n';
import { mapDispatchToProps } from '../actions';
import { StateType } from '../reducer';
import { getHasOngoingCallWithFocusedConvo } from '../selectors/call';
@ -15,6 +14,7 @@ import { getSelectedConversationKey } from '../selectors/selectedConversation';
import { getStagedAttachmentsForCurrentConversation } from '../selectors/stagedAttachments';
import { getTheme } from '../selectors/theme';
import { getOurDisplayNameInProfile, getOurNumber } from '../selectors/user';
import { HTMLDirection } from '../../util/i18n/rtlSupport';
type SmartSessionConversationOwnProps = {
htmlDirection: HTMLDirection;

@ -10,6 +10,7 @@ import {
} from '../../../../util/accountManager';
import { TestUtils } from '../../../test-utils';
import { stubWindow } from '../../../test-utils/utils';
import { resetTranslationDictionary } from '../../../../util/i18n/translationDictionaries';
describe('Onboarding', () => {
const polledDisplayName = 'Hello World';
@ -25,6 +26,7 @@ describe('Onboarding', () => {
});
afterEach(() => {
resetTranslationDictionary();
Sinon.restore();
});

@ -8,6 +8,7 @@ import {
_getSortedConversations,
} from '../../../../state/selectors/conversations';
import { TestUtils } from '../../../test-utils';
import { resetTranslationDictionary } from '../../../../util/i18n/translationDictionaries';
describe('state/selectors/conversations', () => {
beforeEach(() => {
@ -15,6 +16,7 @@ describe('state/selectors/conversations', () => {
TestUtils.stubI18n();
});
afterEach(() => {
resetTranslationDictionary();
Sinon.restore();
});
describe('#getSortedConversationsList', () => {

@ -1,5 +1,5 @@
import { expect } from 'chai';
import { formatAbbreviatedExpireDoubleTimer } from '../../../../../util/i18n';
import { formatAbbreviatedExpireDoubleTimer } from '../../../../../util/i18n/formater/expirationTimer';
describe('formatAbbreviatedExpireDoubleTimer', () => {
it('<= 0 returns 0s', () => {

@ -3,9 +3,16 @@
import { expect } from 'chai';
import { initI18n, testDictionary } from './util';
import { resetTranslationDictionary } from '../../../../../util/i18n/translationDictionaries';
describe('formatMessageWithArgs', () => {
const i18n = initI18n(testDictionary);
let i18n;
beforeEach(() => {
i18n = initI18n(testDictionary);
});
afterEach(() => {
resetTranslationDictionary();
});
it('returns the message with args for a message', () => {
const message = i18n('Hello, {name}!', { name: 'Alice' });

@ -3,9 +3,17 @@
import { expect } from 'chai';
import { initI18n, testDictionary } from './util';
import { resetTranslationDictionary } from '../../../../../util/i18n/translationDictionaries';
describe('getMessage', () => {
const i18n = initI18n(testDictionary);
let i18n;
beforeEach(() => {
i18n = initI18n(testDictionary);
});
afterEach(() => {
resetTranslationDictionary();
});
it('returns the message for a token', () => {
const message = i18n('greeting', { name: 'Alice' });

@ -3,9 +3,17 @@
import { expect } from 'chai';
import { initI18n, testDictionary } from './util';
import { resetTranslationDictionary } from '../../../../../util/i18n/translationDictionaries';
describe('getRawMessage', () => {
const i18n = initI18n(testDictionary);
let i18n;
beforeEach(() => {
i18n = initI18n(testDictionary);
});
afterEach(() => {
resetTranslationDictionary();
});
it('returns the raw message for a token', () => {
const rawMessage = i18n.getRawMessage('greeting', { name: 'Alice' });

@ -1,7 +1,11 @@
import { expect } from 'chai';
import { initI18n } from './util';
import { resetTranslationDictionary } from '../../../../../util/i18n/translationDictionaries';
describe('setupI18n', () => {
afterEach(() => {
resetTranslationDictionary();
});
it('returns setupI18n with all methods defined', () => {
const setupI18nReturn = initI18n();
expect(setupI18nReturn).to.be.a('function');

@ -3,9 +3,16 @@
import { expect } from 'chai';
import { initI18n, testDictionary } from './util';
import { resetTranslationDictionary } from '../../../../../util/i18n/translationDictionaries';
describe('stripped', () => {
const i18n = initI18n(testDictionary);
let i18n;
beforeEach(() => {
i18n = initI18n(testDictionary);
});
afterEach(() => {
resetTranslationDictionary();
});
it('returns the stripped message for a token', () => {
const message = i18n.stripped('greeting', { name: 'Alice' });

@ -1,6 +1,6 @@
import { setupI18n } from '../../../../../util/i18n';
import { en } from '../../../../../localization/locales';
import type { LocalizerDictionary } from '../../../../../types/Localizer';
import { setupI18n } from '../../../../../util/i18n/i18n';
export const testDictionary = {
greeting: 'Hello, {name}!',
@ -12,5 +12,5 @@ export const testDictionary = {
} as const;
export function initI18n(dictionary: Record<string, string> = en) {
return setupI18n({ initialLocale: 'en', initialDictionary: dictionary as LocalizerDictionary });
return setupI18n({ locale: 'en', translationDictionary: dictionary as LocalizerDictionary });
}

@ -6,9 +6,9 @@ import { Data } from '../../../data/data';
import { OpenGroupData } from '../../../data/opengroups';
import { load } from '../../../node/locale';
import { setupI18n } from '../../../util/i18n';
import * as libsessionWorker from '../../../webworker/workers/browser/libsession_worker_interface';
import * as utilWorker from '../../../webworker/workers/browser/util_worker_interface';
import { setupI18n } from '../../../util/i18n/i18n';
const globalAny: any = global;
@ -141,5 +141,5 @@ export async function expectAsyncToThrow(toAwait: () => Promise<any>, errorMessa
/** You must call stubWindowLog() before using */
export const stubI18n = () => {
const locale = load({ appLocale: 'en', logger: window.log });
stubWindow('i18n', setupI18n({ initialLocale: 'en', initialDictionary: locale.messages }));
stubWindow('i18n', setupI18n({ locale: 'en', translationDictionary: locale.messages }));
};

@ -12,7 +12,7 @@ import { ToastUtils } from '../../session/utils';
import { GoogleChrome } from '../../util';
import { autoScaleForAvatar, autoScaleForThumbnail } from '../../util/attachmentsUtil';
import { isAudio } from '../MIME';
import { formatTimeDuration } from '../../util/i18n';
import { formatTimeDuration } from '../../util/i18n/formater/generics';
export const THUMBNAIL_SIDE = 200;
export const THUMBNAIL_CONTENT_TYPE = 'image/png';

@ -4,7 +4,7 @@
/* eslint-disable import/no-mutable-exports */
import { init, I18n } from 'emoji-mart';
import { FixedBaseEmoji, NativeEmojiData } from '../types/Reaction';
import { loadEmojiPanelI18n } from './i18n';
import { loadEmojiPanelI18n } from './i18n/emojiPanelI18n';
export type SizeClassType = 'default' | 'small' | 'medium' | 'large' | 'jumbo';

@ -1,728 +0,0 @@
// this file is a weird one as it is used by both sides of electron at the same time
import {
Duration,
FormatDistanceStrictOptions,
FormatDistanceToNowStrictOptions,
format,
formatDistanceStrict,
formatDistanceToNow,
formatDistanceToNowStrict,
formatDuration,
formatRelative,
intervalToDuration,
isAfter,
isBefore,
subDays,
subMilliseconds,
} from 'date-fns';
import timeLocales from 'date-fns/locale';
import { deSanitizeHtmlTags, sanitizeArgs } from '../components/basic/I18n';
import { LOCALE_DEFAULTS } from '../localization/constants';
import { en } from '../localization/locales';
import { GetNetworkTime } from '../session/apis/snode_api/getNetworkTime';
import { DURATION_SECONDS } from '../session/constants';
import { updateLocale } from '../state/ducks/dictionary';
import {
ArgsRecord,
DictionaryWithoutPluralStrings,
GetMessageArgs,
LocalizerDictionary,
LocalizerToken,
PluralKey,
PluralString,
SetupI18nReturnType,
} from '../types/Localizer';
export function loadDictionary(locale: Locale) {
return import(`../../_locales/${locale}/messages.json`) as Promise<LocalizerDictionary>;
}
const timeLocaleMap = {
ar: timeLocales.ar,
be: timeLocales.be,
bg: timeLocales.bg,
ca: timeLocales.ca,
cs: timeLocales.cs,
da: timeLocales.da,
de: timeLocales.de,
el: timeLocales.el,
en: timeLocales.enUS,
eo: timeLocales.eo,
es: timeLocales.es,
/** TODO - Check this */
es_419: timeLocales.es,
et: timeLocales.et,
fa: timeLocales.faIR,
fi: timeLocales.fi,
/** TODO - Check this */
fil: timeLocales.fi,
fr: timeLocales.fr,
he: timeLocales.he,
hi: timeLocales.hi,
hr: timeLocales.hr,
hu: timeLocales.hu,
/** TODO - Check this */
'hy-AM': timeLocales.hy,
id: timeLocales.id,
it: timeLocales.it,
ja: timeLocales.ja,
ka: timeLocales.ka,
km: timeLocales.km,
/** TODO - Check this */
kmr: timeLocales.km,
kn: timeLocales.kn,
ko: timeLocales.ko,
lt: timeLocales.lt,
lv: timeLocales.lv,
mk: timeLocales.mk,
nb: timeLocales.nb,
nl: timeLocales.nl,
/** TODO - Find this this */
no: timeLocales.enUS,
/** TODO - Find this this */
pa: timeLocales.enUS,
pl: timeLocales.pl,
pt_BR: timeLocales.ptBR,
pt_PT: timeLocales.pt,
ro: timeLocales.ro,
ru: timeLocales.ru,
/** TODO - Find this this */
si: timeLocales.enUS,
sk: timeLocales.sk,
sl: timeLocales.sl,
sq: timeLocales.sq,
sr: timeLocales.sr,
sv: timeLocales.sv,
ta: timeLocales.ta,
th: timeLocales.th,
/** TODO - Find this this */
tl: timeLocales.enUS,
tr: timeLocales.tr,
uk: timeLocales.uk,
uz: timeLocales.uz,
vi: timeLocales.vi,
zh_CN: timeLocales.zhCN,
zh_TW: timeLocales.zhTW,
};
export type Locale = keyof typeof timeLocaleMap;
let initialLocale: Locale = 'en';
function getPluralKey<R extends PluralKey | undefined>(string: PluralString): R {
const match = /{(\w+), plural, one \[.+\] other \[.+\]}/g.exec(string);
return (match?.[1] ?? undefined) 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<Intl.LDMLPluralRule, RegExp> = {
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`.
*
* TODO: Change this to a proper type assertion when the type is fixed.
*/
const isStringWithArgs = <R extends DictionaryWithoutPluralStrings[LocalizerToken]>(
localizedString: string
): localizedString is R => localizedString.includes('{');
/**
* Logs an i18n message to the console.
* @param message - The message to log.
*
* TODO - Replace this logging method when the new logger is created
*/
function i18nLog(message: string) {
// eslint:disable: no-console
// eslint-disable-next-line no-console
(window?.log?.error ?? console.log)(`i18n: ${message}`);
}
/**
* Returns the current locale.
* @param params - An object containing optional parameters.
* @param params.fallback - The fallback locale to use if redux is not available. Defaults to en.
*/
export function getLocale(): Locale {
const locale = window?.inboxStore?.getState().dictionary.locale;
if (locale) {
return locale;
}
if (initialLocale) {
i18nLog(
`getLocale: No locale found in redux store but initialLocale provided: ${initialLocale}`
);
return initialLocale;
}
i18nLog('getLocale: No locale found in redux store. No fallback provided. Using en.');
return 'en';
}
function getLocaleDictionary() {
return timeLocaleMap[getLocale()];
}
/**
* Returns the current dictionary.
* @param params - An object containing optional parameters.
* @param params.fallback - The fallback dictionary to use if redux is not available. Defaults to {@link en}.
*/
function getDictionary(params?: { fallback?: LocalizerDictionary }): LocalizerDictionary {
const dict = window?.inboxStore?.getState().dictionary.dictionary;
if (dict) {
return dict;
}
if (params?.fallback) {
i18nLog('getDictionary: No dictionary found in redux store. Using fallback.');
return params.fallback;
}
i18nLog('getDictionary: No dictionary found in redux store. No fallback provided. Using en.');
return en;
}
/**
* Sets up the i18n function with the provided locale and messages.
*
* @param params - An object containing optional parameters.
* @param params.initialLocale - The locale to use for translations. Defaults to 'en'.
* @param params.initialDictionary - A dictionary of localized messages. Defaults to {@link en}.
*
* @returns A function that retrieves a localized message string, substituting variables where necessary.
*/
export const setupI18n = (params: {
initialLocale: Locale;
initialDictionary: LocalizerDictionary;
}): SetupI18nReturnType => {
initialLocale = params.initialLocale;
let initialDictionary = params.initialDictionary;
if (!initialLocale) {
initialLocale = 'en';
i18nLog(`initialLocale not provided in i18n setup. Falling back to ${initialLocale}`);
}
if (!initialLocale) {
initialDictionary = en;
i18nLog('initialDictionary not provided in i18n setup. Falling back.');
}
if (window?.inboxStore) {
window.inboxStore.dispatch(updateLocale(initialLocale));
i18nLog('Loaded dictionary dispatch');
} else {
i18nLog('No redux store found. Not dispatching dictionary update.');
}
/** NOTE: Because of docstring limitations changes MUST be manually synced between {@link setupI18n.getRawMessage } and {@link window.i18n.getRawMessage } */
/**
* Retrieves a localized message string, without substituting any variables. This resolves any plural forms using the given args
* @param token - The token identifying the message to retrieve.
* @param args - An optional record of substitution variables and their replacement values. This is required if the string has dynamic variables.
*
* @returns The localized message string with substitutions applied.
*
* NOTE: This is intended to be used to get the raw string then format it with {@link formatMessageWithArgs}
*
* @example
* // The string greeting is 'Hello, {name}!' in the current locale
* window.i18n.getRawMessage('greeting', { name: 'Alice' });
* // => 'Hello, {name}!'
*
* // The string search is '{count, plural, one [{found_count} of # match] other [{found_count} of # matches]}' in the current locale
* window.i18n.getRawMessage('search', { count: 1, found_count: 1 });
* // => '{found_count} of {count} match'
*/
function getRawMessage<T extends LocalizerToken, R extends DictionaryWithoutPluralStrings[T]>(
...[token, args]: GetMessageArgs<T>
): R | T {
try {
if (window?.sessionFeatureFlags?.replaceLocalizedStringsWithKeys) {
return token as T;
}
const localizedDictionary = getDictionary({ fallback: initialDictionary });
let localizedString = localizedDictionary[token] as R;
if (!localizedString) {
i18nLog(`Attempted to get translation for nonexistent key: '${token}'`);
localizedString = en[token] as R;
if (!localizedString) {
i18nLog(
`Attempted to get translation for nonexistent key: '${token}' in fallback dictionary`
);
return token as T;
}
}
/** If a localized string does not have any arguments to substitute it is returned with no
* changes. We also need to check if the string contains a curly bracket as if it does
* there might be a default arg */
if (!args && !localizedString.includes('{')) {
return localizedString;
}
if (isPluralForm(localizedString)) {
const pluralKey = getPluralKey(localizedString);
if (!pluralKey) {
i18nLog(`Attempted to nonexistent pluralKey for plural form string '${localizedString}'`);
} else {
const num = args?.[pluralKey as keyof typeof args] ?? 0;
const currentLocale = getLocale();
const cardinalRule = new Intl.PluralRules(currentLocale).select(num);
const pluralString = getStringForCardinalRule(localizedString, cardinalRule);
if (!pluralString) {
i18nLog(`Plural string not found for cardinal '${cardinalRule}': '${localizedString}'`);
return token as T;
}
localizedString = pluralString.replaceAll('#', `${num}`) as R;
}
}
return localizedString;
} catch (error) {
i18nLog(error.message);
return token as T;
}
}
/** NOTE: Because of docstring limitations changes MUST be manually synced between {@link setupI18n.formatMessageWithArgs } and {@link window.i18n.formatMessageWithArgs } */
/**
* Formats a localized message string with arguments and returns the formatted string.
* @param rawMessage - The raw message string to format. After using @see {@link getRawMessage} to get the raw string.
* @param args - An optional record of substitution variables and their replacement values. This
* is required if the string has dynamic variables. This can be optional as a strings args may be defined in @see {@link LOCALE_DEFAULTS}
*
* @returns The formatted message string.
*
* @example
* // The string greeting is 'Hello, {name}!' in the current locale
* window.i18n.getRawMessage('greeting', { name: 'Alice' });
* // => 'Hello, {name}!'
* window.i18n.formatMessageWithArgs('greeting', { name: 'Alice' });
* // => 'Hello, Alice!'
*
* // The string search is '{count, plural, one [{found_count} of # match] other [{found_count} of # matches]}' in the current locale
* window.i18n.getRawMessage('search', { count: 1, found_count: 1 });
* // => '{found_count} of {count} match'
* window.i18n.formatMessageWithArgs('search', { count: 1, found_count: 1 });
* // => '1 of 1 match'
*/
function formatMessageWithArgs<
T extends LocalizerToken,
R extends DictionaryWithoutPluralStrings[T],
>(rawMessage: R, args?: ArgsRecord<T>): R {
/** Find and replace the dynamic variables in a localized string and substitute the variables with the provided values */
// TODO: remove the type casting once we have a proper DictionaryWithoutPluralStrings type
return (rawMessage as `${string}{${string}}${string}`).replace(
/\{(\w+)\}/g,
(match, arg: string) => {
const matchedArg = args ? args[arg as keyof typeof args] : undefined;
return (
matchedArg?.toString() ?? LOCALE_DEFAULTS[arg as keyof typeof LOCALE_DEFAULTS] ?? match
);
}
) as R;
}
/** NOTE: Because of docstring limitations changes MUST be manually synced between {@link setupI18n.getMessage } and {@link window.i18n } */
/**
* Retrieves a localized message string, substituting variables where necessary.
*
* @param token - The token identifying the message to retrieve.
* @param args - An optional record of substitution variables and their replacement values. This is required if the string has dynamic variables.
*
* @returns The localized message string with substitutions applied.
*
* @example
* // The string greeting is 'Hello, {name}!' in the current locale
* window.i18n('greeting', { name: 'Alice' });
* // => 'Hello, Alice!'
*
* // The string search is '{count, plural, one [{found_count} of # match] other [{found_count} of # matches]}' in the current locale
* window.i18n('search', { count: 1, found_count: 1 });
* // => '1 of 1 match'
*/
function getMessage<T extends LocalizerToken, R extends LocalizerDictionary[T]>(
...[token, args]: GetMessageArgs<T>
): R | T {
try {
const rawMessage = getRawMessage<T, R>(...([token, args] as GetMessageArgs<T>));
/** If a localized string does not have any arguments to substitute it is returned with no changes. */
if (!isStringWithArgs<R>(rawMessage)) {
return rawMessage;
}
return formatMessageWithArgs<T, R>(rawMessage, args as ArgsRecord<T>);
} catch (error) {
i18nLog(error.message);
return token as R;
}
}
/** NOTE: Because of docstring limitations changes MUST be manually synced between {@link setupI18n.stripped } and {@link window.i18n.stripped } */
/**
* Retrieves a localized message string, substituting variables where necessary. Then strips the message of any HTML and custom tags.
*
* @param token - The token identifying the message to retrieve.
* @param args - An optional record of substitution variables and their replacement values. This is required if the string has dynamic variables.
*
* @returns The localized message string with substitutions applied. Any HTML and custom tags are removed.
*
* @example
* // The string greeting is 'Hello, {name}! <b>Welcome!</b>' in the current locale
* window.i18n.stripped('greeting', { name: 'Alice' });
* // => 'Hello, Alice! Welcome!'
*/
function stripped<T extends LocalizerToken, R extends LocalizerDictionary[T]>(
...[token, args]: GetMessageArgs<T>
): R | T {
const sanitizedArgs = args ? sanitizeArgs(args, '\u200B') : undefined;
const i18nString = getMessage<T, LocalizerDictionary[T]>(
...([token, sanitizedArgs] as GetMessageArgs<T>)
);
const strippedString = i18nString.replaceAll(/<[^>]*>/g, '');
return deSanitizeHtmlTags(strippedString, '\u200B') as R;
}
/** NOTE: Because of docstring limitations changes MUST be manually synced between {@link setupI18n.inEnglish } and {@link window.i18n.inEnglish } */
/**
* Retrieves a message string in the {@link en} locale, substituting variables where necessary.
*
* NOTE: This does not work for plural strings. This function should only be used for debug and
* non-user-facing strings. Plural string support can be added splitting out the logic for
* {@link setupI18n.formatMessageWithArgs} and creating a new getMessageFromDictionary, which
* specifies takes a dictionary as an argument. This is left as an exercise for the reader.
*
* @param token - The token identifying the message to retrieve.
* @param args - An optional record of substitution variables and their replacement values. This is required if the string has dynamic variables.
*/
function inEnglish<T extends LocalizerToken, R extends LocalizerDictionary[T]>(
...[token, args]: GetMessageArgs<T>
): R | T {
const rawMessage = en[token] as R;
if (!rawMessage) {
i18nLog(
`Attempted to get forced en string for nonexistent key: '${token}' in fallback dictionary`
);
return token as T;
}
return formatMessageWithArgs<T, R>(rawMessage, args as ArgsRecord<T>);
}
getMessage.inEnglish = inEnglish;
getMessage.stripped = stripped;
getMessage.getRawMessage = getRawMessage;
getMessage.formatMessageWithArgs = formatMessageWithArgs;
i18nLog(`Setup Complete with locale: ${initialLocale}`);
return getMessage as SetupI18nReturnType;
};
// eslint-disable-next-line import/no-mutable-exports
export let langNotSupportedMessageShown = false;
export const loadEmojiPanelI18n = async () => {
if (!window) {
return undefined;
}
const lang = getLocale();
if (lang !== 'en') {
try {
const langData = await import(`@emoji-mart/data/i18n/${lang}.json`);
return langData;
} catch (err) {
if (!langNotSupportedMessageShown) {
window?.log?.warn(
'Language is not supported by emoji-mart package. See https://github.com/missive/emoji-mart/tree/main/packages/emoji-mart-data/i18n'
);
langNotSupportedMessageShown = true;
}
}
}
return undefined;
};
/**
* Formats a duration in milliseconds into a localized human-readable string.
*
* @param durationMs - The duration in milliseconds.
* @param options - An optional object containing formatting options.
* @returns A formatted string representing the duration.
*/
export const formatTimeDuration = (
durationMs: number,
options?: Omit<FormatDistanceStrictOptions, 'locale'>
) => {
return formatDistanceStrict(new Date(durationMs), new Date(0), {
locale: getLocaleDictionary(),
...options,
});
};
/**
* date-fns `intervalToDuration` takes a duration in ms.
* This is a simple wrapper to avoid duplicating this (and not forget about it).
*
* Note:
* - date-fns intervalToDuration returns doesn't return 2w for 14d and such, so this forces it to be used.
* - this will throw if the duration is > 4 weeks
*
* @param seconds the seconds to get the durations from
* @returns a date-fns `Duration` type with the fields set
*/
const secondsToDuration = (seconds: number): Duration => {
if (seconds > 3600 * 24 * 28) {
throw new Error('secondsToDuration cannot handle more than 4 weeks for now');
}
const duration = intervalToDuration({ start: 0, end: new Date(seconds * 1000) });
if (!duration) {
throw new Error('intervalToDuration failed to convert duration');
}
if (duration.days) {
duration.weeks = Math.floor(duration.days / 7);
duration.days %= 7;
}
return duration;
};
export const formatWithLocale = ({ formatStr, date }: { date: Date; formatStr: string }) => {
return format(date, formatStr, { locale: getLocaleDictionary() });
};
/**
* Returns a formatted date like `Wednesday, Jun 12, 2024, 4:29 PM`
*/
export const formatFullDate = (date: Date) => {
return date.toLocaleString(getLocale(), {
year: 'numeric',
month: 'short',
weekday: 'long',
day: 'numeric',
hour: 'numeric',
minute: 'numeric',
});
};
export const formatRelativeWithLocale = (timestampMs: number) => {
return formatRelative(timestampMs, Date.now(), { locale: getLocaleDictionary() });
};
/**
* We decided against localizing the abbreviated durations like 1h, 1m, 1s as most apps don't.
* This function just replaces any long form of "seconds?" to "s", "minutes?" to "m", etc.
*
* Note:
* We don't replace to 'months' as it would be the same as 'minutes', so this function shouldn't be used for a string containing months or longer units in it.
*
* Date-fns also doesn't support the 'narrow' syntax for formatDistanceStrict and even if it did, minutes are abbreviated as 'min' in english.
*
* @param unlocalized the string containing the units to abbreviate
* @returns the string with abbreviated units
*/
const unlocalizedDurationToAbbreviated = (unlocalized: string): string => {
return unlocalized
.replace(/ weeks?/g, 'w')
.replace(/ days?/g, 'd')
.replace(/ hours?/g, 'h')
.replace(/ minutes?/g, 'm')
.replace(/ seconds?/g, 's');
};
/**
* Format an expiring/disappearing message timer to its abbreviated form.
* Note: we don't localize this, and cannot have a value > 4 weeks
*
* @param timerSeconds the timer to format, in seconds
* @returns '1h' for a duration of 3600s.
*/
export const formatAbbreviatedExpireTimer = (timerSeconds: number) => {
// Note: we keep this function in this file even if it is not localizing anything
// so we have access to timeLocaleMap.en.
if (timerSeconds > DURATION_SECONDS.WEEKS * 4) {
throw new Error('formatAbbreviatedExpireTimer is not design to handle >4 weeks durations ');
}
const duration = secondsToDuration(timerSeconds);
const unlocalized = formatDuration(duration, {
locale: timeLocaleMap.en, // we want this forced to english
});
return unlocalizedDurationToAbbreviated(unlocalized);
};
/**
* Format an expiring/disappearing message timer to its abbreviated form.
* Note: we don't localize this, and cannot have a value > 4 weeks
*
* @param timerSeconds the timer to format, in seconds
* @returns '1h' for a duration of 3600s.
*/
export const formatAbbreviatedExpireDoubleTimer = (timerSeconds: number) => {
// Note: we keep this function in this file even if it is not localizing anything
// so we have access to timeLocaleMap.en.
if (timerSeconds > DURATION_SECONDS.WEEKS * 4) {
throw new Error(
'formatAbbreviatedExpireDoubleTimer is not design to handle >4 weeks durations '
);
}
if (timerSeconds <= 0) {
return ['0s'];
}
const duration = secondsToDuration(timerSeconds);
const format: Array<keyof Duration> = [];
if (duration.months || duration.years) {
throw new Error("we don't support years or months to be !== 0");
}
if (duration.weeks && format.length < 2) {
format.push('weeks');
}
if (duration.days && format.length < 2) {
format.push('days');
}
if (duration.hours && format.length < 2) {
format.push('hours');
}
if (duration.minutes && format.length < 2) {
format.push('minutes');
}
if (duration.seconds && format.length < 2) {
format.push('seconds');
}
const unlocalized = formatDuration(duration, {
locale: timeLocaleMap.en, // we want this forced to english
delimiter: '#',
format,
});
return unlocalizedDurationToAbbreviated(unlocalized).split('#');
};
export const formatTimeDistanceToNow = (
durationSeconds: number,
options?: Omit<FormatDistanceToNowStrictOptions, 'locale'>
) => {
return formatDistanceToNowStrict(durationSeconds * 1000, {
locale: getLocaleDictionary(),
...options,
});
};
export const formatDateDistanceWithOffset = (date: Date): string => {
const adjustedDate = subMilliseconds(date, GetNetworkTime.getLatestTimestampOffset());
return formatDistanceToNow(adjustedDate, { addSuffix: true, locale: getLocaleDictionary() });
};
/**
* Returns a localized string like "Aug 7, 2024 10:03 AM"
*/
export const getDateAndTimeShort = (date: Date) => {
return formatWithLocale({ date, formatStr: 'Pp' });
};
const getStartOfToday = () => {
const now = new Date();
return new Date(now.getFullYear(), now.getMonth(), now.getDate());
};
/**
* Returns
* - hh:mm when less than 24h ago
* - Tue hh:mm when less than 7d ago
* - dd/mm/yy otherwise
*
*/
export const getConversationItemString = (date: Date) => {
const now = new Date();
// if this is in the future, or older than 7 days ago we display date+time
if (isAfter(date, now) || isBefore(date, subDays(now, 7))) {
const formatter = new Intl.DateTimeFormat(getLocale(), {
year: 'numeric',
month: 'short',
day: 'numeric',
hour: 'numeric',
minute: 'numeric',
hour12: true, // This will switch between 12-hour and 24-hour format depending on the locale
});
return formatter.format(date);
}
// if since our start of the day, display the hour and minute only, am/pm locale dependent
if (isAfter(date, getStartOfToday())) {
const formatter = new Intl.DateTimeFormat(getLocale(), {
hour: 'numeric',
minute: 'numeric',
hour12: true, // This will switch between 12-hour and 24-hour format depending on the locale
});
return formatter.format(date);
}
// less than 7 days ago, display the day of the week + time
const formatter = new Intl.DateTimeFormat(getLocale(), {
weekday: 'short',
hour: 'numeric',
minute: 'numeric',
hour12: true, // This will switch between 12-hour and 24-hour format depending on the locale
});
return formatter.format(date);
};
// RTL Support
export type HTMLDirection = 'ltr' | 'rtl';
export function isRtlBody(): boolean {
const body = document.getElementsByTagName('body').item(0);
return body?.classList.contains('rtl') || false;
}
export const useHTMLDirection = (): HTMLDirection => (isRtlBody() ? 'rtl' : 'ltr');

@ -0,0 +1,25 @@
import { getLocale } from './shared';
let langNotSupportedMessageShown = false;
export const loadEmojiPanelI18n = async () => {
if (!window) {
return undefined;
}
const lang = getLocale();
if (lang !== 'en') {
try {
const langData = await import(`@emoji-mart/data/i18n/${lang}.json`);
return langData;
} catch (err) {
if (!langNotSupportedMessageShown) {
window?.log?.warn(
'Language is not supported by emoji-mart package. See https://github.com/missive/emoji-mart/tree/main/packages/emoji-mart-data/i18n'
);
langNotSupportedMessageShown = true;
}
}
}
return undefined;
};

@ -0,0 +1,49 @@
import { isAfter, isBefore, subDays } from 'date-fns';
import { getLocale } from '../shared';
function getStartOfToday() {
const now = new Date();
return new Date(now.getFullYear(), now.getMonth(), now.getDate());
}
/**
* Returns
* - hh:mm when less than 24h ago
* - Tue hh:mm when less than 7d ago
* - dd/mm/yy otherwise
*
*/
export const getConversationItemString = (date: Date) => {
const now = new Date();
// if this is in the future, or older than 7 days ago we display date+time
if (isAfter(date, now) || isBefore(date, subDays(now, 7))) {
const formatter = new Intl.DateTimeFormat(getLocale(), {
year: 'numeric',
month: 'short',
day: 'numeric',
hour: 'numeric',
minute: 'numeric',
hour12: true, // This will switch between 12-hour and 24-hour format depending on the locale
});
return formatter.format(date);
}
// if since our start of the day, display the hour and minute only, am/pm locale dependent
if (isAfter(date, getStartOfToday())) {
const formatter = new Intl.DateTimeFormat(getLocale(), {
hour: 'numeric',
minute: 'numeric',
hour12: true, // This will switch between 12-hour and 24-hour format depending on the locale
});
return formatter.format(date);
}
// less than 7 days ago, display the day of the week + time
const formatter = new Intl.DateTimeFormat(getLocale(), {
weekday: 'short',
hour: 'numeric',
minute: 'numeric',
hour12: true, // This will switch between 12-hour and 24-hour format depending on the locale
});
return formatter.format(date);
};

@ -0,0 +1,127 @@
import { Duration, formatDuration, intervalToDuration } from 'date-fns';
import { DURATION_SECONDS } from '../../../session/constants';
import { getForcedEnglishTimeLocale } from '../timeLocaleMap';
/**
* We decided against localizing the abbreviated durations like 1h, 1m, 1s as most apps don't.
* This function just replaces any long form of "seconds?" to "s", "minutes?" to "m", etc.
*
* Note:
* We don't replace to 'months' as it would be the same as 'minutes', so this function shouldn't be used for a string containing months or longer units in it.
*
* Date-fns also doesn't support the 'narrow' syntax for formatDistanceStrict and even if it did, minutes are abbreviated as 'min' in english.
*
* @param unlocalized the string containing the units to abbreviate
* @returns the string with abbreviated units
*/
const unlocalizedDurationToAbbreviated = (unlocalized: string): string => {
return unlocalized
.replace(/ weeks?/g, 'w')
.replace(/ days?/g, 'd')
.replace(/ hours?/g, 'h')
.replace(/ minutes?/g, 'm')
.replace(/ seconds?/g, 's');
};
/**
* date-fns `intervalToDuration` takes a duration in ms.
* This is a simple wrapper to avoid duplicating this (and not forget about it).
*
* Note:
* - date-fns intervalToDuration returns doesn't return 2w for 14d and such, so this forces it to be used.
* - this will throw if the duration is > 4 weeks
*
* @param seconds the seconds to get the durations from
* @returns a date-fns `Duration` type with the fields set
*/
const secondsToDuration = (seconds: number): Duration => {
if (seconds > 3600 * 24 * 28) {
throw new Error('secondsToDuration cannot handle more than 4 weeks for now');
}
const duration = intervalToDuration({ start: 0, end: new Date(seconds * 1000) });
if (!duration) {
throw new Error('intervalToDuration failed to convert duration');
}
if (duration.days) {
duration.weeks = Math.floor(duration.days / 7);
duration.days %= 7;
}
return duration;
};
/**
* Format an expiring/disappearing message timer to its abbreviated form.
* Note: we don't localize this, and cannot have a value > 4 weeks
*
* @param timerSeconds the timer to format, in seconds
* @returns '1h' for a duration of 3600s.
*/
export const formatAbbreviatedExpireTimer = (timerSeconds: number) => {
// Note: we keep this function in this file even if it is not localizing anything
// so we have access to timeLocaleMap.en.
if (timerSeconds > DURATION_SECONDS.WEEKS * 4) {
throw new Error('formatAbbreviatedExpireTimer is not design to handle >4 weeks durations ');
}
const duration = secondsToDuration(timerSeconds);
const unlocalized = formatDuration(duration, {
locale: getForcedEnglishTimeLocale(), // we want this forced to english
});
return unlocalizedDurationToAbbreviated(unlocalized);
};
/**
* Format an expiring/disappearing message timer to its abbreviated form.
* Note: we don't localize this, and cannot have a value > 4 weeks
*
* @param timerSeconds the timer to format, in seconds
* @returns '1h' for a duration of 3600s.
*/
export const formatAbbreviatedExpireDoubleTimer = (timerSeconds: number) => {
// Note: we keep this function in this file even if it is not localizing anything
// so we have access to timeLocaleMap.en.
if (timerSeconds > DURATION_SECONDS.WEEKS * 4) {
throw new Error(
'formatAbbreviatedExpireDoubleTimer is not design to handle >4 weeks durations '
);
}
if (timerSeconds <= 0) {
return ['0s'];
}
const duration = secondsToDuration(timerSeconds);
const format: Array<keyof Duration> = [];
if (duration.months || duration.years) {
throw new Error("we don't support years or months to be !== 0");
}
if (duration.weeks && format.length < 2) {
format.push('weeks');
}
if (duration.days && format.length < 2) {
format.push('days');
}
if (duration.hours && format.length < 2) {
format.push('hours');
}
if (duration.minutes && format.length < 2) {
format.push('minutes');
}
if (duration.seconds && format.length < 2) {
format.push('seconds');
}
const unlocalized = formatDuration(duration, {
locale: getForcedEnglishTimeLocale(), // we want this forced to english
delimiter: '#',
format,
});
return unlocalizedDurationToAbbreviated(unlocalized).split('#');
};

@ -0,0 +1,58 @@
import {
FormatDistanceStrictOptions,
formatDistanceStrict,
format,
formatRelative,
FormatDistanceToNowStrictOptions,
formatDistanceToNowStrict,
} from 'date-fns';
import { getTimeLocaleDictionary, getLocale } from '../shared';
/**
* Formats a duration in milliseconds into a localized human-readable string.
*
* @param durationMs - The duration in milliseconds.
* @param options - An optional object containing formatting options.
* @returns A formatted string representing the duration.
*/
export const formatTimeDuration = (
durationMs: number,
options?: Omit<FormatDistanceStrictOptions, 'locale'>
) => {
return formatDistanceStrict(new Date(durationMs), new Date(0), {
locale: getTimeLocaleDictionary(),
...options,
});
};
export const formatWithLocale = ({ formatStr, date }: { date: Date; formatStr: string }) => {
return format(date, formatStr, { locale: getTimeLocaleDictionary() });
};
/**
* Returns a formatted date like `Wednesday, Jun 12, 2024, 4:29 PM`
*/
export const formatFullDate = (date: Date) => {
return date.toLocaleString(getLocale(), {
year: 'numeric',
month: 'short',
weekday: 'long',
day: 'numeric',
hour: 'numeric',
minute: 'numeric',
});
};
export const formatRelativeWithLocale = (timestampMs: number) => {
return formatRelative(timestampMs, Date.now(), { locale: getTimeLocaleDictionary() });
};
export const formatTimeDistanceToNow = (
durationSeconds: number,
options?: Omit<FormatDistanceToNowStrictOptions, 'locale'>
) => {
return formatDistanceToNowStrict(durationSeconds * 1000, {
locale: getTimeLocaleDictionary(),
...options,
});
};

@ -0,0 +1,47 @@
/** NOTE: Because of docstring limitations changes MUST be manually synced between {@link setupI18n.formatMessageWithArgs } and {@link window.i18n.formatMessageWithArgs } */
import { LOCALE_DEFAULTS } from '../../../localization/constants';
import {
ArgsRecord,
DictionaryWithoutPluralStrings,
LocalizerToken,
} from '../../../types/Localizer';
/**
* Formats a localized message string with arguments and returns the formatted string.
* @param rawMessage - The raw message string to format. After using @see {@link getRawMessage} to get the raw string.
* @param args - An optional record of substitution variables and their replacement values. This
* is required if the string has dynamic variables. This can be optional as a strings args may be defined in @see {@link LOCALE_DEFAULTS}
*
* @returns The formatted message string.
*
* @example
* // The string greeting is 'Hello, {name}!' in the current locale
* window.i18n.getRawMessage('greeting', { name: 'Alice' });
* // => 'Hello, {name}!'
* window.i18n.formatMessageWithArgs('greeting', { name: 'Alice' });
* // => 'Hello, Alice!'
*
* // The string search is '{count, plural, one [{found_count} of # match] other [{found_count} of # matches]}' in the current locale
* window.i18n.getRawMessage('search', { count: 1, found_count: 1 });
* // => '{found_count} of {count} match'
* window.i18n.formatMessageWithArgs('search', { count: 1, found_count: 1 });
* // => '1 of 1 match'
*/
export function formatMessageWithArgs<
T extends LocalizerToken,
R extends DictionaryWithoutPluralStrings[T],
>(rawMessage: R, args?: ArgsRecord<T>): R {
/** Find and replace the dynamic variables in a localized string and substitute the variables with the provided values */
// TODO: remove the type casting once we have a proper DictionaryWithoutPluralStrings type
return (rawMessage as `${string}{${string}}${string}`).replace(
/\{(\w+)\}/g,
(match, arg: string) => {
const matchedArg = args ? args[arg as keyof typeof args] : undefined;
return (
matchedArg?.toString() ?? LOCALE_DEFAULTS[arg as keyof typeof LOCALE_DEFAULTS] ?? match
);
}
) as R;
}

@ -0,0 +1,58 @@
/** NOTE: Because of docstring limitations changes MUST be manually synced between {@link setupI18n.getMessage } and {@link window.i18n } */
import {
LocalizerToken,
LocalizerDictionary,
GetMessageArgs,
ArgsRecord,
DictionaryWithoutPluralStrings,
} from '../../../types/Localizer';
import { i18nLog } from '../shared';
import { formatMessageWithArgs } from './formatMessageWithArgs';
import { getRawMessage } from './getRawMessage';
/**
* 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`.
*
* TODO: Change this to a proper type assertion when the type is fixed.
*/
const isStringWithArgs = <R extends DictionaryWithoutPluralStrings[LocalizerToken]>(
localizedString: string
): localizedString is R => localizedString.includes('{');
/**
* Retrieves a localized message string, substituting variables where necessary.
*
* @param token - The token identifying the message to retrieve.
* @param args - An optional record of substitution variables and their replacement values. This is required if the string has dynamic variables.
*
* @returns The localized message string with substitutions applied.
*
* @example
* // The string greeting is 'Hello, {name}!' in the current locale
* window.i18n('greeting', { name: 'Alice' });
* // => 'Hello, Alice!'
*
* // The string search is '{count, plural, one [{found_count} of # match] other [{found_count} of # matches]}' in the current locale
* window.i18n('search', { count: 1, found_count: 1 });
* // => '1 of 1 match'
*/
export function getMessage<T extends LocalizerToken, R extends LocalizerDictionary[T]>(
...[token, args]: GetMessageArgs<T>
): R | T {
try {
const rawMessage = getRawMessage<T, R>(...([token, args] as GetMessageArgs<T>));
/** If a localized string does not have any arguments to substitute it is returned with no changes. */
if (!isStringWithArgs<R>(rawMessage)) {
return rawMessage;
}
return formatMessageWithArgs<T, R>(rawMessage, args as ArgsRecord<T>);
} catch (error) {
i18nLog(error.message);
return token as R;
}
}

@ -0,0 +1,117 @@
/** NOTE: Because of docstring limitations changes MUST be manually synced between {@link setupI18n.getRawMessage } and {@link window.i18n.getRawMessage } */
import { en } from '../../../localization/locales';
import {
DictionaryWithoutPluralStrings,
GetMessageArgs,
LocalizerToken,
PluralKey,
PluralString,
} from '../../../types/Localizer';
import { getLocale, i18nLog } from '../shared';
import { getTranslationDictionary } from '../translationDictionaries';
function getPluralKey<R extends PluralKey | undefined>(string: PluralString): R {
const match = /{(\w+), plural, one \[.+\] other \[.+\]}/g.exec(string);
return (match?.[1] ?? undefined) 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<Intl.LDMLPluralRule, RegExp> = {
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);
/**
* Retrieves a localized message string, without substituting any variables. This resolves any plural forms using the given args
* @param token - The token identifying the message to retrieve.
* @param args - An optional record of substitution variables and their replacement values. This is required if the string has dynamic variables.
*
* @returns The localized message string with substitutions applied.
*
* NOTE: This is intended to be used to get the raw string then format it with {@link formatMessageWithArgs}
*
* @example
* // The string greeting is 'Hello, {name}!' in the current locale
* window.i18n.getRawMessage('greeting', { name: 'Alice' });
* // => 'Hello, {name}!'
*
* // The string search is '{count, plural, one [{found_count} of # match] other [{found_count} of # matches]}' in the current locale
* window.i18n.getRawMessage('search', { count: 1, found_count: 1 });
* // => '{found_count} of {count} match'
*/
export function getRawMessage<
T extends LocalizerToken,
R extends DictionaryWithoutPluralStrings[T],
>(...[token, args]: GetMessageArgs<T>): R | T {
try {
if (window?.sessionFeatureFlags?.replaceLocalizedStringsWithKeys) {
return token as T;
}
const localizedDictionary = getTranslationDictionary();
let localizedString = localizedDictionary[token] as R;
if (!localizedString) {
i18nLog(`Attempted to get translation for nonexistent key: '${token}'`);
localizedString = en[token] as R;
if (!localizedString) {
i18nLog(
`Attempted to get translation for nonexistent key: '${token}' in fallback dictionary`
);
return token as T;
}
}
/** If a localized string does not have any arguments to substitute it is returned with no
* changes. We also need to check if the string contains a curly bracket as if it does
* there might be a default arg */
if (!args && !localizedString.includes('{')) {
return localizedString;
}
if (isPluralForm(localizedString)) {
const pluralKey = getPluralKey(localizedString);
if (!pluralKey) {
i18nLog(`Attempted to nonexistent pluralKey for plural form string '${localizedString}'`);
} else {
const num = args?.[pluralKey as keyof typeof args] ?? 0;
const currentLocale = getLocale();
const cardinalRule = new Intl.PluralRules(currentLocale).select(num);
const pluralString = getStringForCardinalRule(localizedString, cardinalRule);
if (!pluralString) {
i18nLog(`Plural string not found for cardinal '${cardinalRule}': '${localizedString}'`);
return token as T;
}
localizedString = pluralString.replaceAll('#', `${num}`) as R;
}
}
return localizedString;
} catch (error) {
i18nLog(error.message);
return token as T;
}
}

@ -0,0 +1,35 @@
/** NOTE: Because of docstring limitations changes MUST be manually synced between {@link setupI18n.inEnglish } and {@link window.i18n.inEnglish } */
import { en } from '../../../localization/locales';
import {
LocalizerToken,
LocalizerDictionary,
GetMessageArgs,
ArgsRecord,
} from '../../../types/Localizer';
import { i18nLog } from '../shared';
import { formatMessageWithArgs } from './formatMessageWithArgs';
/**
* Retrieves a message string in the {@link en} locale, substituting variables where necessary.
*
* NOTE: This does not work for plural strings. This function should only be used for debug and
* non-user-facing strings. Plural string support can be added splitting out the logic for
* {@link setupI18n.formatMessageWithArgs} and creating a new getMessageFromDictionary, which
* specifies takes a dictionary as an argument. This is left as an exercise for the reader.
*
* @param token - The token identifying the message to retrieve.
* @param args - An optional record of substitution variables and their replacement values. This is required if the string has dynamic variables.
*/
export function inEnglish<T extends LocalizerToken, R extends LocalizerDictionary[T]>(
...[token, args]: GetMessageArgs<T>
): R | T {
const rawMessage = en[token] as R;
if (!rawMessage) {
i18nLog(
`Attempted to get forced en string for nonexistent key: '${token}' in fallback dictionary`
);
return token as T;
}
return formatMessageWithArgs<T, R>(rawMessage, args as ArgsRecord<T>);
}

@ -0,0 +1,32 @@
/** NOTE: Because of docstring limitations changes MUST be manually synced between {@link setupI18n.stripped } and {@link window.i18n.stripped } */
import { deSanitizeHtmlTags, sanitizeArgs } from '../../../components/basic/I18n';
import { GetMessageArgs, LocalizerDictionary, LocalizerToken } from '../../../types/Localizer';
import { getMessage } from './getMessage';
/**
* Retrieves a localized message string, substituting variables where necessary. Then strips the message of any HTML and custom tags.
*
* @param token - The token identifying the message to retrieve.
* @param args - An optional record of substitution variables and their replacement values. This is required if the string has dynamic variables.
*
* @returns The localized message string with substitutions applied. Any HTML and custom tags are removed.
*
* @example
* // The string greeting is 'Hello, {name}! <b>Welcome!</b>' in the current locale
* window.i18n.stripped('greeting', { name: 'Alice' });
* // => 'Hello, Alice! Welcome!'
*/
export function stripped<T extends LocalizerToken, R extends LocalizerDictionary[T]>(
...[token, args]: GetMessageArgs<T>
): R | T {
const sanitizedArgs = args ? sanitizeArgs(args, '\u200B') : undefined;
const i18nString = getMessage<T, LocalizerDictionary[T]>(
...([token, sanitizedArgs] as GetMessageArgs<T>)
);
const strippedString = i18nString.replaceAll(/<[^>]*>/g, '');
return deSanitizeHtmlTags(strippedString, '\u200B') as R;
}

@ -0,0 +1,55 @@
// this file is a weird one as it is used by both sides of electron at the same time
import { isEmpty } from 'lodash';
import { LocalizerDictionary, SetupI18nReturnType } from '../../types/Localizer';
import { formatMessageWithArgs } from './functions/formatMessageWithArgs';
import { getMessage } from './functions/getMessage';
import { getRawMessage } from './functions/getRawMessage';
import { inEnglish } from './functions/inEnglish';
import { stripped } from './functions/stripped';
import { i18nLog, Locale, setInitialLocale } from './shared';
import { setTranslationDictionary } from './translationDictionaries';
export function loadDictionary(locale: Locale) {
return import(`../../_locales/${locale}/messages.json`) as Promise<LocalizerDictionary>;
}
/**
* Sets up the i18n function with the provided locale and messages.
*
* @param params - An object containing optional parameters.
* @param params.locale - The locale to use for translations
* @param params.translationDictionary - A dictionary of localized messages. Defaults to {@link en}.
*
* @returns A function that retrieves a localized message string, substituting variables where necessary.
*/
export const setupI18n = ({
locale,
translationDictionary,
}: {
locale: Locale;
translationDictionary: LocalizerDictionary;
}): SetupI18nReturnType => {
if (!locale) {
throw new Error(`locale not provided in i18n setup`);
}
if (!translationDictionary || isEmpty(translationDictionary)) {
throw new Error('translationDictionary was not provided');
}
setTranslationDictionary(translationDictionary);
setInitialLocale(locale);
const getMessageWithFunctions = getMessage;
// TODO - fix those `any`
(getMessageWithFunctions as any).inEnglish = inEnglish;
(getMessageWithFunctions as any).stripped = stripped;
(getMessageWithFunctions as any).getRawMessage = getRawMessage;
(getMessageWithFunctions as any).formatMessageWithArgs = formatMessageWithArgs;
i18nLog(`Setup Complete with locale: ${locale}`);
return getMessageWithFunctions as SetupI18nReturnType;
};

@ -0,0 +1,11 @@
// RTL Support
export type HTMLDirection = 'ltr' | 'rtl';
export function isRtlBody(): boolean {
const body = document.getElementsByTagName('body').item(0);
return body?.classList.contains('rtl') || false;
}
export const useHTMLDirection = (): HTMLDirection => (isRtlBody() ? 'rtl' : 'ltr');

@ -0,0 +1,39 @@
import { timeLocaleMap } from './timeLocaleMap';
/**
* Logs an i18n message to the console.
* @param message - The message to log.
*
* TODO - Replace this logging method when the new logger is created
*/
export function i18nLog(message: string) {
// eslint:disable: no-console
// eslint-disable-next-line no-console
(window?.log?.error ?? console.log)(`i18n: ${message}`);
}
export type Locale = keyof typeof timeLocaleMap;
export function getTimeLocaleDictionary() {
return timeLocaleMap[getLocale()];
}
/**
* Returns the current locale.
* @param params - An object containing optional parameters.
* @param params.fallback - The fallback locale to use if redux is not available. Defaults to en.
*/
export function getLocale(): Locale {
if (!initialLocale) {
i18nLog(`getLocale: using initialLocale: ${initialLocale}`);
throw new Error('initialLocale is unset');
}
return initialLocale;
}
let initialLocale: Locale | undefined;
export function setInitialLocale(locale: Locale) {
initialLocale = locale;
}

@ -0,0 +1,73 @@
import timeLocales from 'date-fns/locale';
export const timeLocaleMap = {
ar: timeLocales.ar,
be: timeLocales.be,
bg: timeLocales.bg,
ca: timeLocales.ca,
cs: timeLocales.cs,
da: timeLocales.da,
de: timeLocales.de,
el: timeLocales.el,
en: timeLocales.enUS,
eo: timeLocales.eo,
es: timeLocales.es,
/** TODO - Check this */
es_419: timeLocales.es,
et: timeLocales.et,
fa: timeLocales.faIR,
fi: timeLocales.fi,
/** TODO - Check this */
fil: timeLocales.fi,
fr: timeLocales.fr,
he: timeLocales.he,
hi: timeLocales.hi,
hr: timeLocales.hr,
hu: timeLocales.hu,
/** TODO - Check this */
'hy-AM': timeLocales.hy,
id: timeLocales.id,
it: timeLocales.it,
ja: timeLocales.ja,
ka: timeLocales.ka,
km: timeLocales.km,
/** TODO - Check this */
kmr: timeLocales.km,
kn: timeLocales.kn,
ko: timeLocales.ko,
lt: timeLocales.lt,
lv: timeLocales.lv,
mk: timeLocales.mk,
nb: timeLocales.nb,
nl: timeLocales.nl,
/** TODO - Find this this */
no: timeLocales.enUS,
/** TODO - Find this this */
pa: timeLocales.enUS,
pl: timeLocales.pl,
pt_BR: timeLocales.ptBR,
pt_PT: timeLocales.pt,
ro: timeLocales.ro,
ru: timeLocales.ru,
/** TODO - Find this this */
si: timeLocales.enUS,
sk: timeLocales.sk,
sl: timeLocales.sl,
sq: timeLocales.sq,
sr: timeLocales.sr,
sv: timeLocales.sv,
ta: timeLocales.ta,
th: timeLocales.th,
/** TODO - Find this this */
tl: timeLocales.enUS,
tr: timeLocales.tr,
uk: timeLocales.uk,
uz: timeLocales.uz,
vi: timeLocales.vi,
zh_CN: timeLocales.zhCN,
zh_TW: timeLocales.zhTW,
};
export function getForcedEnglishTimeLocale() {
return timeLocaleMap.en;
}

@ -0,0 +1,31 @@
import { en } from '../../localization/locales';
import { LocalizerDictionary } from '../../types/Localizer';
import { i18nLog } from './shared';
let translationDictionary: LocalizerDictionary | undefined;
export function setTranslationDictionary(dictionary: LocalizerDictionary) {
if (translationDictionary) {
throw new Error('translationDictionary is already init');
}
translationDictionary = dictionary;
}
/**
* Only exported for testing, reset the dictionary to use for translations.
*/
export function resetTranslationDictionary() {
translationDictionary = undefined;
}
/**
* Returns the current dictionary to use for translations.
*/
export function getTranslationDictionary(): LocalizerDictionary {
if (translationDictionary) {
return translationDictionary;
}
i18nLog('getTranslationDictionary: dictionary not init yet. Using en.');
return en;
}
Loading…
Cancel
Save