”工欲善其事,必先利其器。“—孔子《论语.录灵公》
首页 > 编程 > 在 Astro Build 中为 i 创建动态路线语言

在 Astro Build 中为 i 创建动态路线语言

发布于2024-08-30
浏览:818

Creating dynamic route language for i in Astro Build

Se quiser ler esse artigo em Portugês clique aqui

最近,我开始学习Astro来创建一个类似仪表板的项目。

我真的很想在这个项目中实现国际化(i18n)——这个想法是每个人都应该能够使用它,无论他们的语言如何。

问题

Astro的i18n支持相当好。它的工作方式类似于 Next.js 或任何其他基于文件/文件夹结构进行路由的框架。

因此,如果我们想要一个英文页面和一个葡萄牙语页面,我们可以这样组织我们的文件:

.
└── src/
    └── pages/
        ├── en/
        │   ├── login.astro
        │   └── dashboard.astro
        └── pt-br/
            ├── login.astro
            └── dashboard.astro

每个页面都有自己的 i18n 字符串 - 很好!

但这就是我的问题开始的地方:我不想克隆我的所有页面;我只想更改这些页面上的字符串。

我需要类似 /[any-language-flag]/all-my-routes 的东西。

你可能会问:“为什么不使用像react-intl这样的东西呢?”我的答案是,我想充分利用 Astro 的引擎,尤其是 SSG/SSR,并避免任何客户端组件。一般来说,这些框架使用React Context,它只在客户端渲染。

尝试与失败

首先,我阅读了关于 i18n 的 Astro 食谱,并查看了一些社区库来解决这个问题。

我测试的第一个库是 astro-i18next,它看起来正是我需要的!

基于配置文件中的数组,astro-i18next 在构建时生成我的 i18n 页面,因此我只需要编写一次代码,而不必担心克隆页面。

问题是 astro-i18next 似乎已存档或不再维护。问题很多,最后一次提交已经是一年多前了。

解决方案

在尝试其他库(值得赞扬的是 astro-i18n)之后,我发现了 Paraglide,它改变了我的项目。

我选择滑翔伞是因为:

  • 它是类型安全的,因此我可以将它与 TypeScript 一起使用并受益于自动完成功能。
  • 它将 i18n 字符串转换为函数,因此如果字符串键发生更改,我的构建将会失败,从而尽早捕获错误。
  • 使用 i18n 函数可以更好地进行树摇动,删除未使用的函数。
  • 有一个 VS Code 扩展可以增强开发体验。

注意:你也可以在JS项目中使用Paraglide,并且它还支持Next.js。

安装和配置后,我使用了这样的消息:

---
import * as m from "../paraglide/messages.js";
---

{m.hello({ name: "Alan" })}

然而,这并没有解决我的路由问题——我仍在为我想要添加的每种语言克隆我的页面。

为了解决这个问题,我将项目更改为在根路由中使用动态路由,因此我的所有路由现在都以语言标志开头。

我的文件夹结构变成了这样:

.
└── src/
    └── pages/
        └── [lang]/
            ├── login.astro
            └── dashboard.astro

此更改后,Paraglide可以自动从路线参数中获取语言:

  • http://localhost:4321/en/login
  • http://localhost:4321/pt-br/login

现在,我只需在 astro.config.ts 中设置新语言并翻译我的字符串文件即可添加新语言。

但是我还有两个问题需要解决:

  1. 当用户第一次访问 http://localhost:4321/ 时不带语言标志。
  2. 如果用户更改特定路线上的语言,我需要将它们保持在同一路线上(例如,/en/create-account 应重定向到 /pt-br/create-account 或 /pt-br/criar -conta).

语言重定向中间件

为了解决第一个问题语言重定向,我使用了Astro中间件。

在src/middleware/index.ts中,我添加了这段代码:

import { defineMiddleware } from 'astro:middleware';
import {
  languageTag,
  setLanguageTag,
  type AvailableLanguageTag,
} from '../paraglide/runtime';

export const onRequest = defineMiddleware((context, next) => {
  // Get lang from url param
  const lang = context.params.lang;

  // If changed
  if (lang !== languageTag()) {
    setLanguageTag(lang as AvailableLanguageTag);
    // Redirect to lang changed or default (en)
    return context.redirect(`/${lang ?? 'en'}`);
  }

  return next();
});

具有当前路线的语言选择器

为了让用户在切换语言时保持相同的路线,我添加了这个组件:

---
import { languageTag } from '../paraglide/runtime';

const pathName = Astro.url.pathname.replace(`/${languageTag()}/`, '');
---

此外,我们也可以使用 Paraglide 消息函数中的第二个参数来翻译这些消息:

注意事项

我不认为我的解决方案是最好的,特别是因为我仍在学习 Astro,所以可能还有其他解决方案。如果您知道,请发表评论,我会尝试一下:)

感谢您阅读本文!如果您有任何疑问,请评论,我很乐意回复。

版本声明 本文转载于:https://dev.to/alancpazetto/creating-dynamic-route-language-for-i18n-in-astro-build-2iim?1如有侵犯,请联系[email protected]删除
最新教程 更多>
  • 雷迪斯
    雷迪斯
    Redis 是一个 NOSQL 数据库,数据以键值对的形式存储。 Redis 将数据存储在主内存上,这意味着它速度快,但易失性,不像 PostgreSQL 等数据库系统将数据存储在磁盘上。 概述 在具有服务器和数据库的应用程序中,访问数据库上的信息会导致延迟。 像redis这样...
    编程 发布于2024-11-06
  • 如何使用预先导入的包启动 python 或 ipython。
    如何使用预先导入的包启动 python 或 ipython。
    不必每次执行 python 或 ipython 时都必须编写“import os”或其他常见包,此脚本将启动 python 并导入这些包 #!/usr/bin/env bash # -----------------------------------------------------------...
    编程 发布于2024-11-06
  • python中的数据类型
    python中的数据类型
    内置数据类型 文本类型:字符串 数值类型:int、float、complex 序列类型:列表、元组、范围 映射类型:字典 集合类型:集合、冻结集合 布尔类型:bool 二进制类型:字节、字节数组、内存视图 无类型:无类型 程序 a=“阿维纳什” 打印(类型(a)) b=45 打印(类型(b)) c=...
    编程 发布于2024-11-06
  • JavaScript 中的提升
    JavaScript 中的提升
    什么是 JavaScript 提升? 提升是指 JavaScript 中在执行之前进行声明的过程。首先处理变量和函数声明。因此,即使变量在声明之前被引用,也不会导致错误,而是返回 undefined。对于函数声明,整个函数被提升,这意味着它可以在代码中定义之前使用。此过程在执行开始...
    编程 发布于2024-11-06
  • 如何在 Go 中同时选择缓冲发送和无缓冲接收通道?
    如何在 Go 中同时选择缓冲发送和无缓冲接收通道?
    同时选择 Go 发送和接收通道在 Go 中,可以使用 select 语句在通道上执行非阻塞 I/O 操作。然而,在处理缓冲发送通道 (chan
    编程 发布于2024-11-06
  • 如何将列表列表转换为统一的 NumPy 数组?
    如何将列表列表转换为统一的 NumPy 数组?
    将列表列表转换为 NumPy 数组数据分析中的一个常见任务是将列表列表转换为 NumPy 数组高效的数值运算。该数组可以通过将每个列表分配给一行来形成,列表中的每个元素占据一列。选项 1:数组数组如果子列表具有不同的长度,合适的方法是创建数组的数组。这保留了列表列表的原始结构,从而可以轻松检索特定元...
    编程 发布于2024-11-06
  • 前端的顶级设计模式
    前端的顶级设计模式
    在过去的几个月里,我为前端开发人员分享了一些流行的设计模式。其中包括 Singleton、Facade、Observer、Publisher/Subscriber 等模式。今天,我想总结一下这些模式的一些要点和好处,以及如何使用它们来改进您的前端开发流程。 什么是设计模式 设计模式是...
    编程 发布于2024-11-06
  • ServBay版本.pdate公告
    ServBay版本.pdate公告
    我们很高兴地宣布新版本 1.4.4 已经到来!让我们来看看新增的备受期待的新功能。 新功能 CA和证书管理: 统一SSL证书管理平台:全新的证书管理平台,旨在简化证书申请和管理流程。 ServBay User CA 和 ServBay Public CA: 引入 Se...
    编程 发布于2024-11-06
  • Spring框架中的控制反转
    Spring框架中的控制反转
    控制反转(IoC)和依赖注入(DI)是Spring框架中的两个基本概念。传统上,对象负责创建和管理它们自己的依赖关系。然而,IoC 通过将对象创建和依赖管理的控制权移交给像 Spring 这样的框架来翻转这一责任。 这种转变有几个优点: 更简单的实现交换:只需对代码库进行最小的更改即可交换不同的实现...
    编程 发布于2024-11-06
  • 使用 React 构建递归文件系统:深入探讨
    使用 React 构建递归文件系统:深入探讨
    简介:在 React 中构建递归文件系统 在现代 Web 开发中,创建交互式动态文件系统是一个常见的要求。无论是管理文档、组织项目还是构建复杂的数据结构,拥有强大的文件系统都至关重要。在这篇博文中,我们将探讨如何在 React 中构建递归文件系统,重点关注可以添加、重命名或删除的嵌...
    编程 发布于2024-11-06
  • SQL 查询速度慢?使用此技术提高应用程序的性能
    SQL 查询速度慢?使用此技术提高应用程序的性能
    挑战 在我的应用程序(React Spring Boot Oracle)中,处理大型数据集导致处理时间极其缓慢。我需要一种解决方案来提高性能而不影响准确性或完整性。 解决方案:NTILE 并行处理 NTILE 是一个功能强大的 SQL 窗口函数,旨在将结果集划分为指...
    编程 发布于2024-11-06
  • 关于测试覆盖率的真相
    关于测试覆盖率的真相
    一个强有力的真理。 看下面这段简单明了的代码: function sum(a, b) { return a b; } 现在,让我们为它编写一些测试: test('sum', () => { expect(sum(1, 2)).toBe(3); expect(s...
    编程 发布于2024-11-06
  • 为什么我的 OpenGL 三角形无法在 Go 中渲染?调查顶点缓冲区问题。
    为什么我的 OpenGL 三角形无法在 Go 中渲染?调查顶点缓冲区问题。
    Go 中的 OpenGL 顶点缓冲区问题在尝试在 Go 中使用 OpenGL 显示三角形时,用户遇到了顶点缓冲区问题缓冲区无法渲染形状。 Go 代码源自教程,但与 C 代码不同的是,它没有产生任何输出。问题原因问题的根本原因位于传递给 vertexAttrib.AttribPointer() 的参数...
    编程 发布于2024-11-06
  • 为什么在 Linux 32 位发行版上的 Go 程序中设置 `ulimit -n` 会导致“参数无效”错误?
    为什么在 Linux 32 位发行版上的 Go 程序中设置 `ulimit -n` 会导致“参数无效”错误?
    如何在 Go 程序中设置 ulimit -n?问题用户尝试在 Go 程序中设置 ulimit -n使用 setrlimit 和 getrlimit 系统调用将其限制在程序内而不是全局。然而,在尝试设置该值时出现错误,提示“参数无效”。解决方案发现该问题是由于 Linux 32 的 Getrlimit...
    编程 发布于2024-11-06
  • 如何在Python中创建无限深度的动态嵌套字典?
    如何在Python中创建无限深度的动态嵌套字典?
    未定义深度的动态嵌套字典在涉及复杂多级数据结构的场景中,经常会遇到变量嵌套字典的需求水平。虽然硬编码插入语句是一种潜在的解决方案,但当事先未知嵌套深度时,这种方法是不切实际的。要克服此限制,请考虑利用 Python 的 collections.defaultdict,它允许动态创建字典。可以使用以下...
    编程 发布于2024-11-06

免责声明: 提供的所有资源部分来自互联网,如果有侵犯您的版权或其他权益,请说明详细缘由并提供版权或权益证明然后发到邮箱:[email protected] 我们会第一时间内为您处理。

Copyright© 2022 湘ICP备2022001581号-3