介绍

Dojo 的 i18n 包解决了 web 应用程序国际化方面的诸多常见需求和挑战。

虽然其中大部分功能可作为独立模块使用,但它主要用于 Dojo 应用程序,以帮助渲染本地化的部件,包括高级消息、日期和数字格式化等。

功能描述
本地化单个部件每个部件实例都可以有自己的本地化设置,能够在单个应用程序中显示多套本地化数据。如果没有指定,部件将回退使用当前的根区域设置(root locale)。
精细控制的消息包包可以被拆分并作用于单个部件,并且只有使用了某一个区域设置时才延迟加载。这样,消息包也可以充分利用应用程序其他资源使用的分层和按包交付技术。
本地化的消息、日期和数字格式使用行业标准的 Unicode CLDR 格式化规则。在构建时,会将用到的 CLDR 数据自动注入到应用程序中。
响应式的区域设置与 Dojo 应用程序其他响应式的状态更改类似,当改变区域设置后,会自动重载消息,并重新渲染受影响的部件。如果单独使用 i18n 模块(如用在非 dojo 应用程序中),则可以通过监听器的回调函数来指定区域变更事件。
可回退的区域设置检测如果没有显式指定根区域,则必须确保设置了默认区域。系统的区域设置(当运行在服务器端时,指进程或主机的区域设置)会与 .dojorc 中定义的区域列表匹配。如果支持系统的区域设置,则将其设置为默认值。否则,使用 .dojorc 中定义的默认区域。

基本用法

国际化部件

  • 从单个默认语言开始(英语)。

.dojorc

  1. {
  2. "build-app": {
  3. "locale": "en"
  4. }
  5. }

src/widgets/MyI18nWidget.tsx

基于函数的部件:

  1. import { create, tsx } from '@dojo/framework/core/vdom';
  2. import i18n from '@dojo/framework/core/middleware/i18n';
  3. import myWidgetMessageBundle from '../nls/en/MyI18nWidget.ts';
  4. const factory = create({ i18n });
  5. export default factory(function MyI18nWidget({ middleware: { i18n } }) {
  6. const { messages } = i18n.localize(myWidgetMessageBundle);
  7. return <div title={messages.title}>{messages.content}</div>;
  8. });

基于类的部件:

  1. import { WidgetBase } from '@dojo/framework/core/WidgetBase';
  2. import { tsx } from '@dojo/framework/core/vdom';
  3. import I18nMixin from '@dojo/framework/core/mixins/I18n';
  4. import myWidgetMessageBundle from '../nls/en/MyI18nWidget.ts';
  5. export default class MyI18nWidget extends I18nMixin(WidgetBase) {
  6. protected render() {
  7. const { messages } = this.localizeBundle(myWidgetMessageBundle);
  8. return <div title={messages.title}>{messages.content}</div>;
  9. }
  10. }

src/nls/en/MyI18nWidget.ts

  1. export default {
  2. messages: {
  3. title: 'Hello',
  4. content: 'This is an internationalized widget'
  5. }
  6. };

添加部件的本地化语言包

  • 支持两种语言 - 默认为英语,同时也支持法语翻译,任何将 fr 设置为主要语言的用户都会使用法语翻译。

.dojorc

  1. {
  2. "build-app": {
  3. "locale": "en",
  4. "supportedLocales": [ "fr" ]
  5. }
  6. }

src/nls/en/MyI18nWidget.ts

  1. export default {
  2. locales: {
  3. fr: () => import('../fr/MyI18nWidget')
  4. },
  5. messages: {
  6. title: 'Hello',
  7. content: 'This is an internationalized widget'
  8. }
  9. };

src/nls/fr/MyI18nWidget.ts

  1. export default {
  2. title: 'Bonjour',
  3. content: 'Ceci est un widget internationalisé'
  4. };

在应用程序中指定根区域

在应用程序中只使用基于函数的部件和 i18n 中间件,则意味着无需在应用程序的 main.tsmain.tsx 入口点添加引导代码(bootstrapping code)。使用 @dojo/framework/core/middleware/i18n 中的 i18n 中间件,可以在顶层的 App 部件中设置默认区域。尚未定义区域时,可设置默认区域。

src/App.tsx

  1. import { create, tsx } from '@dojo/framework/core/vdom';
  2. const factory = create({ i18n });
  3. export default factory(function App({ middleware: { i18n } }) {
  4. if (!i18n.get()) {
  5. i18n.set({ locale: 'en-us', rtl: false });
  6. }
  7. return <div>{/* the application widgets */}</div>;
  8. });

但是,如果应用程序使用了基于类的部件,例如来自 @dojo/widgets 套件中的部件,则需要在应用程序的注册表(registry)中定义默认区域。这需要使用 @dojo/framework/core/mixins/I18n 中的工具函数 registryI18nInjector

src/main.tsx

  1. import renderer, { tsx } from '@dojo/framework/core/vdom';
  2. import Registry from '@dojo/framework/core/Registry';
  3. import { registerI18nInjector } from '@dojo/framework/core/mixins/I18n';
  4. import App from './App';
  5. const registry = new Registry();
  6. registerI18nInjector({ locale: 'en-us', rtl: false }, registry);
  7. const r = renderer(() => <App />);
  8. r.mount({ registry });

更改应用程序中的区域设置

  • 使用 i18n 中间件,用户可在支持的区域设置之间进行选择,然后使用中间件的 .set API 更改区域。

提醒: 当同时使用基于类的部件和基于函数的部件时,此中间件应该与 registeri18nInjector 一起使用,以便将区域设置的变更以响应的方式传播给所有支持 i18n 的部件。

src/widgets/LocaleChanger.tsx

  1. import { create, tsx } from '@dojo/framework/core/vdom';
  2. import i18n from '@dojo/framework/core/middleware/i18n';
  3. const factory = create({ i18n });
  4. export default factory(function LocaleChanger({ middleware: { i18n } }) {
  5. return (
  6. <div>
  7. <button
  8. onclick={() => {
  9. i18n.set({ locale: 'en' });
  10. }}
  11. >
  12. English
  13. </button>
  14. <button
  15. onclick={() => {
  16. i18n.set({ locale: 'fr' });
  17. }}
  18. >
  19. French
  20. </button>
  21. </div>
  22. );
  23. });