${greeting}
\n${description}
\n \n使用您最喜欢的AI助手总结文档,并引用此页面和AI提供商
版本历史
- "更新 Solid useIntlayer API 用法以直接访问属性"v8.9.02026/5/4
- "Astro + Lit 的初始文档"v8.7.72026/4/24
此页面的内容已使用 AI 翻译。
查看英文原文的最新版本如果您有改善此文档的想法,请随时通过在GitHub上提交拉取请求来贡献。
文档的 GitHub 链接复制文档 Markdown 到剪贴板
使用 Intlayer 翻译您的 Astro + Lit 网站 | 国际化 (i18n)
目录
什么是 Intlayer?
Intlayer 是一个创新且开源的国际化 (i18n) 库,旨在简化 modern Web 应用程序的多语言支持。
使用 Intlayer,您可以:
- 轻松管理翻译:使用组件级的声明式词典。
- 动态本地化元数据、路由和内容。
- 确保 TypeScript 支持:通过自动生成的类型增强自动补全和错误检测。
- 受益于高级功能:如动态语言检测和语言切换。
在 Astro + Lit 中配置 Intlayer 的分步指南
在 GitHub 上查看应用程序模板。
第一步:安装依赖
使用您喜欢的包管理器安装所需的软件包:
复制代码到剪贴板
npm install intlayer astro-intlayer lit lit-intlayer @astrojs/litnpx intlayer initastro-intlayer 包含将 Intlayer 与 Vite 构建器集成的 Astro 集成插件,以及用于检测用户首选语言、管理 Cookie 和处理 URL 重定向的中间件。
lit 核心 Lit 软件包,用于构建快速且轻量级的 Web Components。
lit-intlayer 将 Intlayer 与 Lit 应用程序集成的软件包。它提供了基于
ReactiveController的钩子(useIntlayer、useLocale等),在语言更改时自动重新渲染 LitElement。@astrojs/lit 官方 Astro 集成,允许在 Astro 页面中使用 Lit 自定义元素。
第二步:配置您的项目
创建一个配置文件来定义您的应用程序语言:
复制代码到剪贴板
import { Locales, type IntlayerConfig } from "intlayer";const config: IntlayerConfig = { internationalization: { locales: [ Locales.ENGLISH, Locales.FRENCH, Locales.SPANISH, // 您的其他语言 ], defaultLocale: Locales.ENGLISH, },};export default config;通过此配置文件,您可以配置本地化 URL、中间件重定向、Cookie 名称、内容声明的位置和扩展名、在控制台中禁用 Intlayer 日志等。有关可用参数的完整列表,请参阅配置文档。
第三步:在 Astro 配置中集成 Intlayer
将 intlayer 插件和 Lit 集成添加到您的 Astro 配置中。
复制代码到剪贴板
// @ts-checkimport { intlayer } from "astro-intlayer";import lit from "@astrojs/lit";import { defineConfig } from "astro/config";// https://astro.build/configexport default defineConfig({ integrations: [intlayer(), lit()],});intlayer() 集成插件用于将 Intlayer 与 Astro 集成。它确保内容声明文件的构建并在开发模式下进行监视。它在 Astro 应用程序中定义 Intlayer 环境变量,并提供别名以优化性能。
lit() 集成允许在 Astro 页面内使用 Lit 自定义元素。
第四步:声明您的内容
创建并管理您的内容声明以存储翻译:
复制代码到剪贴板
import { t, type Dictionary } from "intlayer";const litDemoContent = { key: "lit-demo", content: { greeting: t({ en: "Hello World", fr: "Bonjour le monde", es: "Hola mundo", zh: "你好,世界", }), description: t({ en: "Welcome to my multilingual Astro + Lit site.", fr: "Bienvenue sur mon site Astro + Lit multilingue.", es: "Bienvenido a mi sitio Astro + Lit multilingüe.", zh: "欢迎来到我的多语言 Astro + Lit 网站。", }), },} satisfies Dictionary;export default litDemoContent;只要您的内容声明包含在contentDir(默认为./src)中,且匹配内容声明文件扩展名(默认为.content.{json,ts,tsx,js,jsx,mjs,cjs}),就可以在应用程序的任何位置定义。
有关更多信息,请参阅内容声明文档。
第五步:在 Astro 中使用内容
您可以使用 intlayer 导出的核心辅助函数直接在 .astro 文件中消费词典。您还应在每个页面添加 SEO 元数据(如 hreflang 和规范链接)。Lit 自定义元素通过客户端 <script> 导入并放置在 body 中。
复制代码到剪贴板
---import { getIntlayer, getLocaleFromPath, getLocalizedUrl, getHTMLTextDir, getPrefix, localeMap, defaultLocale, type LocalesValues,} from "intlayer";export const getStaticPaths = () => { return localeMap(({ locale }) => ({ params: { locale: getPrefix(locale).localePrefix }, }));};const locale = getLocaleFromPath(Astro.url.pathname) as LocalesValues;const { greeting } = getIntlayer("lit-demo", locale);---<!doctype html><html lang={locale} dir={getHTMLTextDir(locale)}> <head> <meta charset="utf-8" /> <meta name="viewport" content="width=device-width" /> <link rel="icon" type="image/svg+xml" href="/favicon.svg" /> <title>{greeting}</title> <!-- 规范链接 --> <link rel="canonical" href={new URL(getLocalizedUrl(Astro.url.pathname, locale), Astro.site)} /> <!-- Hreflang 链接 --> { localeMap(({ locale: mapLocale }) => ( <link rel="alternate" hreflang={mapLocale} href={new URL( getLocalizedUrl(Astro.url.pathname, mapLocale), Astro.site )} /> )) } <link rel="alternate" hreflang="x-default" href={new URL( getLocalizedUrl(Astro.url.pathname, defaultLocale), Astro.site )} /> </head> <body> <!-- Lit 自定义元素 - 接收服务端检测到的语言作为属性 --> <lit-demo locale={locale}></lit-demo> </body></html><script> import "../../components/lit/LitDemo";</script>关于路由设置的说明: 您使用的目录结构取决于
intlayer.config.ts中的middleware.routing设置:
prefix-no-default(默认):在根目录下保留默认语言(无前缀),并为其他语言添加前缀。使用[...locale]来捕获所有情况。prefix-all:所有 URL 都有语言前缀。如果您不需要单独处理根目录,可以使用标准的[locale]。search-param或no-prefix:不需要语言文件夹名。语言通过查询参数或 Cookie 处理。
第六步:创建 Lit 自定义元素
创建一个 Lit 自定义元素。在 connectedCallback 中使用服务端检测到的 locale 属性调用 installIntlayer 以初始化客户端翻译单例。
复制代码到剪贴板
import { LitElement, html } from "lit";import { installIntlayer, useIntlayer, useLocale } from "lit-intlayer";import { getLocalizedUrl, getLocaleName, type LocalesValues } from "intlayer";class LitDemo extends LitElement { static properties = { locale: { type: String }, }; locale: LocalesValues = "en" as LocalesValues; private _content = useIntlayer(this, "lit-demo"); private _localeCtrl = useLocale(this, { onLocaleChange: (newLocale: LocalesValues) => { window.location.href = getLocalizedUrl( window.location.pathname, newLocale ); }, }); override connectedCallback() { super.connectedCallback(); // 使用服务端检测到的语言进行初始化 installIntlayer({ locale: this.locale as any }); } override render() { const { greeting, description } = this._content; const { locale: currentLocale, availableLocales, setLocale, } = this._localeCtrl; return html` <div> <h1>${greeting}</h1> <p>${description}</p> <!-- 语言切换器在 LitElement 内部渲染 --> <div class="locale-switcher"> <span class="switcher-label">切换语言:</span> <div class="locale-buttons"> ${availableLocales.map( (localeItem) => html` <button class="locale-btn ${localeItem === currentLocale ? "active" : ""}" ?disabled=${localeItem === currentLocale} @click=${() => setLocale(localeItem)} > <span class="ls-own-name">${getLocaleName(localeItem)}</span> <span class="ls-current-name" >${getLocaleName(localeItem, currentLocale)}</span > <span class="ls-code">${localeItem.toUpperCase()}</span> </button> ` )} </div> </div> </div> `; }}customElements.define("lit-demo", LitDemo);locale属性从 Astro 页面(服务端检测)传递,用于在connectedCallback中初始化installIntlayer,这决定了元素内所有ReactiveController钩子的初始语言。
useIntlayer已注册为ReactiveController。当语言更改时,它会自动调度元素重新渲染,因此不需要额外的监听逻辑。
第七步:添加语言切换器
语言切换功能直接集成在 Lit 自定义元素的 render() 方法中(参见上面的第六步)。它使用来自 lit-intlayer 的 useLocale,并在用户选择新语言时导航到本地化 URL:
复制代码到剪贴板
// 在 LitElement 类内,延续第六步设置 useLocale 后:private _localeCtrl = useLocale(this, { onLocaleChange: (newLocale: LocalesValues) => { // 更改语言时导航到本地化 URL window.location.href = getLocalizedUrl(window.location.pathname, newLocale); },});override render() { const { locale: currentLocale, availableLocales, setLocale } = this._localeCtrl; return html` <div class="locale-switcher"> <span class="switcher-label">切换语言:</span> <div class="locale-buttons"> ${availableLocales.map( (localeItem) => html` <button class="locale-btn ${localeItem === currentLocale ? "active" : ""}" ?disabled=${localeItem === currentLocale} @click=${() => setLocale(localeItem)} > <span class="ls-own-name">${getLocaleName(localeItem)}</span> <span class="ls-current-name">${getLocaleName(localeItem, currentLocale)}</span> <span class="ls-code">${localeItem.toUpperCase()}</span> </button> ` )} </div> </div> `;}关于 Lit 响应性的说明:
useLocale返回一个ReactiveController。当setLocale被调用时,控制器会自动调度重新渲染,从而在没有手动操作 DOM 的情况下更新活动按钮状态。
关于持久性的说明: 使用
onLocaleChange通过window.location.href重定向,确保访问了新的语言 URL,允许 Intlayer 中间件设置语言 Cookie 并记住用户在未来访问时的首选语言。
第八步:站点地图和 Robots.txt
Intlayer 提供了实用工具来动态创建您的本地化站点地图和 robots.txt 文件。
站点地图
Intlayer 附带一个内置的站点地图生成器,可帮助您轻松为应用程序创建站点地图。它能够处理本地化路由,并为搜索引擎添加必要的元数据。
Intlayer 生成的站点地图支持xhtml:link命名空间(Hreflang XML 扩展)。与仅列出原始 URL 的默认站点地图生成器不同,Intlayer 会自动在页面的所有语言版本(例如/about、/about?lang=fr和/about?lang=es)之间创建所需的双向链接。这确保了搜索引擎能够正确索引并向合适的受众提供正确的语言版本。
创建 src/pages/sitemap.xml.ts 以生成包含所有本地化路由的站点地图。
复制代码到剪贴板
import type { APIRoute } from "astro";import { generateSitemap, type SitemapUrlEntry } from "intlayer";const pathList: SitemapUrlEntry[] = [ { path: "/", changefreq: "daily", priority: 1.0 }, { path: "/about", changefreq: "monthly", priority: 0.7 },];const SITE_URL = import.meta.env.SITE ?? "http://localhost:4321";export const GET: APIRoute = async ({ site }) => { const xmlOutput = generateSitemap(pathList, { siteUrl: SITE_URL }); return new Response(xmlOutput, { headers: { "Content-Type": "application/xml" }, });};Robots.txt
创建 src/pages/robots.txt.ts 以控制搜索引擎抓取。
复制代码到剪贴板
import type { APIRoute } from "astro";import { getMultilingualUrls } from "intlayer";const getAllMultilingualUrls = (urls: string[]) => urls.flatMap((url) => Object.values(getMultilingualUrls(url)) as string[]);const disallowedPaths = getAllMultilingualUrls(["/admin", "/private"]);export const GET: APIRoute = ({ site }) => { const robotsTxt = [ "User-agent: *", "Allow: /", ...disallowedPaths.map((path) => `Disallow: ${path}`), "", `Sitemap: ${new URL("/sitemap.xml", site).href}`, ].join("\n"); return new Response(robotsTxt, { headers: { "Content-Type": "text/plain" }, });};TypeScript 配置
Intlayer 使用模块扩展来利用 TypeScript,使您的代码库更加健壮。如果使用装饰器语法,请务必在编译选项中启用 experimentalDecorators。


确保您的 TypeScript 配置包含自动生成的类型。
复制代码到剪贴板
{ compilerOptions: { // ... experimentalDecorators: true, useDefineForClassFields: false, // 对于在 Lit 装饰器支持中是必需的 }, include: [ // ... 您现有的 TypeScript 配置 ".intlayer/**/*.ts", // 包含自动生成的类型 ],}Git 配置
建议忽略 Intlayer 生成的文件。这可以避免将它们提交到您的 Git 仓库。
为此,您可以将以下说明添加到 .gitignore 文件中:
复制代码到剪贴板
# 忽略 Intlayer 生成的文件.intlayer如果您想在字符串属性(如alt、title、href、aria-label等)中使用内容,可以使用函数的值,例如:
html复制代码复制代码到剪贴板
<img src="{content.image.src.value}" alt="{content.image.value}" /><img src="{content.image.src.toString()}" alt="{content.image.toString()}" /><img src="{String(content.image.src)}" alt="{String(content.image)}" />
VS Code 扩展
为了改善使用 Intlayer 的开发体验,您可以安装官方 Intlayer VS Code 扩展。
此扩展提供:
- 翻译键的自动补全。
- 缺失翻译的实时错误检测。
- 翻译内容的内联预览。
- 轻松创建和更新翻译的快速操作。
有关使用该扩展的更多信息,请参阅 Intlayer VS Code 扩展文档。
第十五步:提取组件中的内容(可选)
如果您有现有的代码库,转换数千个文件可能会非常耗时。
为了简化此过程,Intlayer 提供了 编译器 / 提取器 来转换您的组件并提取内容。
要进行设置,您可以在 intlayer.config.ts 文件中添加 compiler 部分:
复制代码到剪贴板
import { type IntlayerConfig } from "intlayer";const config: IntlayerConfig = { // ... 您的其他配置 compiler: { /** * 指示是否应启用编译器。 */ enabled: true, /** * 定义输出文件路径 */ output: ({ fileName, extension }) => `./${fileName}${extension}`, /** * 指示在转换后是否应保存组件。这样,编译器只需运行一次即可转换应用程序,然后即可将其删除。 */ saveComponents: false, /** * 字典键前缀 */ dictionaryKeyPrefix: "", },};export default config;运行提取器以转换组件并提取内容
复制代码到剪贴板
npx intlayer extract