资讯专栏INFORMATION COLUMN

在 React 项目中使用 React-intl 实现多语言支持

Mr_zhang / 1804人阅读

摘要:最近在项目中添加了语言国际化的功能。项目地址是雅虎的语言国际化开源项目的一部分,通过其提供的组件和可以与绑定。你当然可以使用的方式引用,但是,这样有前途么创建配置文件这里,我们将文件命名为和,代表中文和美式英语的配置包。

最近在项目中添加了语言国际化的功能。

语言国际化,也有人说成是语言本地化,其实就是为Web App添加多语言,我们的项目当前包含了中文版和英文版,按理来说『逐字替换』也不是多大事儿,但是,这么Low的做法,有钱途吗?

一开始的时候,我考虑的是传统的为整个项目添加config文件,根据不同的语言和地区,加载不同的config文件,就能够达到界面语言切换的目的。当然,也正是因为这个想法太过于幼稚,所以才被称为『一开始』的想法。语言的国际化不仅仅是将界面上的UI文字翻译成另一种语言,还包括了日期&时间显示,数字显示(英文环境下每隔3位一个逗号:1,000),量词的显示(一个苹果是apple,两个苹果就应该是apples),甚至还有一个字符串中间插了一个变量的情况("今天午饭吃了{count}个鸡腿")...

所以这并不只是一个简单的字符替换问题,并且,我们要很方便的导出一个目录,放到word或者page当中,给到其他同事对照着进行翻译工作,这个非常重要!!难道你要让产品经理直接在代码里改么?或者你想一个一个搜索替换?不考虑清楚就干的话,相信我,You"ll pay for this。

作为一个有追求的代码家,你肯定不希望在index.html当中增加一行

React-intl
项目地址:https://github.com/yahoo/reac...

React-intl是雅虎的语言国际化开源项目FormatJS的一部分,通过其提供的组件和API可以与ReactJS绑定。上面这句话援引了官方文档的说辞,主要表达的是,这是一个很屌的开源项目,有大团队支持,使用量也很大,不会太坑爹,你们放心用。虽然雅虎都快被收购了。

React-intl提供了两种使用方法,一种是引用React组建,另一种是直接调取API,官方更加推荐在React项目中使用前者,只有在无法使用React组件的地方,才应该调用框架提供的API,事实上,我在项目的过程中真的遇到了无法使用组件的情况,这个我会另外写一篇文章来描述。

React-intl提供的React组件有如下几种:

包裹在需要语言国际化的组建的最外层,为包含在其中的所有组建提供包含id和字符串的键值对。(如:"homepage.title":"Hommily";

日期时间

a. 用于格式化日期,能够将一个时间戳格式化成不同语言中的日期格式。

传入时间戳作为参数:

输出结果:

4/5/2016

b. 用于格式化时间,效果与相似。

传入时间戳作为参数:

输出结果:

1:09 AM

c. 通过这个组件可以显示传入组件的某个时间戳和当前时间的关系,比如 “ 10 minutes ago"。

传入时间戳作为参数:

输出结果:

now

10秒之后的输出结果:

10 seconds ago

1分钟之后的输出结果:

1 minute ago
数字量词

a. 这个组件最主要的用途是用来给一串数字标逗号,比如10000这个数字,在中文的语言环境中应该是1,0000,是每隔4位加一个逗号,而在英语的环境中是10,000,每隔3位加一个逗号。

传入数字作为参数:

输出结果:

1,000

b. 这个组件可用于格式化量词,在中文的语境中,其实不太会用得到,比如我们说一个鸡腿,那么量词就是‘个’,我们说两个鸡腿,量词还是‘个’,不会发生变化。但是在英文的语言环境中,描述一个苹果的时候,量词是apple,当苹果数量为两个时,就会变成apples,这个组件的作用就在于此。

传入组件的参数中,value为数量,其他的为不同数量时对应的量词,在下面的例子中,一个的时候量词为message,两个的时候量词为messages。实际上可以传入组件的量词包括 zero, one, two, few, many, other 已经涵盖了所有的情况。


传入组件的量词参数可以是一个字符串,也可以是一个组件,我们可以选择传入组件,就可以实现量词的不同语言的切换。

输出结果:

messages
字符串的格式化

a. 这个组件用于格式化字符串,是所有的组件中使用频率最高的组件,因为基本上,UI上面的每一个字符串都应该用这个组件替代。这个组件的功能丰富,除了可以根据配置输出不同语言的简单字符串之外,还可以输出包含动态变化的参数的复杂字符串,具体的用法在后面的例子中会慢慢叙述。

比如我们在locale配置文件中写了如下内容:

const app = {
    greeting:"Hello Howard!",
}

export default app;

使用这个组件的时候,我们这么写:

id指代的是这个字符串在locale配置文件中的属性名,description指的是对于这个位置替代的字符串的描述,便于维护代码,不写的话也不会影响输出的结果,当在locale配置文件中没有找到这个id的时候,输出的结果就是defaultMessage的值。

输出的结果:

Hello, Howard!

b. 这个组件的用法和完全相同,唯一的不同就是输出的字符串可以包含HTML标签,但是官方不太推荐使用这个方法,如果可以想办法用的话,就不应该使用这个组件,我揣测应该是性能方面不如,这个组件的用法我就不举例了。

Well,到此为止,已经把React-intl提供的所有组件介绍完了,下面就给大家介绍一下具体如何去使用吧。

React-intl 使用步骤

(本文例子运行在OSX环境,Window操作方法的终端在安装的时候要注意用管理员身份运行)

1. 安装React-intl

假设你已经在你的系统中安装了node.js和npm,如果你还不知道这两个是什么东西,请自行百度,对,在百度都能找到答案。

打开终端,进入项目根目录,输入以下指令安装React-intl:

npm install react-intl -save

注意:为了兼容Safari各个版本,需要同时安装 intl,intl在大部分的『现代』浏览器中是默认自带的,但是Safari和IE11以下的版本就没有了,这里需要留个心眼。

安装intl需要在终端中输入以下指令:

npm install intl --save

这里还有一个注意:由于React-intl的每一个组件的使用方法大同小异,和ReactJS的语法完全一致,所以我就仅仅描述如何使用这个组件的用法,借此抛砖引玉,相信看完之后已经足够帮助你迅速的去使用这个开源框架了。

2. 引用React-intl
import { FormattedMessage } from "react-intl"; 

由于我使用的是ES6 的语法,所以是支持直接引用组件的。你当然可以使用ES5的方式引用,但是,这样有前途么?

require ReactIntl from "react-intl";
3. 创建locale配置文件

这里,我们将文件命名为zh_CN.js和en_US.js,代表中文和美式英语的配置包。

在zh_CN.js编写如下代码:

const zh_CN = {
            hello:"你好,方浩!",
            superHello:"你好,{ someone } !"
        }
export default zh_CN;    

在en_US.js编写如下代码:

const en_US = {
            hello:"Hello, Howard!",
            superHello:"Hello, { someone } !"
        }    
export default en_US;

于是,我们就创建好了locale文件,但是,在实际的项目中配置文件不会这么简单,您可能需要根据业务需求按照不同的页面或者不同的功能块创建不同的文件树,然后用模块化的方法将不同的配置文件进行组织,已达成你的目标,这里我也就没能力逼逼太多了。

你可能想问,{ someone } 是什么鬼?其实悟性高一些的话就应该已经猜到,这个应该就是前面提到过的在字符串中插入动态参数的用法,事实上也是这样的。

4. 使用

使用 组件包裹住需要您需要进行语言国际化的组件,用法和React-redux的差不多,当 包裹住某个组件的时候,这个组件本身和组件内部包含的子组件就可以获得所有React-intl提供的接口以及在 中引入的locale配置文件的内容。

import React from "react";
import { render } from "react-dom";
//引入locale配置文件,具体路径根据实际情况填写
import zh_CN from "./zh_CN";
import en-US from "./en-US";
//如果浏览器没有自带intl,则需要在使用npm安装intl之后添加如下代码
import intl from "intl";
addLocaleDate([...en,...zh]);

...
...

render(    
        
            
        ,    
        document.getElementById("container")
);

需要传递两个参数:

locale是传递需要国际化的语言的缩写,通过这个参数可以确定格式化日期,数字,量词的时候按照哪一种语言的规则,这个是规则是intl提供的,一般浏览器会内置这个库,但是在Safari和IE11之前需要自己安装,安装的方法前面已经提及,请自己翻阅。

messages是用于传递刚刚我们在第3步中定义的配置文件的,从示例代码中我们可以看出,首先我们使用Import语句引入了配置文件,然后将配置文件的内容传递给了messages这个参数,此时组件中的所有组件都可以拿到配置文件中的内容了。

那个跳起来的同学,请先坐下,我猜你是想问,是不是每次都要手动修改这两个参数以适配不同语言呢?

其实不然,我们完全可以按照下面的做法自动识别当前浏览器的语言:

chooseLocale(){
    switch(navigator.language.split("_")[0]){
        case "en":
            return "en_US";
            break;
        case "zh":
            return "zh_CN";
            break;
        ...
        ...
        ...
        default:
            return "en_US";
            break;
    }
}
render(    
        
            
        ,    
        document.getElementById("container")
);

您还需要知道的是,是可以嵌套使用的,也就是说,在一个内部还可以有N个,这个功能的实际意义就是可以在英文网站中嵌套一个中文的或者德语的或者法语的板块,应用起来会更加灵活一些。

5. 使用

前面的几个步骤其实都是为了这个步骤做铺垫的,在添加了之后,我们就可以在其包裹的包含的所有组件中获取到配置文件的信息,传入组件的id参数也能其在配置文件中对应的字符串了。

使用的方法如下:

在Js执行的时候,组件就会找到配置文件中,‘hello"键名对应的字符串"Hello, Howard!".

输出的结果为:

Hello, Howard!

那么如何输出含有动态参数的字符串呢?比如Hello,Johnson!,如果我要问候的对象是一个变量呢?

那就这么写呗。。

以上的例子中,赋给someone的就是一个变量(假设这个变量是通过参数传进这个组件的),注意,如果是这样的话,那么locale配置文件中就要这么写。

 superHello:"你好,{ someone } !"

前面其实提过了,怕你忘了...我已经悄无声息的把id换成了superHello。

更牛逼的是,这个someone还可以包含HTML标签!

this.props.name,
    }
    />

输出结果:

Hello, Howard!

于是,这个名字就被加粗了。

眼尖的同学又要跳起来了,“webFunc,为什么所有的输出都带一个标签,我就不能换成别的么?”

不要着急,我正要说这个,对于这个问题,官方的文档是这么说的。

By default  will render the formatted string into
a . If you need to customize rendering, you can either wrap it
with another React element (recommended), specify a different tagName
(e.g., "div"), or pass a function as the child.

翻译过来就是,默认的是会包裹在标签中的,如果想要让输出的字符串包裹在其他标签中的话,比如你想包裹在

中,你就把组件包含在一对
中间,这是一种官方更加推荐的做法。

Well, that"s stupid...

或者你可以给传入一个tagName的参数。比如:

就会输出:

Hello, Howard!

比较奇葩的是,也是我揣测作者不推荐使用这种方法的原因是...只要你高兴,tagName可以传入任意字符串,比如 shit:

就会输出:

Hello, Howard!

Yes, shit happens.

看到这里,你应该已经会使用React-intl对你的项目进行语言国际化了,没有进一步描述的地方,请自行查阅官方文档(项目地址:https://github.com/yahoo/reac...,或者给我留言,虽然我不一定会及时回复。

--写在后面:

语言国际化应该是一个比较经常遇到的需求,但是我在完成项目的过程中,看到的中文的资料却相当少,虽然这不是一篇非常牛叉的技术文章,但是可能会帮到很多人,如若如此,也便满足了。
——方浩(webFunc)

对了,你可以关注一下我的微信公众号:webcoding

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

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

相关文章

  • React项目国际化(antd)多语开发

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

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

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

    wushuiyong 评论0 收藏0
  • [ 一起学React系列 -- 10 ] i18n

    摘要:假如有这么一段句子这件衣服是人民币如果我们想将一个数字以人民币的形式写进去的话可以这么做最终显示结果是这件衣服是人民币其实它做了两件事一个是加符号,另一个是加分隔符。同时表示人民币,表示美元。 今天来介绍一个非常international的东西。 i18n国际化(internationalization)的简称。之所以叫i18n,是因为字母i和n之间有18个字母,所以才叫i18n。不...

    biaoxiaoduan 评论0 收藏0
  • 初探react技术栈(一)

    摘要:相信用的同学也不少找到函数在其中中添加启用编译。。。react 最近已经开始使用react技术栈了,从头开始搭建项目,有必要的记录一下配置的过程以及项目分层的思路,这次后台项目采用的主要采用react-create-app脚手架以及Ant DesignUI 以及多语言react-intl create-react-app 这是官方维护的脚手架应用 我们一般也采用这个 $ npm or c...

    刘玉平 评论0 收藏0

发表评论

0条评论

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