使用 i 翻译你的 React 项目从未如此简单

使用 i 翻译你的 react 项目从未如此简单

说疯狂的开发者!

今天我将向您展示,将整个项目翻译成 react 从未像现在这样容易。但首先您需要知道为什么这很重要。

当人们开始编程时,代码文本和消息通常使用葡萄牙语(pt-br)。将项目翻译成其他语言从来都不是优先事项,并且被认为是复杂或不必要的。

那么为什么它会相关呢?

这取决于你的实际情况。以下是您应该考虑此过程的一些原因:

公司需要
可能是您工作的公司或您拥有的某些 saas 开始在另一个国家/地区运营并且有此需求。具有此功能的产品有巨大的差异。

申请国​​际职位空缺
如果您正在申请国际职位空缺,拥有包含国际化项目的作品集可以给您带来引人注目的亮点。这表明您已准备好从事全球项目,并且不像大多数人一样懒惰。

你永远不会学太多
国际化不仅是一种特征,也是一种重要的学习经历。这是您的技能和工具库中的另一种武器。

过去是如何做到的?

项目翻译已经是个老问题了。人们在 html 中选择了该国家的国旗,供人们选择,并在代码中用 if 填充以了解将显示哪些文本。
这是非常被忽视的。网站是用单一语言制作的,翻译是随意添加的。如果是在后端,交易会更糟糕。

随着互联网的全球化,对多语言软件的需求不断增长,带来了针对国际化的特定工具。像 gnu gettext 这样的解决方案出现在后端,随后出现了像 i18next 和 react-intl 这样的前端库。然后疑问就来了…

i18next 对比react-intl:选择哪一个?

i18next:这个出现于 2011 年,它是一个 npm 包,适用于客户端的 node.js 和 spa。社区采纳了它,并最终于2015年在react-i18next lib中制作了react版本。因此,作为积极和消极的点,我们有:

优点:灵活性、路上时间(自 2011 年起)、庞大的生态系统(一次学习,随处翻译)和自动回退。缺点:学习曲线。有很多文档需要阅读,但并不是您需要的所有内容都在那里。

react-intl:formatjs 项目的一部分,遵循国际 javascript api 标准,确保与现代浏览器的兼容性。

优点:与 ecmascript 标准保持一致,集成简单。缺点:灵活性较差,插件支持较少。

我们将使用哪一个?

i18下一个我的朋友们!我总是建议阅读文档来开始,但让我们看看 doido 的指南!

使用 i18next 国际化 react 应用程序

安装

   npm install i18next i18next-chained-backend i18next-http-backend i18next-resources-to-backend react-i18next next-i18next 

配置:创建一个i18n.js来配置i18next。

   import i18n from 'i18next';   import { initreacti18next } from 'react-i18next';   import backend from 'i18next-http-backend';   import languagedetector from 'i18next-browser-languagedetector';   i18n     .use(backend)     .use(languagedetector)     .use(initreacti18next)     .init({ fallbacklng: 'en', interpolation: { escapevalue: false } });   export default i18n;

translations:在 locales/en/translation.json 和 locales/pt/translation.json 中创建翻译文件。

   {     "welcome": "welcome to our application!",     "login": "login"   }

翻译的使用:在 react 中使用 usetranslation 钩子。

   import react from 'react';   import { usetranslation } from 'react-i18next';   function app() {     const { t } = usetranslation();     return (       

{t('welcome')}

); } export default app;

语言更改:允许用户更改语言。

   import react from 'react';   import { usetranslation } from 'react-i18next';   function languageswitcher() {     const { i18n } = usetranslation();     const changelanguage = (lng) => i18n.changelanguage(lng);     return (       
); } export default languageswitcher;

这就是全部吗?

当然不是,我现在向您展示我在 crazystack 项目中做了什么。首先,我在 nextjs 中做了一个不同的配置,采用了我在项目本身的公共文件夹中定义的静态 json!看看:

import i18next from "i18next";import chainedbackend from "i18next-chained-backend";import httpbackend from "i18next-http-backend";import resourcestobackend from "i18next-resources-to-backend";import { initreacti18next } from "react-i18next";import { defaulttexts } from "./defaulttexts";i18next  .use(chainedbackend)  .use(initreacti18next)  .init({    lng: "pt-br",    fallbacklng: "pt-br",    interpolation: {      escapevalue: false,    },    compatibilityjson: "v3",    react: {      //wait: true,//usar no react native      usesuspense: false,    },    backend: {      backends: [httpbackend, resourcestobackend(defaulttexts)],      backendoptions: [        {          loadpath: `${process.env.next_public_url}/{{lng}}/{{ns}}.json`,        },      ],    },  });

然后我创建了一个上下文 api 来保存语言并在整个项目中访问它。从进口开始

2. 进口

import { usetranslation } from "react-i18next";import { createcontext, usestate, usecontext } from "react";

usetranslation:react-i18next 挂钩来访问翻译功能。在这里,您将在项目中几乎每个 jsx 组件中使用它。createcontext、usestate、usecontext:用于创建和使用上下文以及管理状态的 react 函数。

3. 上下文创建

const i18ncontext = createcontext({} as any);

创建上下文来通过 dom 存储和提供数据(例如当前语言)。

4. 环境检查

export const isbrowser = typeof window !== "undefined";

此行检查代码是否在浏览器中运行(而不是在服务器上),这对于处理 localstorage 等特定于客户端的功能至关重要。

5.i18nprovider组件

export const i18nprovider = ({ children }: any) => {  const { i18n } = usetranslation() || {};  const [currentlanguage, setcurrentlanguage] = usestate(    formatlanguagefromi18n(i18n?.language)  );  const changelanguage = (language) => {    setcurrentlanguage(language);    i18n?.changelanguage?.(formatlanguagefromselect(language));    localstorage.setitem("language", formatlanguagefromselect(language));  };  return (          {children}      );};

这个组件是一个provider,它包装了react组件树并提供了语言的当前状态以及更改它的函数。

usetranslation:从react-i18next库中检索i18n对象,其中包含有关当前语言的信息。currentlanguage:存储当前语言的state,根据i18n检测到的语言进行初始化。changelanguage:更改语言的函数,这也会将选择保存在 localstorage 中,以便在页面重新加载之间保持持久性。

6. 挂钩使用i18n

export const usei18n = () => {  if (!isbrowser) {    return {      currentlanguage: "pt-br",      setcurrentlanguage: () => {},      changelanguage: () => {},    };  }  return usecontext(i18ncontext);};

这个钩子可以轻松访问任何组件中的国际化上下文。

检查您是否在浏览器中(isbrowser)。如果没有,则返回默认值,以避免服务器端出错。如果在浏览器中,则消耗并返回 i18ncontext 上下文。

7. 转换图

const countrytolanguage = {  br: "pt-br",  us: "en",};const languagetocountry = {  "pt-br": "br",  en: "us",};

这些对象将国家/地区代码映射到语言代码,反之亦然,从而可以轻松地在不同约定之间格式化语言代码。

8. 格式化函数

export const formatlanguagefromi18n = (language) => languagetocountry[language];export const formatlanguagefromselect = (language) => countrytolanguage[language];

这些函数根据需要格式化语言环境。 formatlanguagefromi18n 将语言代码转换为国家/地区代码,而 formatlanguagefromselect 则进行反向转换。

完整代码

"use client";import { usetranslation } from "react-i18next";import { createcontext, usestate, usecontext } from "react";const i18ncontext = createcontext({} as any);export const isbrowser = typeof window !== "undefined";export const i18nprovider = ({ children }: any) => {  const { i18n } = usetranslation() || {};  const [currentlanguage, setcurrentlanguage] = usestate(    formatlanguagefromi18n(i18n?.language)  );  const changelanguage = (language) => {    setcurrentlanguage(language);    i18n?.changelanguage?.(formatlanguagefromselect(language));    localstorage.setitem("language", formatlanguagefromselect(language));  };  return (          {children}      );};export const usei18n = () => {  if (!isbrowser) {    return {      currentlanguage: "pt-br",      setcurrentlanguage: () => {},      changelanguage: () => {},    };  }  return usecontext(i18ncontext);};const countrytolanguage = {  br: "pt-br",  us: "en",};const languagetocountry = {  "pt-br": "br",  en: "us",};export const formatlanguagefromi18n = (language) => languagetocountry[language];export const formatlanguagefromselect = (language) => countrytolanguage[language];

然后我改变了导航栏

在代码中,我使用国家/地区下拉菜单选择语言。看看:

"use client";//@ts-nocheckimport { header, flex, logo, profile, notificationsnav, searchbar } from "@/shared/ui";import { usebreakpointvalue, icon, iconbutton, usemediaquery } from "@chakra-ui/react";import { rimenuline } from "react-icons/ri";import { useauth, usesidebardrawer } from "@/shared/libs";import { useeffect, usestate } from "react";import { countrydropdown } from "react-country-region-selector";import { theme } from "@/application/theme";import { formatlanguagefromi18n, usei18n } from "@/application/providers/i18nprovider";import { usetranslation } from "react-i18next";export const navbar = ({ showlogo = true }) => {  const { isauthenticated } = useauth() || {};  const { i18n } = usetranslation();  const { changelanguage, setcurrentlanguage } = usei18n() || {};  const { onopen = () => {}, onclose } = usesidebardrawer() || {};  const isdesktopversion = usebreakpointvalue({ base: false, lg: true });  const [country, setcountry] = usestate(formatlanguagefromi18n(i18n?.language));  useeffect(() => {    return () => {      onclose?.();    };  }, []);  const dropdown = countrydropdown as any;  useeffect(() => {    const language = localstorage.getitem("language");    if (language) {      setcountry(formatlanguagefromi18n(language));      setcurrentlanguage(language);      i18n?.changelanguage?.(language);    }  }, []);  return (    
{isauthenticated && !isdesktopversion && ( <iconbutton aria-label="open sidebar" fontsize="24" icon={} variant="unstyled" onclick={onopen} mr="1" mt={2} /> )} {/* {islargerthan560 && ( )} */} {isauthenticated && ( {/* */} { setcountry(val); changelanguage(val); }} labeltype="short" valuetype="short" showdefaultoption defaultoptionlabel="selecione o idioma" whitelist={["us", "br"]} style={{ backgroundcolor: theme.colors.secondary[400], padding: 10, width: 60, marginright: 15, borderradius: 8, }} /> )}
);};

导入和初始设置:

useauth:检查用户是否经过身份验证。usebreakpointvalue:根据屏幕大小决定是否显示桌面版本。usestate:设置国家/语言(国家)的初始状态,使用formatlanguagefromi18n函数格式化i18n的当前语言useeffect:第一个效果是在卸载组件时清除侧边栏(onclose)。第二个效果检查语言是否保存在 localstorage 中,如果是,则更新国家/地区状态并更改应用程序中的语言。

语言下拉菜单:

下拉列表是使用react-country-region-selector库的countrydropdown组件实现的,该组件被定制为用作语言选择器。value={country}:下拉列表中选择的值由国家/地区控制。onchange={(val) => { … }}:当下拉值改变时,更新国家状态,并调用changelanguage函数更改应用程序语言。whitelist={[“us”, “br”]}:将下拉选项限制为“us”(英语)和“br”(葡萄牙语)。style={…}:下拉菜单的自定义内联样式,使用主题的颜色和间距。语言选择器行为:下拉列表允许用户选择首选语言,并且此选择会保留在 localstorage 中。更改语言时,下拉列表会反映此更改,并且应用程序会更新为使用新选择的语言。要在文章中包含您在图像中提供的代码片段,您可以遵循以下格式:

以及如何更改文本?

从一个组件到另一个组件,我都遵循相同的过程。下面的代码展示了如何根据本地化密钥用动态翻译替换静态文本:

import { Divider } from "@chakra-ui/react";import { IoExitOutline } from "react-icons/io5";import { useRouter } from "next/navigation";import { useTranslation } from "react-i18next";  // Importando o hook useTranslationtype ProfileProps = {  showProfileData?: boolean;};export const Profile = ({ showProfileData }: ProfileProps) => {  const { t } = useTranslation(["PAGES"]);  // Obtendo a função t para tradução  const { user, logout } = useAuth() || {};  const router = useRouter();  const { showUserMenu, setShowUserMenu } = useProfile();  return (          {/* Outras partes do componente */}                                {t("PAGES:HOME_PAGE.logout", { defaultValue: "Sair" })}  // Chave de tradução com valor padrão                    );};

在此示例中,usetranslation 挂钩用于加载 pages:home_page.logout 翻译键。如果未找到该密钥,将显示默认文本“退出”。

结论

这个想法可以应用于任何静态文本组件。只需使用 usetranslation 钩子即可。
国际化您的应用程序可以打开全球市场的大门,突出您的投资组合并提高您的技能。在 i18next 和 react-intl 之间进行选择取决于您项目的具体需求,但对于那些想要入门的人来说,两者都是很好的选择。

课程建议

2022 年,我创建了crazystack 训练营。在其中,我展示了在线服务调度系统的 2 个完整应用程序,应用了设计模式、简洁架构、功能切片设计、solid、ddd 以及单元、集成和 e2e 测试等先进概念。

在第一个应用程序中,您将学习如何在 node.js 生态系统中构建 rest api。将创建涉及复杂业务规则的用例,例如列出可用时间、根据预订生成订单、忠诚度系统、佣金、付款、客户评论等等。一切都在 typescript 中完成并使用非关系数据库 mongodb

在第二个应用程序中,您将学习如何在 react.js 生态系统中构建管理面板来查看图表和操作记录。一切都是通过 typescript 和 next.js 框架完成的。此外,还将使用chakra ui可视化组件库,将原子设计概念应用于创建的组件。要了解更多信息,请访问crazystack.com.br。

以上就是使用 i 翻译你的 React 项目从未如此简单的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1490791.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月19日 13:27:01
下一篇 2025年12月19日 13:27:13

相关推荐

  • 为什么多年的经验让我选择全栈而不是平均栈

    在全栈和平均栈开发方面工作了 6 年多,我可以告诉您,虽然这两种方法都是流行且有效的方法,但它们满足不同的需求,并且有自己的优点和缺点。这两个堆栈都可以帮助您创建 Web 应用程序,但它们的实现方式却截然不同。如果您在两者之间难以选择,我希望我在两者之间的经验能给您一些有用的见解。 在这篇文章中,我…

    2025年12月24日
    000
  • 使用HTML表单GET方法创建带查询参数的动态搜索链接

    本教程详细阐述了如何在网页中,通过简单地将HTML表单的提交方法从`POST`更改为`GET`,来自动生成包含用户选择搜索条件的动态URL查询参数。这使得搜索结果页面的链接可分享、可收藏,并简化了%ignore_a_1%生成复杂URL的逻辑,后端可直接从URL中解析查询参数进行数据检索。 引言:动态…

    2025年12月23日
    000
  • Web表单提交:如何利用GET方法构建动态查询链接

    本文将指导您如何利用html表单的get方法,根据用户输入的搜索条件动态生成带有查询参数的url。当用户在表单中输入数据并提交时,get方法会自动将表单字段及其值附加到url作为查询字符串,从而创建如`/search?param1=value1&param2=value2`的链接。这对于实现…

    2025年12月23日
    000
  • Node.js中多字段多文件上传与MongoDB路径存储:Multer实践指南

    本教程详细讲解如何在node.js中使用multer处理html表单中的多字段文件上传,并将文件保存到服务器指定目录。核心内容包括multer配置,express路由集成,以及如何在控制器中正确获取上传文件信息并将其存储路径而非文件本身的数据存入mongodb,优化数据库性能和可维护性。 在现代We…

    2025年12月23日
    000
  • Node.js 多字段图片上传与 MongoDB 路径存储实践教程

    本教程详细阐述了如何在 Node.js 环境下,利用 Multer 中间件处理来自 HTML 表单的多个文件字段上传,并将图片分别存储到服务器的不同目录。更重要的是,教程纠正了直接将图片二进制数据存入 MongoDB 的常见误区,转而采用最佳实践——仅在数据库中存储图片的文件路径,从而优化数据库性能…

    2025年12月23日
    000
  • MongoDB/Mongoose中从数组中按ID删除子文档的教程

    本教程详细阐述了如何在mongodb/mongoose环境中,从一个文档的嵌套数组中,根据子文档的_id删除特定对象。我们将利用mongoose自动生成的_id字段和mongodb的$pull操作符,通过构建后端api路由,实现对特定子文档的精确、高效删除,并提供前端集成示例。 在构建复杂的应用程序…

    2025年12月23日
    000
  • 在MongoDB中通过子文档ID删除数组中的对象

    在MongoDB中处理复杂的数据结构时,我们经常会遇到需要在文档内部的数组中存储一系列子文档(或称内嵌文档)的场景。例如,在一个电影数据库中,一部电影文档可能包含一个`references`数组,其中每个元素都是一个描述电影引用的独立对象。当需要删除某个特定引用而非整个电影文档时,就需要一种精确的方…

    2025年12月23日
    000
  • MongoDB中通过子文档ID从数组删除对象的教程

    本教程详细介绍了如何在%ignore_a_1%中,利用mongoose框架从父文档的数组字段中删除特定的子文档。我们将探讨mongodb的`$pull`操作符,并提供完整的服务器端(node.js/express)和客户端(ejs)代码示例,指导你如何通过子文档的`_id`高效地移除数组中的元素,确…

    2025年12月23日
    000
  • 从MongoDB数组中按ID删除内嵌文档的教程

    本教程详细介绍了如何在mongodb中使用mongoose和`$pull`操作符,从一个文档的数组字段中删除特定的内嵌文档。我们将通过一个电影参考数据库的示例,演示如何根据内嵌文档的唯一id(`_id`)来精准定位并移除数组中的元素,确保数据管理的精确性和效率。 在构建数据库应用程序时,经常会遇到需…

    2025年12月23日
    000
  • HTML数据如何实现数据智能 HTML数据智能分析的技术架构

    实现HTML数据智能分析需构建包含采集、解析、存储、分析与可视化的闭环系统,首先通过爬虫技术获取网页数据并进行清洗标准化,接着利用DOM树分析与NLP技术提取结构化信息,随后将数据存入合适数据库或数据仓库并建立元数据管理机制,进而应用AI模型开展分类、情感分析、趋势预测与知识图谱构建等智能分析,最终…

    2025年12月23日
    000
  • HTML数据如何构建数据市场 HTML数据市场平台的架构设计

    答案是构建HTML数据市场需整合采集、管理、交易与合规体系,通过爬虫或上传获取网页数据,利用解析引擎将HTML转为结构化信息并标注元数据,依托分布式存储与版本控制保障数据可用性,建立索引分类和质量评估机制提升检索效率,通过商品发布、搜索推荐、API服务和计费系统实现数据流通,同时遵守法律规范,实施身…

    2025年12月23日
    000
  • HTML数据如何构建数据生态 HTML数据生态系统的建设思路

    HTML数据本身是网页内容的载体,但要将其转化为可用的数据资产并构建完整的数据生态系统,需要系统化的采集、处理、整合与应用流程。关键在于将静态的HTML内容动态化、结构化,并与其他数据源打通,形成可持续更新和高效利用的数据闭环。 1. 数据采集:从HTML中提取有效信息 构建HTML数据生态的第一步…

    2025年12月23日
    000
  • HTML数据如何实现分布式采集 HTML数据分布式爬虫的架构设计

    答案:构建分布式HTML采集系统需整合任务调度、去重、存储与监控模块,以Redis为核心协调任务分发与去重,通过消息队列实现负载均衡,结合布隆过滤器减少重复抓取,利用无状态工作节点支持弹性扩展,依托ZooKeeper保障高可用,并集成反爬适配与请求控制机制,确保系统稳定高效运行。 要实现HTML数据…

    好文分享 2025年12月23日
    000
  • HTML5在线如何添加实时编辑功能 HTML5在线协作工具的集成方案

    实现实时编辑需结合WebSocket与协同算法,核心是内容同步和冲突解决。2. 使用WebSocket实现低延迟通信,用户操作实时发送至服务器并广播给其他客户端。3. 冲突处理推荐CRDT或OT算法,CRDT因逻辑简单、天然无冲突更适用于现代系统。4. 集成Y%ignore_a_1%等成熟库可快速搭…

    好文分享 2025年12月23日
    000
  • 如何实现HTML在线编辑功能_HTML在线编辑功能实现方法与技术选型指南

    答案:实现HTML在线编辑需选型富文本编辑器如TinyMCE、Quill或CKEditor,通过前端获取HTML字符串并经AJAX提交,后端存储至数据库,同时使用DOMPurify防范XSS攻击,结合实时预览与样式隔离提升体验,并可扩展图片上传、代码高亮等功能。 要实现HTML在线编辑功能,核心是提…

    2025年12月23日
    000
  • 如何实现HTML在线协作编辑_HTML在线协作编辑实现技术与实时同步方案

    实现HTML在线协作编辑需选型合适编辑器并构建实时同步体系。推荐Tiptap + Yjs组合,基于CRDT模型实现无冲突数据合并,通过WebSocket或Socket.IO建立低延迟通信,配合Y-WebRTC/Y-Socket.IO完成多端状态同步,后端可用Node.js搭建中继服务器,必要时结合M…

    2025年12月23日
    000
  • Node.js中基于JWT和Cookie实现持久化登录状态管理与免登录访问

    本文详细阐述了如何在node.js应用中,利用jwt(json web tokens)和cookie实现持久化的用户登录状态管理,从而避免用户在每次访问时重复登录。通过引入一个认证中间件,我们能够有效地检查用户是否已通过有效令牌进行身份验证,并据此控制页面访问权限,实现无缝的用户体验,同时提供了登出…

    2025年12月23日
    000
  • 使用FastAPI实现POST请求后文件下载的教程

    本文详细介绍了在fastapi中处理post请求后下载文件的两种主要方法。第一种是直接使用`fileresponse`返回文件,适用于简单场景,通过设置`content-disposition`头部实现强制下载,并探讨了内存加载和流式传输大文件的替代方案。第二种是异步下载模式,通过post请求生成文…

    2025年12月23日
    000
  • Spring Boot中将特定字段映射到HTML页面:DTO与视图渲染实践

    本文将深入探讨如何在Spring Boot应用中,从后端实体中选择性地提取特定字段,并将其高效且安全地渲染到HTML页面。我们将重点介绍数据传输对象(DTO)作为最佳实践,结合Spring MVC控制器和模板引擎(如Thymeleaf),实现数据展示层与业务逻辑的解耦,并讨论其他序列化控制策略。 1…

    2025年12月22日
    000
  • Spring Boot中将后端数据特定字段映射到HTML页面教程

    本教程详细阐述了如何在Spring Boot应用中,利用Thymeleaf模板引擎将后端服务获取的数据,仅提取并展示其特定字段(如标题和描述)到前端HTML页面。通过重构控制器方法并设计相应的HTML模板,实现数据与视图的有效分离与定制化渲染,同时辨析了@JsonIgnore注解的适用场景。 问题背…

    2025年12月22日
    000

发表回复

登录后才能评论
关注微信