跳转到内容

国际化(i18n)路由

Astro 的国际化(i18n)功能允许你将项目适配给国际受众。这个路由 API 帮助你生成、使用并验证你的多语言站点产生的 URL。

Astro 的 i18n 路由允许你带来你的多语言内容,并支持配置默认语言、计算相对页面 URL 以及接受访客浏览器提供的首选语言。你还可以基于每种语言指定回退语言,以便你的访客始终可以被引导至你站点上现有的内容。

Astro 通过使用一个 中间件 来实现其路由逻辑。这个中间件函数被放置在 第一个位置,它会在最终执行自己的逻辑之前,等待来自任何额外中间件和每个页面路由的每一个 Response

这意味着来自你自己的中间件和页面逻辑的操作(例如重定向)首先被执行,你的路由被渲染,然后 i18n 中间件执行它自己的动作,比如验证一个本地化的 URL 是否对应一个有效的路由。

你也可以选择在 Astro 的 i18n 中间件之外,添加你自己的 i18n 逻辑,这样你可以在仍然能访问 astro:i18n 辅助函数的同时,对你的路由有更多的控制权。

i18n 配置对象中必须指定一个默认语言(defaultLocale)和所有支持的语言列表(locales)。此外,你还可以配置更具体的路由和回退行为,以匹配你期望的 URL。

astro.config.mjs
import { defineConfig } from "astro/config"
export default defineConfig({
i18n: {
defaultLocale: "en",
locales: ["es", "en", "pt-br"],
}
})

根据语言组织你的内容文件夹中的本地化内容。在 src/pages/ 内的任何位置创建单独的 /[locale]/ 文件夹,Astro 的 基于文件的路由 将会在相应的 URL 路径下创建你的页面。

你的文件夹名称必须与 locales 中的项完全匹配。仅当你配置了 prefixDefaultLocale: true 来为你的默认语言显示一个本地化的 URL 路径时(例如 /zh-cn/about/),才包含一个为你的 defaultLocale 设定的本地化文件夹。

  • 文件夹src
    • 文件夹pages
      • about.astro
      • index.astro
      • 文件夹es
        • about.astro
        • index.astro
      • 文件夹pt-br
        • about.astro
        • index.astro

配置了 i18n 路由后,你现在可以使用诸如 astro:i18n 模块 中提供的 getRelativeLocaleURL() 辅助函数来计算站点内页面的链接。这些生成的链接将始终提供正确的、本地化的路由,并且可以帮助你正确地使用或检查站点上的 URL。

你也可以选择手动编写链接。

src/pages/es/index.astro
---
import { getRelativeLocaleUrl } from 'astro:i18n';
// defaultLocale 为 "es"
const aboutURL = getRelativeLocaleUrl("es", "about");
---
<a href="/get-started/">¡Vamos!</a>
<a href={getRelativeLocaleUrl('es', 'blog')}>Blog</a>
<a href={aboutURL}>Acerca</a>

Astro 的内置基于文件的路由系统会根据你在 src/pages/ 中的文件结构自动为你创建 URL 路由。

当你配置 i18n 路由时,这个文件结构的信息(以及生成的相应 URL 路径)将可供 i18n 辅助函数使用,以便它们能够为你的项目生成、使用和验证路由。许多这样的选项可以一起使用,以实现更多的自定义和按语言的灵活性。

你甚至可以选择手动实现自己的路由逻辑,以获得更大的控制权。

添加于: astro@3.5.0

此路由选项定义了你的默认语言的 URL 是否应该使用语言前缀(例如 /en/about/)。

所有非默认支持的语言使用本地化前缀(例如 /fr//french/),并且内容文件必须位于适当的文件夹中。此配置选项允许你指定你的默认语言是否也应该遵循本地化的 URL 结构。

此设置还决定了你的默认语言的页面文件必须存在于哪个位置(例如 src/pages/about/src/pages/en/about),因为文件结构和 URL 结构必须对所有语言都匹配。

  • "prefixDefaultLocale: false"(默认):你的默认语言的 URL 不会/[locale]/ 前缀。所有其他语言都会有。

  • "prefixDefaultLocale: true":所有 URL,包括你的默认语言,都会有 /[locale]/ 前缀。

prefixDefaultLocale: false

段落标题 prefixDefaultLocale: false
astro.config.mjs
import { defineConfig } from "astro/config"
export default defineConfig({
i18n: {
defaultLocale: "en",
locales: ["es", "en", "fr"],
routing: {
prefixDefaultLocale: false
}
}
})

这是默认值。当你的默认语言的 URL 不会/[locale]/ 前缀,并且你的默认语言的文件位于 src/pages/ 的根目录时,设置这个选项:

  • 文件夹src
    • 文件夹pages
      • about.astro
      • index.astro
      • 文件夹es
        • about.astro
        • index.astro
      • 文件夹fr
        • about.astro
        • index.astro
  • src/pages/about.astro 将生成路由 example.com/about/
  • src/pages/fr/about.astro 将生成路由 example.com/fr/about/
astro.config.mjs
import { defineConfig } from "astro/config"
export default defineConfig({
i18n: {
defaultLocale: "en",
locales: ["es", "en", "fr"],
routing: {
prefixDefaultLocale: true
}
}
})

当所有路由在其 URL 中都有 /locale/ 前缀,并且包括默认语言环境的所有页面内容文件都存在于一个本地化文件夹中时,设置这个选项:

  • 文件夹src
    • 文件夹pages
      • index.astro // 注意:这个文件是必需的
      • 文件夹en
        • index.astro
        • about.astro
      • 文件夹es
        • about.astro
        • index.astro
      • 文件夹pt-br
        • about.astro
        • index.astro
  • 没有区域前缀的 URL(例如 example.com/about/)将返回 404(未找到)状态码,除非你指定了一个回退策略

添加于: astro@4.2.0

配置是否将由 src/pages/index.astro 生成的首页 URL (/) 重定向到 /<defaultLocale>

设置 prefixDefaultLocale: true 也会在你的 routing 配置对象中自动设置 redirectToDefaultLocale: true。默认情况下,必需的 src/pages/index.astro 文件将自动重定向到你的默认语言环境的首页。

你可以通过设置 redirectToDefaultLocale: false 来选择退出此行为。这允许你拥有一个存在于你配置的语言环境文件夹结构之外的站点首页。

添加于: astro@4.6.0

当这个选项被启用时,Astro 将禁用其 i18n 中间件,以便你可以实现自己的自定义逻辑。其他的 routing 选项(例如 prefixDefaultLocale)不能与 routing: "manual" 一起配置。

你将负责编写自己的路由逻辑,或者手动执行 Astro 的 i18n 中间件与你自己的逻辑一起。

astro.config.mjs
import { defineConfig } from "astro/config"
export default defineConfig({
i18n: {
defaultLocale: "en",
locales: ["es", "en", "fr"],
routing: "manual"
}
})

Astro 为你的中间件提供了辅助函数,以便你可以控制自己的默认路由、异常、回退行为、错误捕获等:redirectToDefaultLocale()notFound()redirectToFallback()

src/middleware.js
import { defineMiddleware } from "astro:middleware";
import { redirectToDefaultLocale } from "astro:i18n"; // 在 `manual` 路由下可用的函数
export const onRequest = defineMiddleware(async (ctx, next) => {
if (ctx.url.startsWith("/about")) {
return next();
} else {
return redirectToDefaultLocale(302);
}
})

middleware 函数手动创建 Astro 的 i18n 中间件。这允许你扩展 Astro 的 i18n 路由,而不是完全替换它。

你可以将带有 路由选项middleware 与你自己的中间件结合使用,通过使用 sequence 工具来确定顺序运行 middleware

src/middleware.js
import { defineMiddleware, sequence } from "astro:middleware";
import { middleware } from "astro:i18n"; // Astro 自己的 i18n 路由配置
export const userMiddleware = defineMiddleware(async (ctx, next) => {
// 这个响应可能来自 Astro 的 i18n 中间件,它可能会返回一个 404
const response = await next();
// /about 页面是一个例外,我们想要渲染它
if (ctx.url.startsWith("/about")) {
return new Response("关于页面", {
status: 200
});
} else {
return response;
}
});
export const onRequest = sequence(
userMiddleware,
middleware({
redirectToDefaultLocale: false,
prefixDefaultLocale: true
})
)

添加于: astro@4.9.0

这个路由选项允许你为使用 @astrojs/node@astrojs/vercel 适配器并配置了 site 的服务端渲染项目,按语言自定义你的域名。

添加 i18n.domains 来将你支持的任何 locales 映射到自定义 url:

astro.config.mjs
import { defineConfig } from "astro/config"
export default defineConfig({
site: "https://example.com",
output: "server", // 必须,没有预渲染的页面
adapter: node({
mode: 'standalone',
}),
i18n: {
defaultLocale: "en",
locales: ["es", "en", "fr", "ja"],
routing: {
prefixDefaultLocale: false
},
domains: {
fr: "https://fr.example.com",
es: "https://example.es"
}
}
})

所有未映射的 locales 将遵循你的 prefixDefaultLocales 配置。然而,即使这个值为 false,你的 defaultLocale 的页面文件也必须存在于一个本地化文件夹中。对于上面的配置,需要一个 /en/ 文件夹。

根据上述配置:

  • 文件 /fr/about.astro 将创建 URL https://fr.example.com/about
  • 文件 /es/about.astro 将创建 URL https://example.es/about
  • 文件 /ja/about.astro 将创建 URL https://example.com/ja/about
  • 文件 /en/about.astro 将创建 URL https://example.com/about

上述 URL 也将通过 getAbsoluteLocaleUrl()getAbsoluteLocaleUrlList() 函数返回。

Astro 的 i18n 路由允许你配置一个回退路由策略。当一个页面在某种语言下不存在时(例如,一个尚未翻译的页面),你可以基于每种语言将用户从一个语言环境重定向到另一个语言环境,而不是显示 404 页面。这在你还没有为每个路由准备好页面,但仍想为访客提供一些内容时非常有用。

例如,下面的配置为任何缺失的 fr 路由设置 es 作为回退语言环境。这意味着,当 src/pages/fr/my-page.astro 不存在时,访问 example.com/fr/my-page/ 的用户将被重定向到 example.com/es/my-page/ 并显示那里的内容,而不是被带到 404 页面。

astro.config.mjs
import { defineConfig } from "astro/config"
export default defineConfig({
i18n: {
defaultLocale: "en",
locales: ["es", "en", "fr"],
fallback: {
fr: "es"
}
}
})

Astro 会确保 src/pages/es/ 中的每个页面都在 src/pages/fr 中构建一个对应页面。如果页面尚未存在,那么将创建一个页面,并将其重定向到相应的 es 路由。

自定义语言环境路径

段落标题 自定义语言环境路径

除了将你的网站支持的 locales 定义为字符串(例如 “en”, “pt-br”)之外,Astro 还允许你将任意数量的浏览器识别的语言 codes映射到自定义的 URL path 上。虽然语言环境可以是任何格式的字符串,只要它们与你的项目文件夹结构相对应,但 codes 必须遵循浏览器接受的语法。

locales 数组传递一个对象,并使用 path 键来定义自定义的 URL 前缀,使用 codes 来指示映射到此 URL 的语言。在这种情况下,你的 /[locale]/ 文件夹名称必须与 path 的值完全匹配,而你的 URL 将使用 path 的值来生成。

如果你支持一个语言的多个变体(例如 "fr""fr-BR""fr-CA"),并且你希望所有这些变体都映射到同一个 URL /fr/ 下,或者甚至完全自定义它(例如 /french/),这将非常有用:

astro.config.mjs
import { defineConfig } from "astro/config"
export default defineConfig({
i18n: {
defaultLocale: "en",
locales: ["es", "en", "fr"],
locales: ["es", "en", {
path: "french", // 不包含斜杠
codes: ["fr", "fr-BR", "fr-CA"]
}],
routing: {
prefixDefaultLocale: true
}
}
})

当使用来自 astro:i18n 虚拟模块 的函数根据你的配置计算有效的 URL 路径时(例如 getRelativeLocaleUrl()),请使用 path 作为 locale 的值

此功能有一些限制:

  • 必须设置 site 选项。
  • output 选项必须设置为 "server"
  • 不能有任何单独的预渲染页面。
  • 不支持适配器功能 functionPerRoute

Astro 依赖以下头信息以支持此功能:

请确保你的服务器代理/托管平台能够提供这些信息。获取这些头信息失败将导致一个 404(状态码)页面。

Astro 的 i18n 路由允许你在按需渲染的页面中访问两个用于浏览器语言检测的属性:Astro.preferredLocaleAstro.preferredLocaleList。所有页面,包括静态预渲染页面,都可以访问 Astro.currentLocale

这些属性结合了浏览器的 Accept-Language 头部信息和你的 locales(字符串或 codes),以自动遵循访问者的首选语言。

  • Astro.preferredLocale: 如果访客的浏览器首选语言环境包含在你的 locales 数组中,Astro 可以为你的访客计算一个首选语言环境。如果没有这样的匹配存在,此值为 undefined。

  • Astro.preferredLocaleList: 一个包含了浏览器请求并且你的网站支持的所有语言环境的数组。这将产生一个你的网站和访客之间所有兼容语言的列表。如果在你的 locales 数组中找不到浏览器请求的任何语言,该值为 []。如果浏览器没有指定任何首选语言,则此值将是 i18n.locales

  • Astro.currentLocale: 从当前 URL 计算的语言环境,使用你的 locales 配置指定的语法。如果 URL 不包含 /[locale]/ 前缀,则该值将默认为 i18n.defaultLocale

为了成功匹配访客的偏好,请使用与浏览器使用的模式相同的模式提供你的 codes

贡献

你有什么想法?

社区