2020-10-30 20:34:04 +00:00
|
|
|
// Copyright 2019-2020 Signal Messenger, LLC
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2019-12-17 20:25:57 +00:00
|
|
|
import * as React from 'react';
|
|
|
|
|
|
|
|
export type I18nFn = (
|
|
|
|
key: string,
|
2020-07-29 23:20:05 +00:00
|
|
|
substitutions?: Array<string | number> | ReplacementValuesType
|
2019-12-17 20:25:57 +00:00
|
|
|
) => string;
|
|
|
|
|
2020-07-29 23:20:05 +00:00
|
|
|
export type ReplacementValuesType = {
|
|
|
|
[key: string]: string | number;
|
|
|
|
};
|
|
|
|
|
2019-12-17 20:25:57 +00:00
|
|
|
const I18nContext = React.createContext<I18nFn>(() => 'NO LOCALE LOADED');
|
|
|
|
|
|
|
|
export type I18nProps = {
|
|
|
|
children: React.ReactNode;
|
|
|
|
messages: { [key: string]: { message: string } };
|
|
|
|
};
|
|
|
|
|
2020-09-14 21:56:35 +00:00
|
|
|
export const I18n = ({ messages, children }: I18nProps): JSX.Element => {
|
2019-12-17 20:25:57 +00:00
|
|
|
const getMessage = React.useCallback<I18nFn>(
|
2020-07-29 23:20:05 +00:00
|
|
|
(key, substitutions) => {
|
|
|
|
if (Array.isArray(substitutions) && substitutions.length > 1) {
|
|
|
|
throw new Error(
|
|
|
|
'Array syntax is not supported with more than one placeholder'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
const { message } = messages[key];
|
|
|
|
if (!substitutions) {
|
|
|
|
return message;
|
2020-09-14 21:56:35 +00:00
|
|
|
}
|
|
|
|
if (Array.isArray(substitutions)) {
|
2020-07-29 23:20:05 +00:00
|
|
|
return substitutions.reduce(
|
|
|
|
(result, substitution) =>
|
|
|
|
result.toString().replace(/\$.+?\$/, substitution.toString()),
|
|
|
|
message
|
|
|
|
) as string;
|
|
|
|
}
|
|
|
|
|
|
|
|
const FIND_REPLACEMENTS = /\$([^$]+)\$/g;
|
|
|
|
|
|
|
|
let match = FIND_REPLACEMENTS.exec(message);
|
|
|
|
let builder = '';
|
|
|
|
let lastTextIndex = 0;
|
|
|
|
|
|
|
|
while (match) {
|
|
|
|
if (lastTextIndex < match.index) {
|
|
|
|
builder += message.slice(lastTextIndex, match.index);
|
|
|
|
}
|
|
|
|
|
|
|
|
const placeholderName = match[1];
|
|
|
|
const value = substitutions[placeholderName];
|
|
|
|
if (!value) {
|
2020-09-14 21:56:35 +00:00
|
|
|
// eslint-disable-next-line no-console
|
2020-07-29 23:20:05 +00:00
|
|
|
console.error(
|
|
|
|
`i18n: Value not provided for placeholder ${placeholderName} in key '${key}'`
|
|
|
|
);
|
|
|
|
}
|
|
|
|
builder += value || '';
|
|
|
|
|
|
|
|
lastTextIndex = FIND_REPLACEMENTS.lastIndex;
|
|
|
|
match = FIND_REPLACEMENTS.exec(message);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (lastTextIndex < message.length) {
|
|
|
|
builder += message.slice(lastTextIndex);
|
|
|
|
}
|
|
|
|
|
|
|
|
return builder;
|
|
|
|
},
|
2019-12-17 20:25:57 +00:00
|
|
|
[messages]
|
|
|
|
);
|
|
|
|
|
|
|
|
return (
|
|
|
|
<I18nContext.Provider value={getMessage}>{children}</I18nContext.Provider>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
2020-09-14 21:56:35 +00:00
|
|
|
export const useI18n = (): I18nFn => React.useContext(I18nContext);
|