If you want to read this article in English go here
I recently started learning Astro to create a dashboard-style project.
I really want to implement internationalization (i18n) in this project—the goal is that anyone can use it, regardless of language.
Problem
i18n support in Astro is very good. It works similar to Next.js or any other framework with routing based on file/folder structure.
So, if we want to have a page in English and the same page in Portuguese, we can organize our files like this:
. └── src/ └── pages/ ├── en/ │ ├── login.astro │ └── dashboard.astro └── pt-br/ ├── login.astro └── dashboard.astro
And each page has its own i18n strings—cool!
But this is where my problem starts: I don't want to clone all my pages; I just want to change the strings on those pages.
I need something like /[any-language-flag]/all-my-routes.
You may ask, "Why not use something like react-intl?" My answer is that I want to take full advantage of the Astro engine, especially for SSG/SSR, and avoid any client-side components. Generally, these frameworks use React Context, which is rendered only on the client side.
Attempts and Failures
First of all, I read Astro's recipe about i18n and checked out some community libraries to solve this problem.
The first library I tried was astro-i18next, and it looked like exactly what I needed!
Based on an array in the configuration file, astro-i18next generates my i18n pages at build time, so I only need to code once and not worry about cloning pages.
The problem is that astro-i18next appears to be archived or no longer maintained. There are a lot of problems and the last commit was over a year ago.
Solution
After trying other libraries (honorable mention for astro-i18n), I found Paraglide, and it was a game changer for my project.
I chose Paraglide because:
- It's type-safe, so I can use it with TypeScript and take advantage of autocomplete.
- It converts i18n strings to functions, so if a string key changes, my build will fail, catching errors early.
- Using i18n functions allows for better tree shaking, removing unused functions.
- There is a VS Code extension that improves the development experience.
Note: You can use Paraglide in a JS project too, and it also supports Next.js.
After installation and configuration, I used my messages like this:
--- import * as m from "../paraglide/messages.js"; --- <h1 id="m-hello-name-Alan">{m.hello({ name: "Alan" })}</h1>
However, this didn't solve my routing problem—I was still cloning my pages for each language I wanted to add.
To resolve this, I changed my project to use dynamic routes in the root route, so all my routes now start with the language flag.
My folder structure looked like this:
. └── src/ └── pages/ └── [lang]/ ├── login.astro └── dashboard.astro
After this change, Paraglide can automatically get the language of the route parameter:
- http://localhost:4321/en/login
- http://localhost:4321/pt-br/login
Now I can add a new language just by configuring it in astro.config.ts and translating my string file.
But I still have two problems to solve:
- When the user accesses http://localhost:4321/ for the first time without a language flag.
- If the user changes language on a specific route, I need to keep it on the same route (e.g. /en/create-account should redirect to /pt-br/create-account or /pt-br/criar- account).
Middleware for Language Redirection
To solve the first language redirection problem, I used Astro middleware.
In src/middleware/index.ts, I added this code:
import { defineMiddleware } from 'astro:middleware'; import { languageTag, setLanguageTag, type AvailableLanguageTag, } from '../paraglide/runtime'; export const onRequest = defineMiddleware((context, next) => { // Obter o idioma do parâmetro da URL const lang = context.params.lang; // Se mudou if (lang !== languageTag()) { setLanguageTag(lang as AvailableLanguageTag); // Redirecionar para o idioma alterado ou padrão (en) return context.redirect(`/${lang ?? 'en'}`); } return next(); });
Language Selector with Current Route
To keep the user on the same route when changing languages, I added this component:
--- import { languageTag } from '../paraglide/runtime'; const pathName = Astro.url.pathname.replace(`/${languageTag()}/`, ''); ---
Additionally, we can translate these messages too, using the second parameter in Paraglide's messages function:
- {m.goToLanguage(undefined, { languageTag: 'pt-br' })}
- {m.goToLanguage(undefined, { languageTag: 'en' })}
Considerations
I don't consider my solution to be the best, especially since I'm still learning Astro, so there may be other solutions. If you know of any, please comment, and I'll try :)
Thank you for reading this article! If you have any questions, comment, I'll be happy to answer.
The above is the detailed content of Creating dynamic routes to internationalization (i) with Astro Build. For more information, please follow other related articles on the PHP Chinese website!

Detailed explanation of JavaScript string replacement method and FAQ This article will explore two ways to replace string characters in JavaScript: internal JavaScript code and internal HTML for web pages. Replace string inside JavaScript code The most direct way is to use the replace() method: str = str.replace("find","replace"); This method replaces only the first match. To replace all matches, use a regular expression and add the global flag g: str = str.replace(/fi

This tutorial shows you how to integrate a custom Google Search API into your blog or website, offering a more refined search experience than standard WordPress theme search functions. It's surprisingly easy! You'll be able to restrict searches to y

Leverage jQuery for Effortless Web Page Layouts: 8 Essential Plugins jQuery simplifies web page layout significantly. This article highlights eight powerful jQuery plugins that streamline the process, particularly useful for manual website creation

So here you are, ready to learn all about this thing called AJAX. But, what exactly is it? The term AJAX refers to a loose grouping of technologies that are used to create dynamic, interactive web content. The term AJAX, originally coined by Jesse J

Core points This in JavaScript usually refers to an object that "owns" the method, but it depends on how the function is called. When there is no current object, this refers to the global object. In a web browser, it is represented by window. When calling a function, this maintains the global object; but when calling an object constructor or any of its methods, this refers to an instance of the object. You can change the context of this using methods such as call(), apply(), and bind(). These methods call the function using the given this value and parameters. JavaScript is an excellent programming language. A few years ago, this sentence was

jQuery is a great JavaScript framework. However, as with any library, sometimes it’s necessary to get under the hood to discover what’s going on. Perhaps it’s because you’re tracing a bug or are just curious about how jQuery achieves a particular UI

This post compiles helpful cheat sheets, reference guides, quick recipes, and code snippets for Android, Blackberry, and iPhone app development. No developer should be without them! Touch Gesture Reference Guide (PDF) A valuable resource for desig

Article discusses creating, publishing, and maintaining JavaScript libraries, focusing on planning, development, testing, documentation, and promotion strategies.


Hot AI Tools

Undresser.AI Undress
AI-powered app for creating realistic nude photos

AI Clothes Remover
Online AI tool for removing clothes from photos.

Undress AI Tool
Undress images for free

Clothoff.io
AI clothes remover

AI Hentai Generator
Generate AI Hentai for free.

Hot Article

Hot Tools

Atom editor mac version download
The most popular open source editor

mPDF
mPDF is a PHP library that can generate PDF files from UTF-8 encoded HTML. The original author, Ian Back, wrote mPDF to output PDF files "on the fly" from his website and handle different languages. It is slower than original scripts like HTML2FPDF and produces larger files when using Unicode fonts, but supports CSS styles etc. and has a lot of enhancements. Supports almost all languages, including RTL (Arabic and Hebrew) and CJK (Chinese, Japanese and Korean). Supports nested block-level elements (such as P, DIV),

SublimeText3 Linux new version
SublimeText3 Linux latest version

VSCode Windows 64-bit Download
A free and powerful IDE editor launched by Microsoft

ZendStudio 13.5.1 Mac
Powerful PHP integrated development environment
