」工欲善其事,必先利其器。「—孔子《論語.錄靈公》
首頁 > 程式設計 > 在 Astro Build 中為 i 建立動態路線語言

在 Astro Build 中為 i 建立動態路線語言

發佈於2024-08-30
瀏覽:693

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]刪除
最新教學 更多>
  • 如何克服PHP的功能重新定義限制?
    如何克服PHP的功能重新定義限制?
    克服PHP的函數重新定義限制在PHP中,多次定義一個相同名稱的函數是一個no-no。嘗試這樣做,如提供的代碼段所示,將導致可怕的“不能重新列出”錯誤。 但是,PHP工具腰帶中有一個隱藏的寶石:runkit擴展。它使您能夠靈活地重新定義函數。 runkit_function_renction_...
    程式設計 發佈於2025-06-29
  • 如何避免Go語言切片時的內存洩漏?
    如何避免Go語言切片時的內存洩漏?
    ,a [j:] ...雖然通常有效,但如果使用指針,可能會導致內存洩漏。這是因為原始的備份陣列保持完整,這意味著新切片外部指針引用的任何對象仍然可能佔據內存。 copy(a [i:] 對於k,n:= len(a)-j i,len(a); k
    程式設計 發佈於2025-06-29
  • 同實例無需轉儲複製MySQL數據庫方法
    同實例無需轉儲複製MySQL數據庫方法
    在同一實例上複製一個MySQL數據庫而無需轉儲在同一mySQL實例上複製數據庫,而無需創建InterMediate sqql script。以下方法為傳統的轉儲和IMPORT過程提供了更簡單的替代方法。 直接管道數據 MySQL手動概述了一種允許將mysqldump直接輸出到MySQL cli...
    程式設計 發佈於2025-06-29
  • 我可以將加密從McRypt遷移到OpenSSL,並使用OpenSSL遷移MCRYPT加密數據?
    我可以將加密從McRypt遷移到OpenSSL,並使用OpenSSL遷移MCRYPT加密數據?
    將我的加密庫從mcrypt升級到openssl 問題:是否可以將我的加密庫從McRypt升級到OpenSSL?如果是這樣,如何? 答案:是的,可以將您的Encryption庫從McRypt升級到OpenSSL。 可以使用openssl。 附加說明: [openssl_decrypt()函數要求...
    程式設計 發佈於2025-06-29
  • PHP與C++函數重載處理的區別
    PHP與C++函數重載處理的區別
    作為經驗豐富的C開發人員脫離謎題,您可能會遇到功能超載的概念。這個概念雖然在C中普遍,但在PHP中構成了獨特的挑戰。讓我們深入研究PHP功能過載的複雜性,並探索其提供的可能性。 在PHP中理解php的方法在PHP中,函數超載的概念(如C等語言)不存在。函數簽名僅由其名稱定義,而與他們的參數列表無關...
    程式設計 發佈於2025-06-29
  • 如何為PostgreSQL中的每個唯一標識符有效地檢索最後一行?
    如何為PostgreSQL中的每個唯一標識符有效地檢索最後一行?
    postgresql:為每個唯一標識符提取最後一行,在Postgresql中,您可能需要遇到與在數據庫中的每個不同標識相關的信息中提取信息的情況。考慮以下數據:[ 1 2014-02-01 kjkj 在數據集中的每個唯一ID中檢索最後一行的信息,您可以在操作員上使用Postgres的有效效率: ...
    程式設計 發佈於2025-06-29
  • 如何簡化PHP中的JSON解析以獲取多維陣列?
    如何簡化PHP中的JSON解析以獲取多維陣列?
    php 試圖在PHP中解析JSON數據的JSON可能具有挑戰性,尤其是在處理多維數組時。 To simplify the process, it's recommended to parse the JSON as an array rather than an object.To do...
    程式設計 發佈於2025-06-29
  • Java的Map.Entry和SimpleEntry如何簡化鍵值對管理?
    Java的Map.Entry和SimpleEntry如何簡化鍵值對管理?
    A Comprehensive Collection for Value Pairs: Introducing Java's Map.Entry and SimpleEntryIn Java, when defining a collection where each element com...
    程式設計 發佈於2025-06-29
  • 在Python中如何創建動態變量?
    在Python中如何創建動態變量?
    在Python 中,動態創建變量的功能可以是一種強大的工具,尤其是在使用複雜的數據結構或算法時,Dynamic Variable Creation的動態變量創建。 Python提供了幾種創造性的方法來實現這一目標。 利用dictionaries 一種有效的方法是利用字典。字典允許您動態創建密鑰並...
    程式設計 發佈於2025-06-29
  • 如何在Java字符串中有效替換多個子字符串?
    如何在Java字符串中有效替換多個子字符串?
    在java 中有效地替換多個substring,需要在需要替換一個字符串中的多個substring的情況下,很容易求助於重複應用字符串的刺激力量。 However, this can be inefficient for large strings or when working with nu...
    程式設計 發佈於2025-06-29
  • Python高效去除文本中HTML標籤方法
    Python高效去除文本中HTML標籤方法
    在Python中剝離HTML標籤,以獲取原始的文本表示Achieving Text-Only Extraction with Python's MLStripperTo streamline the stripping process, the Python standard librar...
    程式設計 發佈於2025-06-29
  • 版本5.6.5之前,使用current_timestamp與時間戳列的current_timestamp與時間戳列有什麼限制?
    版本5.6.5之前,使用current_timestamp與時間戳列的current_timestamp與時間戳列有什麼限制?
    在時間戳列上使用current_timestamp或MySQL版本中的current_timestamp或在5.6.5 此限制源於遺留實現的關注,這些限制需要對當前的_timestamp功能進行特定的實現。 創建表`foo`( `Productid` int(10)unsigned not ...
    程式設計 發佈於2025-06-29
  • 使用jQuery如何有效修改":after"偽元素的CSS屬性?
    使用jQuery如何有效修改":after"偽元素的CSS屬性?
    在jquery中了解偽元素的限制:訪問“ selector 嘗試修改“:”選擇器的CSS屬性時,您可能會遇到困難。 This is because pseudo-elements are not part of the DOM (Document Object Model) and are th...
    程式設計 發佈於2025-06-29
  • 如何使用PHP將斑點(圖像)正確插入MySQL?
    如何使用PHP將斑點(圖像)正確插入MySQL?
    essue VALUES('$this->image_id','file_get_contents($tmp_image)')";This code builds a string in PHP, but the function call fil...
    程式設計 發佈於2025-06-29
  • 如何同步迭代並從PHP中的兩個等級陣列打印值?
    如何同步迭代並從PHP中的兩個等級陣列打印值?
    同步的迭代和打印值來自相同大小的兩個數組使用兩個數組相等大小的selectbox時,一個包含country代碼的數組,另一個包含鄉村代碼,另一個包含其相應名稱的數組,可能會因不當提供了exply for for for the uncore for the forsion for for ytry...
    程式設計 發佈於2025-06-29

免責聲明: 提供的所有資源部分來自互聯網,如果有侵犯您的版權或其他權益,請說明詳細緣由並提供版權或權益證明然後發到郵箱:[email protected] 我們會在第一時間內為您處理。

Copyright© 2022 湘ICP备2022001581号-3