React 做国际化,我推荐使用 React-intl , 这个库提供了 React 组件和Api两种方式来格式化日期,数字和字符串等。知道这个库了,那让我们开始使用它
效果预览
React Intl 国际化步骤
- 创建国际化资源文件
- 根据语言获取国际化资源
- 引入 react-intl 的 local data
- 创建 LocaleProvider 国际化上下文组件
- 创建 react-intl 国际化上下文组件
- 使用 react-intl’s components & apis,进行国际化开发
1. 创建国际化资源文件
目前我们管理资源文件的方式是在 src/locales 文件夹下:
1 | . |
*.messages.js 是我们的资源文件(这里我们采用了 js 格式,你也可以使用 json 等等),返回的是一个对象,key 为我们翻译用的 id,value 为具体语言的翻译,内容是:
1 | export default { |
en-US.js 文件封装了 messages、locale 等国际化上下文组件需要的内容:
1 | import appLocaleData from 'react-intl/locale-data/en'; |
有了这些资源文件以及相关的封装之后,我们就可以在 LocaleProvider
和 InltProvider
中使用了。
2. 根据语言加载国际化资源
上一步我们创建了不同语言版本的国际化资源文件,我们还需要一个函数能够根据语言,加载对应的资源文件:
1 | /** |
3. 引入 react-intl 的 local data
1 | import { addLocaleData } from 'react-intl'; |
react-intl 在做国际化的时候需要一些特有的 local data,主要是进行相对时间翻译时,比如昨天、今天、明天、几分钟前、几个月前之类的。
我们通过 addLocaleData
这个方法加载相关内容,大家可以根据实际情况加载需要的 locale-data。
4. 创建 LocaleProvider 国际化上下文组件
为了组件能够国际化资源信息,我们需要一个 LocaleProvider 组件,用它来提供国际化的上下文,具体用法:
1 | export default class LocaleProvider extends React.Component { |
5. 创建 react-intl 国际化上下文组件
为了能够使用 react-intl 进行国际化,跟 redux 这些框架一样,我们需要一个 Provider Component,用它来提供国际化的上下文,具体用法:
1 | ... |
1 | LocaleProvider 有三个配置参数: |
在定义好 IntlProvider
之后,我们就可以在页面使用它提供的 api 或者组件来进行国际化了。
6. 使用 react-intl’s components & apis
react-intl 提供了丰富的组件和 api 来完成页面部分的国际化。
字符串的格式化
a. <FormattedMessage />
这个组件用于格式化字符串,是所有的组件中使用频率最高的组件。除了可以根据配置输出不同语言的简单字符串之外,还可以输出包含动态变化的参数的复杂字符串,具体的用法在后面的例子中会慢慢叙述。
比如我们在 *.message.js 配置文件中写了如下内容:1
2
3export default {
'page.localeProvider.react': '{ name }, a JavaScript library for building user interfaces.',
};
使用这个组件的时候,我们这么写:1
2
3
4
5
6
7
8
9<FormattedMessage
tagName="p"
id="page.localeProvider.react"
values={{
name: 'React',
}}
defaultMessage="{name} 是一个用于构建用户界面的 JAVASCRIPT 库。"
description="{name} 是什么?"
/>
- id 指代的是这个字符串在配置文件中的属性名
- description 指的是对于这个位置替代的字符串的描述,便于维护代码,不写的话也不会影响输出的结果
- defaultMessage 当在locale配置文件中没有找到这个id的时候,输出的默认值
- tagName 实际生成的标签,默认是
span
- values 动态参数. 格式为对象
输出的结果:1
<p>React, a JavaScript library for building user interfaces.</p>
b. <FormattedHTMLMessage />
这个组件的用法和
日期时间
a. <FormattedDate />
用于格式化日期,能够将一个时间戳格式化成不同语言中的日期格式。
传入时间戳作为参数:1
2
3<FormattedDate
value={new Date(1459832991883)}
/>
输出结果:1
2
3
4
5<!-- 英文 -->
<span>4/5/2016</span>
<!-- 中文 -->
<span>2016/5/4</span>
b. <FormattedTime>
用于格式化时间,效果与<FormattedDate />
相似。
传入时间戳作为参数:1
2
3<FormattedTime
value={new Date(1459832991883)}
/>
输出结果:1
2
3
4
5<!-- 英文 -->
<span>1:09 AM</span>
<!-- 中文 -->
<span>上午1:09</span>
c. <FormattedRelative />
通过这个组件可以显示传入组件的某个时间戳和当前时间的关系,比如“10 minutes ago”。
传入时间戳作为参数:1
2
3<FormattedRelative
value={Date.now()}
/>
输出结果:1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17<!-- 英文 =>> 运行时的输出结果: -->
<span>now</span>
<!-- 英文 =>> 10秒之后的输出结果: -->
<span>10 seconds ago</span>
<!-- 英文 =>> 1分钟之后的输出结果: -->
<span>1 minute ago</span>
<!-- 中文 =>> 运行时的输出结果: -->
<span>现在</span>
<!-- 中文 =>> 10秒之后的输出结果: -->
<span>10秒前</span>
<!-- 中文 =>> 1分钟之后的输出结果: -->
<span>1分钟前</span>
数字量词
a. <FormattedPlural />
这个组件可用于格式化量词,在中文的语境中,其实不太会用得到,比如我们说一个鸡腿,那么量词就是‘个’,我们说两个鸡腿,量词还是‘个’,不会发生变化。但是在英文的语言环境中,描述一个苹果的时候,量词是apple,当苹果数量为两个时,就会变成apples,这个组件的作用就在于此。
传入组件的参数中,value为数量,其他的为不同数量时对应的量词,在下面的例子中,一个的时候量词为message,两个的时候量词为messages。实际上可以传入组件的量词包括 zero, one, two, few, many, other 已经涵盖了所有的情况。
结合 <FormattedMessage />
运用:1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58const unreadCount = 10;
const unreadCount2 = 1;
...
<p>
<FormattedMessage
id="page.localeProvider.unreadCount"
defaultMessage={'你有{ unreadCount }条新信息'}
values={{
unreadCount: (
<strong
style={{
color: '#f30',
fontWeight: 'normal',
}}
>
<FormattedNumber
value={unreadCount}
/>
</strong>
),
notifications: (
<FormattedPlural
value={unreadCount}
one="notification"
other="notifications"
/>
),
}}
/>
</p>
<p>
<FormattedMessage
id="page.localeProvider.unreadCount"
defaultMessage={'你有{ unreadCount2 }条新信息'}
values={{
unreadCount: (
<strong
style={{
color: '#f30',
fontWeight: 'normal',
}}
>
<FormattedNumber
value={unreadCount2}
/>
</strong>
),
notifications: (
<FormattedPlural
value={unreadCount2}
one="notification"
other="notifications"
/>
),
}}
/>
</p>
输出结果:1
2
3
4
5
6
7<!-- 英文 -->
<p>You have 10 new notifications</p>
<p>You have 1 notification</p>
<!-- 中文 -->
<p>你有10条新信息</p>
<p>你有1条新信息</p>
b. <FormattedNumber />
这个组件最主要的用途是用来给一串数字标逗号,比如10000这个数字,在中文的语言环境中应该是1,0000,是每隔4位加一个逗号,而在英语的环境中是10,000,每隔3位加一个逗号。
传入数字作为参数:1
2
3<FormattedNumber
value={1000}
/>
输出结果:1
<span>1,000</span>
<FormattedNumber />
输出百分比
传入小数作为参数:1
2
3
4<FormattedNumber
value={0.5}
style="percent"
/>
输出结果:1
<span>50%</span>
<FormattedNumber />
输出货币
传入数字作为参数:1
2
3
4
5
6
7// locale.formats.money.currency 是 /locales/*.js 国际化资源配置的货币信息。中文: 'CNY'; 英文: 'USD'
<FormattedNumber
value={123456.78}
style="currency"
currency={locale.formats.money.currency}
/>
输出结果:1
2
3
4
5<!-- 英文 -->
<span>$123,456.78</span>
<!-- 中文 -->
<span>¥123,456.78</span>
注:项目在中文情况下也是每隔3位加一个逗号,具体原因详,如果有知道原因的请告知。
组件国际化
1. 创建获取上下文国际化资源函数
1 | /** |
2. 创建国际化的组件
1 | ... |
国际化规范附录
React Intl 编写规范
- 必须填写 defaultMessage,并将 defaultMessage 作为中文翻译
- id 不得重复
- 在使用 intl.formatMessage() 时,必须使用 defineMessages,预定义消息
源码
整个项目源码