0
Fork 0
mirror of https://github.com/verdaccio/verdaccio.git synced 2024-12-30 22:34:10 -05:00

refactor: one single file for i18n conf ui (#2440)

* refactor: one single file for i18n conf ui

* add test for crowdin

* cleanup

* space
This commit is contained in:
Juan Picado 2021-09-17 07:17:34 +02:00 committed by GitHub
parent 7dde848d0c
commit 5e784d1188
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
6 changed files with 229 additions and 81 deletions

View file

@ -5,6 +5,10 @@ preserve_hierarchy: true
files: files:
[ [
{
source: 'packages/plugins/ui-theme/src/i18n/crowdin/**/*',
translation: '/ui/i18n/%locale%/**/%original_file_name%',
},
{ {
source: '/website/i18n/en/**/*', source: '/website/i18n/en/**/*',
translation: '/website/i18n/%locale%/**/%original_file_name%', translation: '/website/i18n/%locale%/**/%original_file_name%',

View file

@ -2,71 +2,24 @@
import styled from '@emotion/styled'; import styled from '@emotion/styled';
import { withStyles } from '@material-ui/core/styles'; import { withStyles } from '@material-ui/core/styles';
import LanguageIcon from '@material-ui/icons/Language'; import LanguageIcon from '@material-ui/icons/Language';
import Flags from 'country-flag-icons/react/3x2'; import i18next from 'i18next';
import i18next, { TFunctionKeys } from 'i18next';
import React, { useCallback, useContext, useState, useMemo } from 'react'; import React, { useCallback, useContext, useState, useMemo } from 'react';
import { useTranslation } from 'react-i18next'; import { useTranslation } from 'react-i18next';
import { Language } from 'src/i18n/enabledLanguages';
import { AutoComplete } from 'verdaccio-ui/components/AutoComplete/AutoCompleteV2'; import { AutoComplete } from 'verdaccio-ui/components/AutoComplete/AutoCompleteV2';
import MenuItem from 'verdaccio-ui/components/MenuItem'; import MenuItem from 'verdaccio-ui/components/MenuItem';
import { Theme } from 'verdaccio-ui/design-tokens/theme'; import { Theme } from 'verdaccio-ui/design-tokens/theme';
import ThemeContext from 'verdaccio-ui/design-tokens/ThemeContext'; import ThemeContext from 'verdaccio-ui/design-tokens/ThemeContext';
const lngDetails: Record<Language, { translation: TFunctionKeys; icon: React.ReactElement }> = { import { Language, listLanguages } from '../../i18n/enabledLanguages';
'fr-FR': {
translation: 'lng.french', const listConverted = listLanguages.reduce((prev, item) => {
icon: <Flags.FR />, prev[item.lng] = {
}, translation: item.menuKey,
'pt-BR': { icon: item.icon,
translation: 'lng.portuguese',
icon: <Flags.BR />,
},
'de-DE': {
translation: 'lng.german',
icon: <Flags.DE />,
},
'es-ES': {
translation: 'lng.spanish',
icon: <Flags.ES />,
},
'zh-CN': {
translation: 'lng.chinese',
icon: <Flags.CN />,
},
'ru-RU': {
translation: 'lng.russian',
icon: <Flags.RU />,
},
'tr-TR': {
translation: 'lng.turkish',
icon: <Flags.TR />,
},
'uk-UA': {
translation: 'lng.ukraine',
icon: <Flags.UA />,
},
'km-KH': {
translation: 'lng.khmer',
icon: <Flags.KH />,
},
'ja-JP': {
translation: 'lng.japanese',
icon: <Flags.JP />,
},
'en-US': {
translation: 'lng.english',
icon: <Flags.US />,
},
'cs-CZ': {
translation: 'lng.czech',
icon: <Flags.CZ />,
},
'zh-TW': {
translation: 'lng.chineseTraditional',
icon: <Flags.TW />,
},
}; };
return prev;
}, {});
const LanguageSwitch = () => { const LanguageSwitch = () => {
const themeContext = useContext(ThemeContext); const themeContext = useContext(ThemeContext);
@ -90,10 +43,10 @@ const LanguageSwitch = () => {
const getCurrentLngDetails = useCallback( const getCurrentLngDetails = useCallback(
(language: Language) => { (language: Language) => {
const { icon, translation } = lngDetails[language] || lngDetails['en-US']; const lng = listConverted[language] || listConverted['en-US'];
return { return {
icon, icon: <lng.icon />,
translation: t(translation), translation: t(lng.translation),
}; };
}, },
[t] [t]

View file

@ -37,7 +37,7 @@ interface Props<Option extends {}> extends Pick<TextFieldProps, 'variant'> {
inputStartAdornment?: React.ReactNode; inputStartAdornment?: React.ReactNode;
hasClearIcon?: boolean; hasClearIcon?: boolean;
className?: string; className?: string;
onClick?: (option: Option) => void; onClick?: (option: any) => void;
} }
const AutoComplete = <Option extends {}>({ const AutoComplete = <Option extends {}>({

View file

@ -1,11 +1,16 @@
import i18n from 'i18next'; import i18n from 'i18next';
import { initReactI18next } from 'react-i18next'; import { initReactI18next } from 'react-i18next';
import { DEFAULT_LANGUAGE, enabledLanguages } from './enabledLanguages'; import {
DEFAULT_LANGUAGE,
LanguageConfiguration,
listLanguages,
listLanguagesAsString,
} from './enabledLanguages';
const languages = enabledLanguages.reduce((acc, lng) => { const languages = listLanguages.reduce((acc, item: LanguageConfiguration) => {
acc[lng] = { acc[item.lng] = {
translation: require(`./translations/${lng}.json`), translation: require(`./translations/${item.lng}.json`),
}; };
return acc; return acc;
}, {}); }, {});
@ -19,7 +24,7 @@ i18n
// in case window.VEDACCIO_LANGUAGE is undefined,it will fall back to 'en-US' // in case window.VEDACCIO_LANGUAGE is undefined,it will fall back to 'en-US'
lng: window?.__VERDACCIO_BASENAME_UI_OPTIONS?.language || DEFAULT_LANGUAGE, lng: window?.__VERDACCIO_BASENAME_UI_OPTIONS?.language || DEFAULT_LANGUAGE,
fallbackLng: DEFAULT_LANGUAGE, fallbackLng: DEFAULT_LANGUAGE,
whitelist: [...enabledLanguages], whitelist: [...listLanguagesAsString],
load: 'currentOnly', load: 'currentOnly',
resources: languages, resources: languages,
debug: false, debug: false,

View file

@ -0,0 +1,170 @@
{
"copy-to-clipboard": "Copy to clipboard",
"author-anonymous": "Anonymous",
"author-unknown": "Unknown",
"action-bar-action": {
"visit-home-page": "Visit homepage",
"open-an-issue": "Open an issue",
"download-tarball": "Download tarball"
},
"dialog": {
"registry-info": {
"title": "Registry Info"
}
},
"header": {
"documentation": "Documentation",
"registry-info": "Registry Information",
"registry-info-link": "Learn more",
"registry-no-conf": "No configurations available",
"greetings": "Hi "
},
"search": {
"packages": "Search Packages"
},
"autoComplete": {
"loading": "Loading...",
"no-results-found": "No results found",
"clear": "Clear",
"expand": "Expand",
"collapse": "Collapse"
},
"tab": {
"uplinks": "Uplinks",
"versions": "Versions",
"dependencies": "Dependencies",
"readme": "Readme"
},
"uplinks": {
"title": "Uplinks",
"no-items": "{{name}} has no uplinks."
},
"versions": {
"current-tags": "Current Tags",
"version-history": "Version history",
"not-available": "Not available"
},
"package": {
"published-on": "Published on {{time}} •",
"version": "v{{version}}",
"visit-home-page": "Visit homepage",
"homepage": "Homepage",
"open-an-issue": "Open an issue",
"bugs": "Bugs",
"download": "Download {{what}}",
"the-tar-file": "the tar file",
"tarball": "Tarball"
},
"dependencies": {
"has-no-dependencies": "{{package}} has no dependencies.",
"dependency-block": "{{package}}@{{version}}"
},
"form": {
"username": "Username",
"password": "Password"
},
"form-placeholder": {
"username": "Your username",
"password": "Your strong password"
},
"form-validation": {
"required-field": "This field is required",
"required-min-length": "This field required the min length of {{length}}",
"unable-to-sign-in": "Unable to sign in",
"username-or-password-cant-be-empty": "Username or password can't be empty!"
},
"help": {
"title": "No Package Published Yet.",
"sub-title": "To publish your first package just:",
"first-step": "1. Login",
"first-step-command-line": "npm adduser --registry {{registryUrl}}",
"second-step": "2. Publish",
"second-step-command-line": "npm publish --registry {{registryUrl}}",
"third-step": "3. Refresh this page."
},
"sidebar": {
"detail": {
"latest-version": "Latest v{{version}}",
"version": "v{{version}}"
},
"installation": {
"title": "Installation",
"install-using-yarn": "Install using yarn",
"install-using-yarn-command": "yarn add {{packageName}}",
"install-using-npm": "Install using npm",
"install-using-npm-command": "npm install {{packageName}}",
"install-using-pnpm": "Install using pnpm",
"install-using-pnpm-command": "pnpm install {{packageName}}"
},
"repository": {
"title": "Repository"
},
"author": {
"title": "Author"
},
"distribution": {
"title": "Latest Distribution",
"license": "License",
"size": "Size",
"file-count": "file count"
},
"maintainers": {
"title": "Maintainers"
},
"contributors": {
"title": "Contributors"
},
"engines": {
"npm-version": "NPM Version",
"node-js": "NODE JS"
}
},
"footer": {
"powered-by": "Powered by",
"made-with-love-on": "Made with <0>♥</0> on"
},
"button": {
"close": "Close",
"cancel": "Cancel",
"login": "Login",
"logout": "Logout",
"go-to-the-home-page": "Go to the home page",
"learn-more": "Learn More",
"fund-this-package": "<0>Fund</0> this package"
},
"error": {
"unspecific": "Something went wrong.",
"404": {
"page-not-found": "404 - Page not found",
"sorry-we-could-not-find-it": "Sorry, we couldn't find it..."
},
"app-context-not-correct-used": "The app context was not used correctly",
"theme-context-not-correct-used": "The theme context was not used correctly",
"package-meta-is-required-at-detail-context": "packageMeta is required at DetailContext"
},
"lng": {
"english": "English",
"czech": "Czech",
"japanese": "Japanese",
"portuguese": "Portuguese",
"spanish": "Spanish",
"german": "German",
"chinese": "Chinese",
"chineseTraditional": "Chinese (Traditional)",
"french": "French",
"russian": "Russian",
"turkish": "Turkish",
"ukraine": "Ukraine",
"khmer": "Khmer"
},
"flag": {
"austria": "Austria",
"brazil": "Brazil",
"spain": "Spain",
"nicaragua": "Nicaragua",
"india": "India",
"china": "China",
"germany": "Germany",
"taiwan": "Taiwan"
}
}

View file

@ -1,19 +1,35 @@
import Flags from 'country-flag-icons/react/3x2';
import React from 'react';
export const DEFAULT_LANGUAGE = 'en-US'; export const DEFAULT_LANGUAGE = 'en-US';
export const enabledLanguages = [ export type LanguageConfiguration = {
DEFAULT_LANGUAGE, lng: string;
'cs-CZ', menuKey: string;
'pt-BR', icon: React.ReactElement;
'es-ES', };
'de-DE',
'fr-FR', export const listLanguages: LanguageConfiguration[] = [
'zh-CN', { lng: DEFAULT_LANGUAGE, icon: Flags.US, menuKey: 'lng.english' },
'ja-JP', { lng: 'cs-CZ', icon: Flags.CZ, menuKey: 'lng.czech' },
'ru-RU', { lng: 'pt-BR', icon: Flags.BR, menuKey: 'lng.portuguese' },
'tr-TR', { lng: 'es-ES', icon: Flags.ES, menuKey: 'lng.spanish' },
'uk-UA', { lng: 'de-DE', icon: Flags.DE, menuKey: 'lng.german' },
'km-KH', { lng: 'fr-FR', icon: Flags.FR, menuKey: 'lng.french' },
'zh-TW', { lng: 'zh-CN', icon: Flags.CN, menuKey: 'lng.chinese' },
{ lng: 'ja-JP', icon: Flags.JP, menuKey: 'lng.japanese' },
{ lng: 'ru-RU', icon: Flags.RU, menuKey: 'lng.russian' },
{ lng: 'tr-TR', icon: Flags.TR, menuKey: 'lng.turkish' },
{ lng: 'uk-UA', icon: Flags.UA, menuKey: 'lng.ukraine' },
{ lng: 'km-KH', icon: Flags.KH, menuKey: 'lng.khme' },
{ lng: 'zh-TW', icon: Flags.TW, menuKey: 'lng.chineseTraditional' },
]; ];
export type Language = keyof typeof enabledLanguages; const languages = listLanguages.reduce((acc, item: LanguageConfiguration) => {
acc.push(item.lng);
return acc;
}, [] as string[]);
export const listLanguagesAsString = languages;
export type Language = keyof typeof listLanguagesAsString;