资讯专栏INFORMATION COLUMN

react 国际化了解一下

CrazyCodes / 769人阅读

摘要:先睹为快先看一下最后的成果来一发控制台中对应中的信息开始原理原理其实很简单字符串替换。拉取远程的国际化文件到本地,再根据语言做一个映射就可以了。

背景

楼主最近新接了一个项目,从0开始做,需要做多语言的国际化,今天搞了一下,基本达到了想要的效果, 在这里简单分享下:

一些探索

也说不上是探索吧,就Google了一波, GitHub 上找了一个比较成熟的库 react-i18next, 写了一些代码,现将过程分享一下, 附带详细代码,手把手教你实现国际化。

先睹为快

先看一下最后的成果:

</>复制代码

  1. // ...
  2. import i18n from "@src/i18n";
  3. // xxx component
  4. console.log("i18n来一发:", i18n.t("INVALID_ORDER"));
  5. render() {
  6. // ...
  7. }

控制台中:

对应json 中的信息:

开始 原理

原理其实很简单: 字符串替换。

拉取远程的国际化json文件到本地,再根据语言做一个映射就可以了。

废话不多说, 来看代码吧。

先简单看一下目录结构:

先看一下 config 里面的 相关代码:

env.js:

</>复制代码

  1. "use strict";
  2. const fs = require("fs");
  3. const path = require("path");
  4. const paths = require("./paths");
  5. const languages = require("./languages");
  6. // Make sure that including paths.js after env.js will read .env variables.
  7. delete require.cache[require.resolve("./paths")];
  8. const NODE_ENV = process.env.NODE_ENV;
  9. if (!NODE_ENV) {
  10. throw new Error(
  11. "The NODE_ENV environment variable is required but was not specified."
  12. );
  13. }
  14. // https://github.com/bkeepers/dotenv#what-other-env-files-can-i-use
  15. var dotenvFiles = [
  16. `${paths.dotenv}.${NODE_ENV}.local`,
  17. `${paths.dotenv}.${NODE_ENV}`,
  18. // Don"t include `.env.local` for `test` environment
  19. // since normally you expect tests to produce the same
  20. // results for everyone
  21. NODE_ENV !== "test" && `${paths.dotenv}.local`,
  22. paths.dotenv,
  23. ].filter(Boolean);
  24. // Load environment variables from .env* files. Suppress warnings using silent
  25. // if this file is missing. dotenv will never modify any environment variables
  26. // that have already been set. Variable expansion is supported in .env files.
  27. // https://github.com/motdotla/dotenv
  28. // https://github.com/motdotla/dotenv-expand
  29. dotenvFiles.forEach(dotenvFile => {
  30. if (fs.existsSync(dotenvFile)) {
  31. require("dotenv-expand")(
  32. require("dotenv").config({
  33. path: dotenvFile,
  34. })
  35. );
  36. }
  37. });
  38. // We support resolving modules according to `NODE_PATH`.
  39. // This lets you use absolute paths in imports inside large monorepos:
  40. // https://github.com/facebookincubator/create-react-app/issues/253.
  41. // It works similar to `NODE_PATH` in Node itself:
  42. // https://nodejs.org/api/modules.html#modules_loading_from_the_global_folders
  43. // Note that unlike in Node, only *relative* paths from `NODE_PATH` are honored.
  44. // Otherwise, we risk importing Node.js core modules into an app instead of Webpack shims.
  45. // https://github.com/facebookincubator/create-react-app/issues/1023#issuecomment-265344421
  46. // We also resolve them to make sure all tools using them work consistently.
  47. const appDirectory = fs.realpathSync(process.cwd());
  48. process.env.NODE_PATH = (process.env.NODE_PATH || "")
  49. .split(path.delimiter)
  50. .filter(folder => folder && !path.isAbsolute(folder))
  51. .map(folder => path.resolve(appDirectory, folder))
  52. .join(path.delimiter);
  53. // Grab NODE_ENV and REACT_APP_* environment variables and prepare them to be
  54. // injected into the application via DefinePlugin in Webpack configuration.
  55. const REACT_APP = /^REACT_APP_/i;
  56. function getClientEnvironment(publicUrl) {
  57. const raw = Object.keys(process.env)
  58. .filter(key => REACT_APP.test(key))
  59. .reduce(
  60. (env, key) => {
  61. env[key] = process.env[key];
  62. return env;
  63. },
  64. {
  65. // Useful for determining whether we’re running in production mode.
  66. // Most importantly, it switches React into the correct mode.
  67. NODE_ENV: process.env.NODE_ENV || "development",
  68. // Useful for resolving the correct path to static assets in `public`.
  69. // For example, .
  70. // This should only be used as an escape hatch. Normally you would put
  71. // images into the `src` and `import` them in code to get their paths.
  72. PUBLIC_URL: publicUrl,
  73. LANGUAGE: {
  74. resources: languages.resources,
  75. defaultLng: languages.defaultLng
  76. },
  77. COUNTRY: process.env.COUNTRY
  78. }
  79. );
  80. // Stringify all values so we can feed into Webpack DefinePlugin
  81. const stringified = {
  82. "process.env": Object.keys(raw).reduce((env, key) => {
  83. env[key] = JSON.stringify(raw[key]);
  84. return env;
  85. }, {}),
  86. };
  87. return { raw, stringified };
  88. }
  89. module.exports = getClientEnvironment;

主要看lannguage 相关的代码就好了, 其他的都create-react-app 的相关配置, 不用管。

再看下 language.js 里面的逻辑:

</>复制代码

  1. const path = require("path");
  2. const paths = require("./paths");
  3. const localesHash = require("../i18n/localesHash");
  4. const resourcesHash = require("../i18n/resourcesHash");
  5. const COUNTRY = process.env.COUNTRY || "sg";
  6. const country = (COUNTRY).toUpperCase();
  7. const defaultLng = localesHash[country][0];
  8. const langs = [
  9. "en",
  10. "id"
  11. ];
  12. const prefixLangs = [];
  13. const entries = {};
  14. for (let i = 0, len = langs.length; i < len; i++) {
  15. const prefixLang = `dict_${langs[i]}`
  16. prefixLangs.push(prefixLang)
  17. entries[prefixLang] = path.resolve(paths.appSrc, `../i18n/locales/${langs[i]}.json`)
  18. }
  19. const resources = {
  20. [defaultLng]: {
  21. common: resourcesHash[defaultLng]
  22. }
  23. }
  24. exports.resources = resources;
  25. exports.defaultLng = defaultLng;

逻辑也比较简单, 根据语言列表把对应的json 内容加进来。 作为示例,这里我设置的是 英文 和 印尼语。

下面看 i18n 文件里面的内容:

locales 里面放的是语言的json 文件, 内容大概是:

</>复制代码

  1. {
  2. "msg_Created": "Pesanan telah terbuat"
  3. // ...
  4. }

localesHash.js:

</>复制代码

  1. module.exports = {
  2. SG: ["en"],
  3. ID: ["id"]
  4. }

resourcesHash.js:

</>复制代码

  1. module.exports = {
  2. "en": require("./locales/en.json"),
  3. "id": require("./locales/id.json")
  4. }

index.js

</>复制代码

  1. const path = require("path")
  2. const fs = require("fs")
  3. const fetch = require("isomorphic-fetch")
  4. const localesHash = require("./localesHash")
  5. const argv = process.argv.slice(2)
  6. const country = (argv[0] || "").toUpperCase()
  7. const i18nServerURI = locale => {
  8. const keywords = {
  9. "en": "en",
  10. "id": "id"
  11. }
  12. const keyword = keywords[locale]
  13. return keyword === "en"
  14. ? "your/transify/website/json/download"
  15. : `your/transify/website/${keyword}/json/download`
  16. }
  17. const fetchKeys = async (locale) => {
  18. const uri = i18nServerURI(locale)
  19. console.log(`Downloading ${locale} keys...
  20. ${uri}`)
  21. const respones = await fetch(uri)
  22. const keys = await respones.json()
  23. return keys
  24. }
  25. const access = async (filepath) => {
  26. return new Promise((resolve, reject) => {
  27. fs.access(filepath, (err) => {
  28. if (err) {
  29. if (err.code === "EXIST") {
  30. resolve(true)
  31. }
  32. resolve(false)
  33. }
  34. resolve(true)
  35. })
  36. })
  37. }
  38. const run = async () => {
  39. const locales = localesHash[country] || Object
  40. .values(localesHash)
  41. .reduce(
  42. (previous, current) =>
  43. previous.concat(current), []
  44. )
  45. if (locales === undefined) {
  46. console.error("This country is not in service.")
  47. return
  48. }
  49. for (const locale of locales) {
  50. const keys = await fetchKeys(locale)
  51. const data = JSON.stringify(keys, null, 2)
  52. const directoryPath = path.resolve(__dirname, "locales")
  53. if (!fs.existsSync(directoryPath)) {
  54. fs.mkdirSync(directoryPath)
  55. }
  56. const filepath = path.resolve(__dirname, `locales/${locale}.json`)
  57. const isExist = await access(filepath)
  58. const operation = isExist ? "update" : "create"
  59. console.log(operation)
  60. fs.writeFileSync(filepath, `${data}
  61. `)
  62. console.log(`${operation}
  63. ${filepath}`)
  64. }
  65. }
  66. run();

再看下src 中的配置:

i18nn.js

</>复制代码

  1. import i18next from "i18next"
  2. import { firstLetterUpper } from "./common/helpers/util";
  3. const env = process.env;
  4. let LANGUAGE = process.env.LANGUAGE;
  5. LANGUAGE = typeof LANGUAGE === "string" ? JSON.parse(LANGUAGE) : LANGUAGE
  6. const { defaultLng, resources } = LANGUAGE
  7. i18next
  8. .init({
  9. lng: defaultLng,
  10. fallbackLng: defaultLng,
  11. defaultNS: "common",
  12. keySeparator: false,
  13. debug: env.NODE_ENV === "development",
  14. resources,
  15. interpolation: {
  16. escapeValue: false
  17. },
  18. react: {
  19. wait: false,
  20. bindI18n: "languageChanged loaded",
  21. bindStore: "added removed",
  22. nsMode: "default"
  23. }
  24. })
  25. function isMatch(str, substr) {
  26. return str.indexOf(substr) > -1 || str.toLowerCase().indexOf(substr) > -1
  27. }
  28. export const changeLanguage = (locale) => {
  29. i18next.changeLanguage(locale)
  30. }
  31. // Uppercase the first letter of every word. abcd => Abcd or abcd efg => Abcd Efg
  32. export const tUpper = (str, allWords = true) => {
  33. return firstLetterUpper(i18next.t(str), allWords)
  34. }
  35. // Uppercase all letters. abcd => ABCD
  36. export const tUpperCase = (str) => {
  37. return i18next.t(str).toUpperCase()
  38. }
  39. export const loadResource = lng => {
  40. let p;
  41. return new Promise((resolve, reject) => {
  42. if (isMatch(defaultLng, lng)) resolve()
  43. switch (lng) {
  44. case "id":
  45. p = import("../i18n/locales/id.json")
  46. break
  47. default:
  48. p = import("../i18n/locales/en.json")
  49. }
  50. p.then(data => {
  51. i18next.addResourceBundle(lng, "common", data)
  52. changeLanguage(lng)
  53. })
  54. .then(resolve)
  55. .catch(reject)
  56. })
  57. }
  58. export default i18next

</>复制代码

  1. // firstLetterUpper
  2. export const firstLetterUpper = (str, allWords = true) => {
  3. let tmp = str.replace(/^(.)/g, $1 => $1.toUpperCase())
  4. if (allWords) {
  5. tmp = tmp.replace(/s(.)/g, $1 => $1.toUpperCase())
  6. }
  7. return tmp;
  8. }

这些准备工作做好后, 还需要把i18n 注入到app中:

index.js:

</>复制代码

  1. import React from "react";
  2. import { render } from "react-dom";
  3. import { Provider } from "react-redux";
  4. import rootReducer from "./common/redux/reducers";
  5. import { configureStore } from "./common/redux/store";
  6. import { Router } from "react-router-dom";
  7. import createBrowserHistory from "history/createBrowserHistory";
  8. import { I18nextProvider } from "react-i18next";
  9. import i18n from "./i18n";
  10. import "./common/styles/index.less";
  11. import App from "./App";
  12. export const history = createBrowserHistory();
  13. const ROOT = document.getElementById("root");
  14. render(
  15. ,
  16. ROOT
  17. );
如何使用

加入上面的代码后, 控制台会有一些log 信息, 表示语言已经加载好了。

在具体的业务组件中,使用方法是:

</>复制代码

  1. // ...
  2. import i18n from "@src/i18n";
  3. console.log("哈哈哈哈哈i18n来一发:", i18n.t("INVALID_ORDER"));

控制台中:

对应json 中的信息:

后面你就可以愉快的加各种词条了。

Tips

我们在src 中的文件中引入了src 目录外的文件, 这是create-react-app 做的限制, 编译会报错, 把它去掉就好了:

关于多语言的设置

代码示例:

</>复制代码

  1. i18next.init({
  2. lng: "en",
  3. });

初始化的时候可以设置默认语言, 如需要切换系统语言, 可以调用 i18n 提供的方法:

</>复制代码

  1. import { changeLanguage } from "@src/i18n";
  2. // ...
  3. // 设置为印尼语
  4. changeLanguage("id");

为了保存语言设置, 可以把 language 保存在 localStorage 中, 使用的时候直接从 storage 里取。

结语

这里作为例, 就是把语言的json 文件下载下来放到locales 目录里, 如果想实时拉取,要保证文件下载完之后再render app.

类似:

</>复制代码

  1. loadResource(getLocale())
  2. .then(() => {
  3. import("./app.js")
  4. })


当然你也可以免了这一步,直接下载好放到工程里来。

大概就是这样,以上就是实现国际化的全部代码,希望对大家有所帮助。

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/113950.html

相关文章

  • react 际化了解一下

    摘要:先睹为快先看一下最后的成果来一发控制台中对应中的信息开始原理原理其实很简单字符串替换。拉取远程的国际化文件到本地,再根据语言做一个映射就可以了。 背景 楼主最近新接了一个项目,从0开始做,需要做多语言的国际化,今天搞了一下,基本达到了想要的效果, 在这里简单分享下: 一些探索 也说不上是探索吧,就Google了一波, GitHub 上找了一个比较成熟的库 react-i18next,...

    魏明 评论0 收藏0
  • React项目际化(antd)多语言开发

    摘要:本国际化方案仅针对技术栈,且不会涉及服务端国际化内容。引入多语言环境的数据虽然我只用到了文本翻译的功能,以为就不需要加载这些数据,但后来发现这是必须的步骤。 前言 最近新接了一个项目,从0开始做,需要做多语言的国际化,今天搞了一下,基本达到了想要的效果, 在这里简单分享下: showImg(https://segmentfault.com/img/bVbuiJB); 背景国际化方案国际...

    tracymac7 评论0 收藏0
  • React项目际化(antd)多语言开发

    摘要:本国际化方案仅针对技术栈,且不会涉及服务端国际化内容。引入多语言环境的数据虽然我只用到了文本翻译的功能,以为就不需要加载这些数据,但后来发现这是必须的步骤。 前言 最近新接了一个项目,从0开始做,需要做多语言的国际化,今天搞了一下,基本达到了想要的效果, 在这里简单分享下: showImg(https://segmentfault.com/img/bVbuiJB); 背景国际化方案国际...

    wushuiyong 评论0 收藏0
  • 程序员到底要学什么?

    摘要:程序员到底要学什么程序员到底要学什么或者说,程序员到底要学多少东西呢这个问题问到你了吗今天就来简单聊一聊程序员的学习之路。程序员的种类很多,这里只讲前端工程师和后端工程师,因为自己也就接触到这两个层面。 ...

    mo0n1andin 评论0 收藏0

发表评论

0条评论

最新活动
阅读需要支付1元查看
<