Skip to content

Latest commit

 

History

History
242 lines (176 loc) · 8.28 KB

File metadata and controls

242 lines (176 loc) · 8.28 KB

English | 中文 | CHANGELOG

@enum-plus/plugin-react

npm version license

集成 i18nextreact-i18next ,实现 React 应用中枚举标签的国际化,支持切换语言后自动更新UI。

简介

@enum-plus/plugin-reactenum-plus 的一个插件,可以认为是@enum-plus/plugin-i18next 插件的进阶版本。它提供了 i18nextreact-i18next 两个版本的插件。插件允许你在枚举定义中使用 i18next 的本地化键,并动态显示为当前语言的翻译文本,使得在 React 应用中使用 i18next 变得更加简单和高效。支持切换语言后,UI 自动更新,无需刷新页面。

它与 @enum-plus/plugin-i18next 插件的区别在于:

@enum-plus/plugin-i18next

  • 适用于任何 JavaScript 项目,返回的标签是字符串类型,适合在各种主流框架中使用。
  • 可以直接用于文本搜索等场景,例如,Array.includes 方法可以用于检查标签是否包含某个子字符串,或者绑定到Select等UI组件,搜索功能可以正常工作。
  • 缺点是无法监听语言变化,当语言变化时,需要手动重新渲染组件。

@enum-plus/plugin-react

  • 专为 React 应用设计,返回的标签是 React 组件,可以直接在 JSX 中使用。
  • 监听语言变化,当语言变化时,组件会自动重新渲染,无需刷新页面或手动干预。
  • 由于返回值不是字符串类型,因此无法直接用于文本搜索等场景。例如,Array.includes 方法无法用于检查标签是否包含某个子字符串,或者绑定到Select等UI组件,可能搜索功能会失效。为了解决这个问题,建议使用 @enum-plus/i18next 中提供的 isMatchisMatchCaseSensitive 方法。

安装

npm install @enum-plus/plugin-react

在应用程序的入口文件中,导入 @enum-plus/plugin-react 插件并安装:

  • 如果你使用 i18next
import { i18nextPlugin } from '@enum-plus/plugin-react';
import { Enum } from 'enum-plus';

Enum.install(i18nextPlugin);
  • 如果你使用 react-i18next
import { reactI18nextPlugin } from '@enum-plus/plugin-react';
import { Enum } from 'enum-plus';

Enum.install(reactI18nextPlugin);

插件选项

安装插件时,可以传入一个配置对象,用于设置插件的全局选项:

  • i18nextPlugin
Enum.install(i18nextPlugin, {
  localize: {
    // 设置 i18next 实例,如果有必要,默认为全局的 i18next 实例
    instance: i18next,
    // 传递给 i18next.t 方法的默认选项
    tOptions: {
      // 设置命名空间
      ns: 'my-namespace',
      // 设置返回值的默认值
      defaultValue: '-',
      // 其它 i18next.t 方法支持的选项
      // 请参考 https://www.i18next.com/translation-function/essentials#overview-options
    },
    defaultSearchField: 'label', // 设置 isMatch 和 isMatchCaseSensitive 方法中用于搜索的字段,默认为 'label'
  },
});
  • reactI18nextPlugin
Enum.install(reactI18nextPlugin, {
  localize: {
    // 设置 react-i18next 的 useTranslation 钩子选项
    useTranslationOptions: {
      // 设置命名空间
      ns: 'my-namespace',
      // 其它 useTranslation 方法支持的选项
      // 请参考 https://react.i18next.com/latest/usetranslation-hook#usetranslation-params
    },
    // 传递给 i18next.t 方法的默认选项
    tOptions: {
      // 设置返回值的默认值
      defaultValue: '-',
      // 其它 i18next.t 方法支持的选项
      // 请参考 https://www.i18next.com/translation-function/essentials#overview-options
    },
  },
});

tOptions 还支持函数形式,以便动态生成选项,甚至可以直接返回最终的翻译文本,两个插件均适用:

// 使用函数形式动态生成 tOptions
Enum.install(i18nextPlugin, {
  localize: {
    tOptions: (key, instance) => {
      if (key === 'week.sunday') {
        return { ns: 'my-namespace' };
      }
      return { ns: 'translation' }; // 默认命名空间
    },
  },
});

// 直接返回翻译文本
Enum.install(i18nextPlugin, {
  localize: {
    tOptions: (key, instance) => {
      if (key === 'week.sunday') {
        return '周日'; // 直接返回翻译文本
      }
      return instance.t(key); // 其它情况返回默认翻译
    },
  },
});

你甚至可以在 tOptions 中直接返回一个字符串,作为最终的翻译文本,以完全控制 localize 方法的行为。

Enum.install(i18nextPlugin, {
  localize: {
    tOptions: (key, instance) => {
      if (key === 'week.sunday') {
        return '周日'; // 直接返回翻译文本
      }
      return instance.t(key); // 其它情况返回默认翻译
    },
  },
});

用法

枚举标签响应语言的变化

可以通过在枚举定义中使用本地化键,来实现枚举标签的国际化。

import { Enum } from 'enum-plus';

const WeekEnum = Enum(
  {
    Monday: { value: 1, label: 'week.monday' },
    Tuesday: { value: 2, label: 'week.tuesday' },
  },
  {
    name: 'weekDays.name', // 枚举类型名称,可选
  }
);

WeekEnum.label(1); // Monday - ReactElement
WeekEnum.name; // Week - ReactElement

i18next.changeLanguage('zh-CN');
WeekEnum.label(1); // 星期一 - ReactElement
WeekEnum.name; // 周 - ReactElement

从枚举生成的UI组件,当语言变化时,标签会自动更新:

import { Button, Select } from 'antd';
import { changeLanguage } from 'i18next';

<Select options={WeekEnum.items} defaultValue={WeekEnum.Monday} />;
// 选中并显示: Monday

<Button onClick={() => changeLanguage('zh-CN')}>切换语言</Button>;

// 切换语言后,选中项的文本会自动更新为: 星期一

下拉框搜索

由于枚举的 label 已经变成了组件实例,而不是字符串类型,故无法直接用于文本搜索。可以使用 isMatchisMatchCaseSensitive 方法来实现对枚举项的过滤。

import { Select } from 'antd';

<Select options={WeekEnum.items} filterOption={WeekEnum.isMatch} />;

其它API

💎 isMatch

[方法]   isMatch(searchText: string, item: EnumItem): boolean

isMatch 方法用于根据搜索文本过滤枚举项,支持对枚举项的 label 进行模糊匹配,且忽略大小写。

此方法仅适用于Enum.localize返回非字符串的情况。例如,Enum.localize 在React框架下返回一个组件,以便能在切换语言后实时更新UI。在这种情况下,无法对枚举项的label进行字符串匹配,可以考虑使用此方法过滤枚举项。

  • 下拉框搜索
import { Select } from 'antd';

<Select options={WeekEnum.items} filterOption={WeekEnum.isMatch} />;
  • 常规过滤的用法
WeekEnum.items.filter((item) => WeekEnum.isMatch('Mon', item)); // 过滤出 label 中包含 'Mon' 的枚举项

💎 isMatchCaseSensitive

[方法]   isMatchCaseSensitive(searchText: string, item: EnumItem): boolean

isMatchCaseSensitive 方法用于根据搜索文本过滤枚举项,支持对枚举项的 label 进行模糊匹配,且区分大小写。

此方法仅适用于Enum.localize返回非字符串的情况。例如,Enum.localize 在React框架下返回一个组件,以便能在切换语言后实时更新UI。在这种情况下,无法对枚举项的label进行字符串匹配,可以考虑使用此方法过滤枚举项。

  • 下拉框搜索
import { Select } from 'antd';

<Select options={WeekEnum.items} filterOption={WeekEnum.isMatchCaseSensitive} />;
  • 常规过滤用法
WeekEnum.items.filter((item) => WeekEnum.isMatch('mon', item)); // 过滤出 label 中包含 'mon' 的枚举项 (区分大小写)