Skip to content

Commit

Permalink
Use i18next plurals
Browse files Browse the repository at this point in the history
  • Loading branch information
tassoevan committed Jan 12, 2024
1 parent 7d9a2c4 commit be1718b
Show file tree
Hide file tree
Showing 45 changed files with 335 additions and 362 deletions.
2 changes: 1 addition & 1 deletion apps/meteor/.scripts/translation-check.ts
Original file line number Diff line number Diff line change
Expand Up @@ -237,8 +237,8 @@ const checkFiles = async (sourceDirPath: string, sourceLng: string, fix = false)

for await (const { path, json, lng } of translations) {
await checkPlaceholdersFormat({ json, path, fix });
await checkMissingPlurals({ json, path, lng, fix });
await checkExceedingKeys({ json, path, lng, sourceJson, sourceLng, fix });
await checkMissingPlurals({ json, path, lng, fix });
}
};

Expand Down
20 changes: 1 addition & 19 deletions apps/meteor/app/utils/lib/i18n.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,31 +6,13 @@ import { isObject } from '../../../lib/utils/isObject';

export const i18n = i18next.use(sprintf);

type HasCounter = {
counter?: number;
};

const hasCounterProperty = (obj: any): obj is HasCounter => obj?.hasOwnProperty('counter') && typeof obj.counter === 'number';

export const addSprinfToI18n = function (t: (typeof i18n)['t'], i18nInstance?: typeof i18n) {
export const addSprinfToI18n = function (t: (typeof i18n)['t']) {
return function (key: string, ...replaces: any): string {
if (replaces[0] === undefined) {
return t(key, ...replaces);
}

if (isObject(replaces[0]) && !Array.isArray(replaces[0])) {
if (!hasCounterProperty(replaces[0])) {
return t(key, ...replaces);
}

const pluralKey = `${key}_plural`;
const pluralKeyExists = i18nInstance?.exists(pluralKey);
const counter = replaces[0]?.counter;

if (counter !== undefined && pluralKeyExists) {
return counter === 1 ? t(key, ...replaces) : t(pluralKey, ...replaces);
}

return t(key, ...replaces);
}

Expand Down
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
import { Icon } from '@rocket.chat/fuselage';
import { useConnectionStatus, useTranslation } from '@rocket.chat/ui-contexts';
import type { MouseEventHandler, FC } from 'react';
import type { MouseEventHandler } from 'react';
import React, { useEffect, useRef, useState } from 'react';

import './ConnectionStatusBar.styles.css';
Expand Down Expand Up @@ -45,7 +45,7 @@ const useReconnectCountdown = (
return reconnectCountdown;
};

const ConnectionStatusBar: FC = function ConnectionStatusBar() {
function ConnectionStatusBar() {
const { connected, retryTime, status, reconnect } = useConnectionStatus();
const reconnectCountdown = useReconnectCountdown(retryTime, status);
const t = useTranslation();
Expand All @@ -65,12 +65,7 @@ const ConnectionStatusBar: FC = function ConnectionStatusBar() {
<Icon name='warning' /> {t('meteor_status' as Parameters<typeof t>[0], { context: status })}
</strong>

{status === 'waiting' && (
<>
{' '}
{t(reconnectCountdown === 1 ? 'meteor_status_reconnect_in' : 'meteor_status_reconnect_in_plural', { count: reconnectCountdown })}
</>
)}
{status === 'waiting' && <> {t('meteor_status_reconnect_in', { count: reconnectCountdown })}</>}

{['waiting', 'offline'].includes(status) && (
<>
Expand All @@ -82,6 +77,6 @@ const ConnectionStatusBar: FC = function ConnectionStatusBar() {
)}
</div>
);
};
}

export default ConnectionStatusBar;
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,7 @@ const DiscussionMetrics = ({ lm, count, rid, drid }: DiscussionMetricsProps): Re
<MessageBlock>
<MessageMetrics>
<MessageMetricsReply data-rid={rid} data-drid={drid} onClick={() => goToRoom(drid)}>
{count ? t('message_counter', { counter: count, count }) : t('Reply')}
{count ? t('message_counter', { count }) : t('Reply')}
</MessageMetricsReply>
<MessageMetricsItem title={lm?.toLocaleString()}>
<MessageMetricsItem.Icon name='clock' />
Expand Down
2 changes: 1 addition & 1 deletion apps/meteor/client/providers/TranslationProvider.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -238,7 +238,7 @@ const TranslationProviderInner = ({
loadLanguage: async (language: string) => {
i18n.changeLanguage(language);
},
translate: Object.assign(addSprinfToI18n(t, i18n), {
translate: Object.assign(addSprinfToI18n(t), {
has: ((key, options) => key && i18n.exists(key, options)) as TranslationContextValue['translate']['has'],
}),
}),
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -42,17 +42,17 @@ const getBadgeTitle = (
) => {
const title = [] as string[];
if (userMentions) {
title.push(t('mentions_counter', { counter: userMentions }));
title.push(t('mentions_counter', { count: userMentions }));
}
if (threadUnread) {
title.push(t('threads_counter', { counter: threadUnread }));
title.push(t('threads_counter', { count: threadUnread }));
}
if (groupMentions) {
title.push(t('group_mentions_counter', { counter: groupMentions }));
title.push(t('group_mentions_counter', { count: groupMentions }));
}
const count = unread - userMentions - groupMentions;
if (count > 0) {
title.push(t('unread_messages_counter', { counter: count }));
title.push(t('unread_messages_counter', { count }));
}
return title.join(', ');
};
Expand Down
16 changes: 2 additions & 14 deletions apps/meteor/client/sidebar/footer/voip/index.tsx
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import type { VoIpCallerInfo } from '@rocket.chat/core-typings';
import { useEndpoint, useTranslation } from '@rocket.chat/ui-contexts';
import type { ReactElement } from 'react';
import React, { useCallback, useMemo, useState } from 'react';
import React, { useCallback, useState } from 'react';

import { useVoipFooterMenu } from '../../../../ee/client/hooks/useVoipFooterMenu';
import {
Expand Down Expand Up @@ -62,18 +62,6 @@ export const VoipFooter = (): ReactElement | null => {
return subtitles[state] || '';
};

const getCallsInQueueText = useMemo((): string => {
if (queueCounter === 0) {
return t('Calls_in_queue_empty');
}

if (queueCounter === 1) {
return t('Calls_in_queue', { calls: queueCounter });
}

return t('Calls_in_queue_plural', { calls: queueCounter });
}, [queueCounter, t]);

if (!('caller' in callerInfo)) {
return <SidebarFooterDefault />;
}
Expand All @@ -91,7 +79,7 @@ export const VoipFooter = (): ReactElement | null => {
togglePause={togglePause}
createRoom={createRoom}
openRoom={openRoom}
callsInQueue={getCallsInQueueText}
callsInQueue={t('Calls_in_queue', { count: queueCounter })}
dispatchEvent={dispatchEvent}
openedRoomInfo={openedRoomInfo}
isEnterprise={isEnterprise}
Expand Down
Original file line number Diff line number Diff line change
@@ -1,9 +1,10 @@
import { Pagination, Field, FieldLabel, FieldRow } from '@rocket.chat/fuselage';
import { useDebouncedValue, useMediaQuery, useMutableCallback } from '@rocket.chat/fuselage-hooks';
import { useEndpoint, useToastMessageDispatch, useRoute, useTranslation } from '@rocket.chat/ui-contexts';
import { useEndpoint, useToastMessageDispatch, useRoute } from '@rocket.chat/ui-contexts';
import { useQuery } from '@tanstack/react-query';
import type { FC } from 'react';
import React, { useMemo, useState } from 'react';
import { useTranslation } from 'react-i18next';

import FilterByText from '../../../components/FilterByText';
import GenericNoResults from '../../../components/GenericNoResults';
Expand All @@ -23,7 +24,7 @@ import DateRangePicker from './helpers/DateRangePicker';
const ModerationConsoleTable: FC = () => {
const [text, setText] = useState('');
const moderationRoute = useRoute('moderation-console');
const t = useTranslation();
const { t } = useTranslation();
const isDesktopOrLarger = useMediaQuery('(min-width: 1024px)');

const { sortBy, sortDirection, setSort } = useSort<'reports.ts' | 'reports.message.u.username' | 'reports.description' | 'count'>(
Expand Down Expand Up @@ -110,7 +111,7 @@ const ModerationConsoleTable: FC = () => {
{t('Moderation_Report_date')}
</GenericTableHeaderCell>,
<GenericTableHeaderCell key='reports' direction={sortDirection} active={sortBy === 'count'} onClick={setSort} sort='count'>
{t('Moderation_Report_plural')}
{t('Moderation_Report_reports')}
</GenericTableHeaderCell>,
<GenericTableHeaderCell key='actions' width='x48' />,
],
Expand Down
Original file line number Diff line number Diff line change
@@ -1,12 +1,13 @@
import { useEndpoint, useRouter, useSetModal, useToastMessageDispatch, useTranslation } from '@rocket.chat/ui-contexts';
import { useEndpoint, useRouter, useSetModal, useToastMessageDispatch } from '@rocket.chat/ui-contexts';
import { useMutation, useQueryClient } from '@tanstack/react-query';
import React from 'react';
import { useTranslation } from 'react-i18next';

import type { GenericMenuItemProps } from '../../../../components/GenericMenu/GenericMenuItem';
import GenericModal from '../../../../components/GenericModal';

const useDismissUserAction = (userId: string): GenericMenuItemProps => {
const t = useTranslation();
const { t } = useTranslation();
const setModal = useSetModal();
const dispatchToastMessage = useToastMessageDispatch();
const moderationRoute = useRouter();
Expand All @@ -20,7 +21,7 @@ const useDismissUserAction = (userId: string): GenericMenuItemProps => {
dispatchToastMessage({ type: 'error', message: error });
},
onSuccess: () => {
dispatchToastMessage({ type: 'success', message: t('Moderation_Reports_dismissed_plural') });
dispatchToastMessage({ type: 'success', message: t('Moderation_Reports_all_dismissed') });
},
});

Expand Down
Original file line number Diff line number Diff line change
@@ -1,9 +1,8 @@
import { Field } from '@rocket.chat/fuselage';
import type { ServerMethods } from '@rocket.chat/ui-contexts';
import type { ServerMethods, TranslationKey } from '@rocket.chat/ui-contexts';
import type { ComponentMeta, ComponentStory } from '@storybook/react';
import React from 'react';

import type keys from '../../../../../packages/rocketchat-i18n/i18n/en.i18n.json';
import ActionSettingInput from './ActionSettingInput';

export default {
Expand All @@ -17,22 +16,22 @@ const Template: ComponentStory<typeof ActionSettingInput> = (args) => <ActionSet
export const Default = Template.bind({});
Default.args = {
_id: 'setting_id',
actionText: 'Action text' as keyof typeof keys,
actionText: 'Action text' as TranslationKey,
value: 'methodName' as keyof ServerMethods,
};

export const Disabled = Template.bind({});
Disabled.args = {
_id: 'setting_id',
actionText: 'Action text' as keyof typeof keys,
actionText: 'Action text' as TranslationKey,
value: 'methodName' as keyof ServerMethods,
disabled: true,
};

export const WithinChangedSection = Template.bind({});
WithinChangedSection.args = {
_id: 'setting_id',
actionText: 'Action text' as keyof typeof keys,
actionText: 'Action text' as TranslationKey,
value: 'methodName' as keyof ServerMethods,
sectionChanged: true,
};
Original file line number Diff line number Diff line change
@@ -1,8 +1,8 @@
import { Field } from '@rocket.chat/fuselage';
import type { TranslationKey } from '@rocket.chat/ui-contexts';
import type { ComponentMeta, ComponentStory } from '@storybook/react';
import React from 'react';

import type keys from '../../../../../packages/rocketchat-i18n/i18n/en.i18n.json';
import type { valuesOption } from './MultiSelectSettingInput';
import MultiSelectSettingInput from './MultiSelectSettingInput';

Expand All @@ -20,9 +20,9 @@ export default {
const Template: ComponentStory<typeof MultiSelectSettingInput> = (args) => <MultiSelectSettingInput {...args} />;

const options: valuesOption[] = [
{ key: '1', i18nLabel: '1' as keyof typeof keys },
{ key: '2', i18nLabel: '2' as keyof typeof keys },
{ key: '3', i18nLabel: '3' as keyof typeof keys },
{ key: '1', i18nLabel: '1' as TranslationKey },
{ key: '2', i18nLabel: '2' as TranslationKey },
{ key: '3', i18nLabel: '3' as TranslationKey },
];

export const Default = Template.bind({});
Expand Down
Original file line number Diff line number Diff line change
@@ -1,8 +1,8 @@
import { Field } from '@rocket.chat/fuselage';
import type { TranslationKey } from '@rocket.chat/ui-contexts';
import type { ComponentMeta, ComponentStory } from '@storybook/react';
import React from 'react';

import type keys from '../../../../../packages/rocketchat-i18n/i18n/en.i18n.json';
import SelectSettingInput from './SelectSettingInput';

export default {
Expand All @@ -24,9 +24,9 @@ Default.args = {
label: 'Label',
placeholder: 'Placeholder',
values: [
{ key: '1', i18nLabel: '1' as keyof typeof keys },
{ key: '2', i18nLabel: '2' as keyof typeof keys },
{ key: '3', i18nLabel: '3' as keyof typeof keys },
{ key: '1', i18nLabel: '1' as TranslationKey },
{ key: '2', i18nLabel: '2' as TranslationKey },
{ key: '3', i18nLabel: '3' as TranslationKey },
],
};

Expand All @@ -36,9 +36,9 @@ Disabled.args = {
label: 'Label',
placeholder: 'Placeholder',
values: [
{ key: '1', i18nLabel: '1' as keyof typeof keys },
{ key: '2', i18nLabel: '2' as keyof typeof keys },
{ key: '3', i18nLabel: '3' as keyof typeof keys },
{ key: '1', i18nLabel: '1' as TranslationKey },
{ key: '2', i18nLabel: '2' as TranslationKey },
{ key: '3', i18nLabel: '3' as TranslationKey },
],
disabled: true,
};
Expand All @@ -50,9 +50,9 @@ WithValue.args = {
placeholder: 'Placeholder',
value: '2',
values: [
{ key: '1', i18nLabel: '1' as keyof typeof keys },
{ key: '2', i18nLabel: '2' as keyof typeof keys },
{ key: '3', i18nLabel: '3' as keyof typeof keys },
{ key: '1', i18nLabel: '1' as TranslationKey },
{ key: '2', i18nLabel: '2' as TranslationKey },
{ key: '3', i18nLabel: '3' as TranslationKey },
],
};

Expand All @@ -62,9 +62,9 @@ WithResetButton.args = {
label: 'Label',
placeholder: 'Placeholder',
values: [
{ key: '1', i18nLabel: '1' as keyof typeof keys },
{ key: '2', i18nLabel: '2' as keyof typeof keys },
{ key: '3', i18nLabel: '3' as keyof typeof keys },
{ key: '1', i18nLabel: '1' as TranslationKey },
{ key: '2', i18nLabel: '2' as TranslationKey },
{ key: '3', i18nLabel: '3' as TranslationKey },
],
hasResetButton: true,
};
Original file line number Diff line number Diff line change
Expand Up @@ -19,12 +19,9 @@ const EnabledAppsCount = ({
}): ReactElement | null => {
const t = useTranslation();

const privateAppsCountText: string = t('Private_Apps_Count_Enabled', { counter: enabled });
const marketplaceAppsCountText: string = t('Apps_Count_Enabled', { counter: enabled });

return (
<GenericResourceUsage
title={context === 'private' ? privateAppsCountText : marketplaceAppsCountText}
title={context === 'private' ? t('Private_Apps_Count_Enabled', { count: enabled }) : t('Apps_Count_Enabled', { count: enabled })}
value={enabled}
max={limit}
percentage={percentage}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -96,7 +96,7 @@ const PruneMessagesWithData = (): ReactElement => {
throw new Error(t('No_messages_found_to_prune'));
}

dispatchToastMessage({ type: 'success', message: t('__count__message_pruned', { counter: count }) });
dispatchToastMessage({ type: 'success', message: t('__count__message_pruned', { count }) });
methods.reset();
} catch (error: unknown) {
dispatchToastMessage({ type: 'error', message: error });
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -94,7 +94,7 @@ const VideoConfListItem = ({
</Avatar.Stack>
<Box mis={4}>
{joinedUsers.length > VIDEOCONF_STACK_MAX_USERS
? t('__usersCount__member_joined', { counter: joinedUsers.length - VIDEOCONF_STACK_MAX_USERS })
? t('__usersCount__member_joined', { count: joinedUsers.length - VIDEOCONF_STACK_MAX_USERS })
: t('joined')}
</Box>
</Box>
Expand Down
Loading

0 comments on commit be1718b

Please sign in to comment.