9router/src/shared/components/LanguageSwitcher.js

178 lines
7.1 KiB
JavaScript

"use client";
import { useState, useEffect, useRef } from "react";
import { createPortal } from "react-dom";
import { LOCALES, LOCALE_COOKIE, normalizeLocale } from "@/i18n/config";
import { reloadTranslations } from "@/i18n/runtime";
function getLocaleFromCookie() {
if (typeof document === "undefined") return "en";
const cookie = document.cookie
.split(";")
.find((c) => c.trim().startsWith(`${LOCALE_COOKIE}=`));
const value = cookie ? decodeURIComponent(cookie.split("=")[1]) : "en";
return normalizeLocale(value);
}
// Locale display names and flags - will be translated by runtime i18n
const getLocaleInfo = (locale) => {
const locales = {
"en": { name: "English", flag: "🇺🇸" },
"vi": { name: "Tiếng Việt", flag: "🇻🇳" },
"zh-CN": { name: "简体中文", flag: "🇨🇳" },
"zh-TW": { name: "繁體中文", flag: "🇹🇼" },
"ja": { name: "日本語", flag: "🇯🇵" },
"pt-BR": { name: "Português (Brasil)", flag: "🇧🇷" },
"pt-PT": { name: "Português (Portugal)", flag: "🇵🇹" },
"ko": { name: "한국어", flag: "🇰🇷" },
"es": { name: "Español", flag: "🇪🇸" },
"de": { name: "Deutsch", flag: "🇩🇪" },
"fr": { name: "Français", flag: "🇫🇷" },
"he": { name: "עברית", flag: "🇮🇱" },
"ar": { name: "العربية", flag: "🇸🇦" },
"ru": { name: "Русский", flag: "🇷🇺" },
"pl": { name: "Polski", flag: "🇵🇱" },
"cs": { name: "Čeština", flag: "🇨🇿" },
"nl": { name: "Nederlands", flag: "🇳🇱" },
"tr": { name: "Türkçe", flag: "🇹🇷" },
"uk": { name: "Українська", flag: "🇺🇦" },
"tl": { name: "Tagalog", flag: "🇵🇭" },
"id": { name: "Indonesia", flag: "🇮🇩" },
"th": { name: "ไทย", flag: "🇹🇭" },
"hi": { name: "हिन्दी", flag: "🇮🇳" },
"bn": { name: "বাংলা", flag: "🇧🇩" },
"ur": { name: "اردو", flag: "🇵🇰" },
"ro": { name: "Română", flag: "🇷🇴" },
"sv": { name: "Svenska", flag: "🇸🇪" },
"it": { name: "Italiano", flag: "🇮🇹" },
"el": { name: "Ελληνικά", flag: "🇬🇷" },
"hu": { name: "Magyar", flag: "🇭🇺" },
"fi": { name: "Suomi", flag: "🇫🇮" },
"da": { name: "Dansk", flag: "🇩🇰" },
"no": { name: "Norsk", flag: "🇳🇴" }
};
return locales[locale] || { name: locale, flag: "🌐" };
};
export default function LanguageSwitcher({ className = "" }) {
const [locale, setLocale] = useState("en");
const [isPending, setIsPending] = useState(false);
const [isOpen, setIsOpen] = useState(false);
const modalRef = useRef(null);
useEffect(() => {
setLocale(getLocaleFromCookie());
}, []);
// Close modal when clicking outside
useEffect(() => {
function handleClickOutside(event) {
if (modalRef.current && !modalRef.current.contains(event.target)) {
setIsOpen(false);
}
}
if (isOpen) {
document.addEventListener("mousedown", handleClickOutside);
return () => document.removeEventListener("mousedown", handleClickOutside);
}
}, [isOpen]);
const handleSetLocale = async (nextLocale) => {
if (nextLocale === locale || isPending) return;
setIsPending(true);
setIsOpen(false);
try {
await fetch("/api/locale", {
method: "POST",
headers: { "Content-Type": "application/json" },
body: JSON.stringify({ locale: nextLocale }),
});
// Reload translations without full page reload
await reloadTranslations();
setLocale(nextLocale);
} catch (err) {
console.error("Failed to set locale:", err);
} finally {
setIsPending(false);
}
};
return (
<div className={className}>
{/* Trigger button */}
<button
onClick={() => setIsOpen(!isOpen)}
disabled={isPending}
className="flex items-center gap-2 px-3 py-2 rounded-lg text-text-muted hover:text-text-main hover:bg-surface/60 transition-colors"
title="Language"
data-i18n-skip="true"
>
<span className="material-symbols-outlined text-[20px]">language</span>
<span className="text-sm font-medium">{getLocaleInfo(locale).name}</span>
<span className="text-lg">{getLocaleInfo(locale).flag}</span>
</button>
{/* Portal modal - renders at document.body to avoid parent layout constraints */}
{isOpen && createPortal(
<div className="fixed inset-0 z-50 flex items-center justify-center p-4" data-i18n-skip="true">
{/* Overlay */}
<div
className="absolute inset-0 bg-black/30 backdrop-blur-sm"
onClick={() => setIsOpen(false)}
/>
{/* Modal content */}
<div
ref={modalRef}
className="relative w-full bg-surface border border-black/10 dark:border-white/10 rounded-xl shadow-2xl animate-in fade-in zoom-in-95 duration-200 max-w-2xl flex flex-col max-h-[80vh]"
>
{/* Modal header */}
<div className="flex items-center justify-between p-3 border-b border-black/5 dark:border-white/5">
<h2 className="text-lg font-semibold text-text-main">Select Language</h2>
<button
onClick={() => setIsOpen(false)}
className="p-1.5 rounded-lg text-text-muted hover:bg-black/5 dark:hover:bg-white/5 transition-colors"
aria-label="Close"
>
<span className="material-symbols-outlined text-[20px]">close</span>
</button>
</div>
{/* Modal body - fixed grid columns, equal sizing */}
<div className="p-6 overflow-y-auto flex-1">
<div className="grid grid-cols-[repeat(auto-fill,minmax(100px,1fr))] gap-2">
{LOCALES.map((item) => {
const active = locale === item;
const info = getLocaleInfo(item);
return (
<button
key={item}
onClick={() => handleSetLocale(item)}
disabled={isPending}
className={`flex flex-col items-center justify-start gap-1 px-2 py-3 rounded-lg text-xs font-medium transition-colors w-full ${
active
? "bg-primary/15 text-primary ring-2 ring-primary"
: "text-text-main hover:bg-black/5 dark:hover:bg-white/5"
} ${isPending ? "opacity-70 cursor-wait" : ""}`}
title={info.name}
>
<span className="text-2xl">{info.flag}</span>
{/* Fixed 2-line height so all cards are uniform */}
<span className="text-center leading-tight line-clamp-2 h-8 flex items-center">{info.name}</span>
{active && (
<span className="material-symbols-outlined text-sm">check</span>
)}
</button>
);
})}
</div>
</div>
</div>
</div>,
document.body
)}
</div>
);
}