使用您最喜欢的AI助手总结文档,并引用此页面和AI提供商
版本历史
- "更新 Solid useIntlayer API 用法以直接访问属性"v8.9.02026/5/4
- "Init doc"v8.4.52026/3/20
此页面的内容已使用 AI 翻译。
查看英文原文的最新版本If you have an idea for improving this documentation, please feel free to contribute by submitting a pull request on GitHub.
GitHub link to the documentationCopy doc Markdown to clipboard
在 Storybook 中使用 Intlayer
目录
为什么选择 Inlayer 而不是替代品?
与“storybook-react-i18next”或“i18next”等主要解决方案相比,Intlayer是一个具有集成优化的解决方案,例如:
完整的故事书报道
Intlayer 经过优化,可与 Storybook 完美配合,提供多语言故事装饰器、区域设置切换以及在整个设计系统中扩展国际化 (i18n) 所需的所有功能。
捆绑尺寸
不要将大量 JSON 文件加载到页面中,而只需加载必要的内容。 Intlayer 有助于将捆绑包和页面大小减少多达 50%。
可维护性
确定应用程序内容的范围有利于大型应用程序的维护。您可以复制或删除单个功能文件夹,而无需承担检查整个内容代码库的精神负担。此外,Intlayer 具有完全类型化 (fully typed),以确保您的内容的准确性。
人工智能代理
共置内容减少大型语言模型 (LLM) 所需的上下文。 Intlayer 还附带了一套工具,例如用于测试缺失翻译的 CLI、LSP、MCP 和 agent技能,使 AI 代理的开发者体验 (DX) 更加流畅。
自动化
使用您选择的法学硕士,通过自动化在 CI/CD 管道中进行翻译,而费用由您的 AI 提供商承担。 Intlayer 还提供了一个编译器来自动提取内容,以及一个网络平台来帮助在后台翻译。
表现
将大量 JSON 文件连接到组件可能会导致性能和反应性问题。 Intlayer 可在构建时 (build time)优化您的内容加载。
无需开发即可扩展
Intlayer 不仅仅是一个 i18n 解决方案,还提供了一个自托管的可视化编辑器和一个完整的 CMS 来帮助您管理多语言内容实时,与译员、文案人员和其他团队成员无缝协作。内容可以本地和/或远程存储。
为什么要结合 Storybook 使用 Intlayer?
Storybook 是开发和记录 UI 组件的行业标准工具。通过将它与 Intlayer 结合使用,您可以:
- 直接在 Storybook 画布中预览每种语言:使用工具栏切换器。
- 提前捕获缺失的翻译:在进入生产环境之前修正问题。
- 记录多语言组件:使用真实的、类型安全的内容,而不是硬编码的字符串。
逐步设置指南
第一步:安装依赖项
复制代码到剪贴板
npm install intlayer react-intlayernpm install vite-intlayer --save-dev在弹窗中打开表格以清晰地查看所有数据
| 软件包 | 角色 |
|---|---|
intlayer | 核心 - 配置、内容编译、命令行工具 |
react-intlayer | React 绑定 - IntlayerProvider, useIntlayer 钩子 |
vite-intlayer | Vite 插件 - 监听并编译内容声明文件 |
第二步:创建 Intlayer 配置
在项目的根目录(或设计系统包内)创建 intlayer.config.ts:
复制代码到剪贴板
import { Locales, type IntlayerConfig } from "intlayer";const config: IntlayerConfig = { internationalization: { locales: [ Locales.ENGLISH, Locales.FRENCH, Locales.SPANISH, // 根据需要添加更多语言 ], defaultLocale: Locales.ENGLISH, }, content: { contentDir: ["./src"], // 您的 *.content.ts 文件所在位置 },};export default config;有关完整选项列表,请参见 配置参考。
第三步:将 Vite 插件添加到 Storybook
Storybook 的 viteFinal 钩子允许您扩展内部的 Vite 配置。在此处导入并添加 intlayer() 插件:
复制代码到剪贴板
import type { StorybookConfig } from "@storybook/react-vite";import { defineConfig, mergeConfig } from "vite";import { intlayer } from "vite-intlayer";const config: StorybookConfig = { stories: ["../src/**/*.stories.@(js|jsx|ts|tsx)"], addons: [ "@storybook/addon-essentials", // …其他插件 ], framework: { name: "@storybook/react-vite", options: {}, }, async viteFinal(baseConfig, { configType }) { const env = { command: configType === "DEVELOPMENT" ? "serve" : "build", mode: configType === "DEVELOPMENT" ? "development" : "production", } as const; const viteConfig = defineConfig(() => ({ plugins: [intlayer()], })); return mergeConfig(baseConfig, viteConfig(env)); },};export default config;intlayer() 插件会监听您的 *.content.ts 文件,并在 Storybook 开发过程中文件发生变化时自动重新构建字典。
第四步:添加 IntlayerProvider 装饰器和语言工具栏
Storybook 的 preview 文件是使用 IntlayerProvider 包裹每个 story,并在工具栏中暴露语言切换器的最佳场所:
复制代码到剪贴板
import type { Preview, StoryContext } from "@storybook/react";import { IntlayerProvider } from "react-intlayer";const preview: Preview = { // 在 IntlayerProvider 中包裹每个 story decorators: [ (Story, context: StoryContext) => { const locale = context.globals.locale ?? "en"; return ( <IntlayerProvider locale={locale}> <Story /> </IntlayerProvider> ); }, ], // 在 Storybook 工具栏中暴露语言切换器 globalTypes: { locale: { description: "当前语言", defaultValue: "en", toolbar: { title: "语言", icon: "globe", items: [ { value: "en", title: "English" }, { value: "fr", title: "Français" }, { value: "es", title: "Español" }, ], dynamicTitle: true, }, }, }, parameters: { controls: { matchers: { color: /(background|color)$/i, date: /Date$/i, }, }, },};export default preview;locale的值必须与您在intlayer.config.ts中声明的语言一致。
声明内容
在每个组件旁边创建一个 *.content.ts 文件。Intlayer 会在编译期间自动识别它。
复制代码到剪贴板
import { type Dictionary, t } from "intlayer";
const copyButtonContent = {
key: "copy-button",
content: {
label: t({
en: "Copy content",
fr: "Copier le contenu",
es: "Copiar contenido",
}),
},
} satisfies Dictionary;
export default copyButtonContent;有关更多内容声明格式和功能,请参阅 内容声明文档。
在组件中使用 useIntlayer
复制代码到剪贴板
"use client";import { type FC } from "react";import { useIntlayer } from "react-intlayer";type CopyButtonProps = { content: string;};export const CopyButton: FC<CopyButtonProps> = ({ content }) => { const { label } = useIntlayer("copy-button"); return ( <button onClick={() => navigator.clipboard.writeText(content)} aria-label={label.value} title={label.value} > 复制 </button> );};useIntlayer 会返回由最近的 IntlayerProvider 提供的当前语言的编译后的字典。在 Storybook 工具栏中切换语言会自动重新渲染对应的 story 并更新翻译。
为国际化组件编写 Story
在配置好 IntlayerProvider 装饰器之后,您的 story 工作方式与以前完全相同。语言工具栏控制整个画布的当前语言:
复制代码到剪贴板
import type { Meta, StoryObj } from "@storybook/react";import { CopyButton } from ".";const meta: Meta<typeof CopyButton> = { title: "Components/CopyButton", component: CopyButton, tags: ["autodocs"], argTypes: { content: { control: "text" }, },};export default meta;type Story = StoryObj<typeof CopyButton>;/** 默认 Story - 在工具栏中切换语言来预览翻译。 */export const Default: Story = { args: { content: "npm install intlayer react-intlayer", },};/** 在代码块内渲染该按钮,这是一个常见的现实用例。 */export const InsideCodeBlock: Story = { render: (args) => ( <div style={{ position: "relative", display: "inline-block" }}> <pre style={{ background: "#1e1e1e", color: "#fff", padding: "1rem" }}> <code>{args.content}</code> </pre> <CopyButton content={args.content} style={{ position: "absolute", top: 8, right: 8 }} /> </div> ), args: { content: "npx intlayer init", },};每个 story 都会从工具栏继承 locale 全局变量,因此您可以在不更改任何 story 代码的情况下验证每种语言。
在 Story 中测试翻译
使用 Storybook 的 play 函数来断言在指定语言下是否渲染了正确的翻译文本:
复制代码到剪贴板
import type { Meta, StoryObj } from "@storybook/react";import { expect, within } from "@storybook/test";import { CopyButton } from ".";const meta: Meta<typeof CopyButton> = { title: "Components/CopyButton", component: CopyButton, tags: ["autodocs"],};export default meta;type Story = StoryObj<typeof CopyButton>;export const AccessibleLabel: Story = { args: { content: "Hello World" }, play: async ({ canvasElement }) => { const canvas = within(canvasElement); const button = canvas.getByRole("button"); // 验证按钮是否具有非空的无障碍名称 await expect(button).toHaveAccessibleName(); // 验证按钮未被禁用 await expect(button).not.toBeDisabled(); // 验证键盘可访问性 await expect(button).toHaveAttribute("tabindex", "0"); },};