2026-05-16 00:18:22 +08:00
|
|
|
|
import React, {
|
|
|
|
|
|
createContext,
|
|
|
|
|
|
useCallback,
|
|
|
|
|
|
useContext,
|
|
|
|
|
|
useMemo,
|
|
|
|
|
|
useState,
|
|
|
|
|
|
} from "react";
|
|
|
|
|
|
|
2026-05-26 10:03:12 +08:00
|
|
|
|
export type Lang = "zh-CN" | "en" | "ja" | "ko" | "vi" | "id" | "ms";
|
2026-05-16 00:18:22 +08:00
|
|
|
|
|
|
|
|
|
|
type Dict = Record<string, string>;
|
|
|
|
|
|
|
2026-05-26 07:36:53 +08:00
|
|
|
|
const zhDict: Dict = {
|
2026-05-28 10:15:03 +08:00
|
|
|
|
brand: "ARK 资料库",
|
2026-05-26 07:36:53 +08:00
|
|
|
|
mainNav: "网站导航",
|
|
|
|
|
|
home: "首页",
|
|
|
|
|
|
all: "全部资料",
|
2026-05-28 15:31:45 +08:00
|
|
|
|
categories: "资料分类",
|
2026-05-26 07:36:53 +08:00
|
|
|
|
latest: "最新更新",
|
|
|
|
|
|
official: "官方推荐",
|
|
|
|
|
|
popular: "热门资料",
|
|
|
|
|
|
search: "搜索",
|
|
|
|
|
|
searchPlaceholder: "搜索资料...",
|
2026-05-30 13:55:52 +08:00
|
|
|
|
searchPanelPlaceholder: "搜索资料...",
|
2026-05-26 07:36:53 +08:00
|
|
|
|
searchNow: "立即搜索资料",
|
2026-05-28 18:51:55 +08:00
|
|
|
|
searchSubmit: "搜索",
|
|
|
|
|
|
cancel: "取消",
|
|
|
|
|
|
clear: "清除",
|
|
|
|
|
|
searchPanelHint: "支持搜索 标题・分类・标签・简介・文件类型・正文",
|
|
|
|
|
|
currentTags: "现有标签",
|
|
|
|
|
|
noTagsAvailable: "暂无可选择的标签。",
|
|
|
|
|
|
tagPostsTitle: "#{{tag}} 相关资料",
|
|
|
|
|
|
noTagPosts: "暂时找不到带有此标签的资料。",
|
2026-05-26 07:36:53 +08:00
|
|
|
|
viewAll: "查看全部",
|
2026-05-29 11:50:27 +08:00
|
|
|
|
backToTop: "回到顶部",
|
2026-05-26 07:36:53 +08:00
|
|
|
|
heroTitle: "ARK 官方数据库",
|
|
|
|
|
|
heroSub:
|
|
|
|
|
|
"集中、分类、管理 ARK 数据库,让你快速找到所需资源,推动社群共识与成长。",
|
|
|
|
|
|
categorySection: "资料分类",
|
|
|
|
|
|
officialSection: "官方推荐",
|
|
|
|
|
|
latestSection: "最新更新",
|
2026-05-29 15:42:16 +08:00
|
|
|
|
popularSection: "热门资料",
|
2026-05-26 07:36:53 +08:00
|
|
|
|
preview: "预览",
|
|
|
|
|
|
download: "下载",
|
2026-05-27 12:23:06 +08:00
|
|
|
|
downloading: "下载中…",
|
2026-05-29 11:50:27 +08:00
|
|
|
|
downloadOk: "下载完成",
|
|
|
|
|
|
downloadFail: "下载失败,请重试",
|
2026-05-30 18:14:58 +08:00
|
|
|
|
longPressImageSave: "长按图片保存到相册",
|
2026-05-29 23:49:59 +08:00
|
|
|
|
showMore: "展开全部",
|
|
|
|
|
|
showLess: "收起全部",
|
2026-05-26 07:36:53 +08:00
|
|
|
|
share: "分享",
|
|
|
|
|
|
langLabel: "语言",
|
|
|
|
|
|
admin: "后台",
|
|
|
|
|
|
login: "登录",
|
|
|
|
|
|
logout: "退出",
|
|
|
|
|
|
email: "邮箱",
|
|
|
|
|
|
password: "密码",
|
|
|
|
|
|
dashboard: "仪表盘",
|
|
|
|
|
|
resources: "资料管理",
|
|
|
|
|
|
newResource: "新增资料",
|
|
|
|
|
|
save: "保存",
|
|
|
|
|
|
title: "标题",
|
|
|
|
|
|
description: "简介",
|
|
|
|
|
|
type: "类型",
|
|
|
|
|
|
language: "语言",
|
|
|
|
|
|
category: "分类",
|
|
|
|
|
|
status: "状态",
|
|
|
|
|
|
public: "公开",
|
|
|
|
|
|
downloadable: "可下载",
|
|
|
|
|
|
recommended: "首页推荐",
|
|
|
|
|
|
cover: "封面图 URL",
|
|
|
|
|
|
fileUrl: "文件 URL",
|
|
|
|
|
|
externalUrl: "外部链接",
|
|
|
|
|
|
body: "文案内容",
|
|
|
|
|
|
badge: "推荐标签",
|
|
|
|
|
|
published: "已发布",
|
|
|
|
|
|
draft: "草稿",
|
|
|
|
|
|
archived: "归档",
|
|
|
|
|
|
noResults: "找不到符合的资料,请换个关键字或浏览分类。",
|
|
|
|
|
|
copyLink: "复制链接",
|
|
|
|
|
|
related: "相关资料",
|
|
|
|
|
|
total: "总资料",
|
|
|
|
|
|
views: "浏览",
|
|
|
|
|
|
downloads: "下载",
|
2026-05-26 10:03:12 +08:00
|
|
|
|
lang_zh_CN: "中文",
|
2026-05-26 07:36:53 +08:00
|
|
|
|
lang_en: "English",
|
|
|
|
|
|
lang_ja: "日本語",
|
|
|
|
|
|
lang_ko: "한국어",
|
|
|
|
|
|
lang_vi: "Tiếng Việt",
|
|
|
|
|
|
lang_id: "Bahasa Indonesia",
|
|
|
|
|
|
lang_ms: "Bahasa Melayu",
|
|
|
|
|
|
filterAll: "全部",
|
|
|
|
|
|
sortPublished: "发布时间",
|
|
|
|
|
|
type_ppt: "PPT",
|
|
|
|
|
|
type_music: "音乐",
|
|
|
|
|
|
type_video: "视频",
|
|
|
|
|
|
type_image: "图片",
|
|
|
|
|
|
type_pdf: "PDF",
|
|
|
|
|
|
type_link: "链接",
|
|
|
|
|
|
type_text: "文字",
|
|
|
|
|
|
type_archive: "压缩包",
|
|
|
|
|
|
type_zip: "ZIP",
|
|
|
|
|
|
adminLoginTitle: "管理后台登录",
|
|
|
|
|
|
adminEditResource: "编辑资料",
|
|
|
|
|
|
adminVideoFileHint:
|
|
|
|
|
|
"上传视频文件(MP4/WebM/MOV 等),类型请选择「视频」;保存后前台自动播放(默认静音,可点喇叭开声音)。",
|
|
|
|
|
|
adminStatTodayNew: "今日新增",
|
|
|
|
|
|
adminStatFavorites: "收藏",
|
|
|
|
|
|
adminMetricDownloads: "下载",
|
|
|
|
|
|
adminMetricFavorites: "收藏",
|
|
|
|
|
|
adminMetricViews: "浏览",
|
|
|
|
|
|
edit: "编辑",
|
|
|
|
|
|
backToList: "返回列表",
|
|
|
|
|
|
sortOrderLabel: "排序权重",
|
|
|
|
|
|
previewUrlLabel: "预览网址",
|
|
|
|
|
|
tagsCommaLabel: "标签(逗号分隔)",
|
|
|
|
|
|
uploadFile: "上传文件",
|
|
|
|
|
|
loading: "加载中…",
|
|
|
|
|
|
paginationPrev: "上一页",
|
|
|
|
|
|
paginationNext: "下一页",
|
|
|
|
|
|
listRange: "显示 {{from}}–{{to}},共 {{total}} 条",
|
|
|
|
|
|
pageIndicator: "{{c}} / {{p}} 页",
|
|
|
|
|
|
resourceLangFilter: "资料语言",
|
|
|
|
|
|
filterTagClear: "清除标签",
|
|
|
|
|
|
filterLanguageAll: "全部语言",
|
|
|
|
|
|
footerAdminLogin: "管理员登录",
|
|
|
|
|
|
adminSearchLogs: "搜索记录",
|
|
|
|
|
|
adminMetricShares: "分享",
|
|
|
|
|
|
adminSearchQuery: "查询词",
|
|
|
|
|
|
adminSearchTime: "时间",
|
|
|
|
|
|
adminSearchId: "编号",
|
2026-05-28 10:36:38 +08:00
|
|
|
|
favorites: "我的收藏",
|
|
|
|
|
|
favoritesComingSoon: "功能即将推出",
|
|
|
|
|
|
favoritesComingSoonDesc: "登入与收藏功能开发中,敬请期待。",
|
2026-05-28 16:19:21 +08:00
|
|
|
|
featureUnavailable: "未开放",
|
|
|
|
|
|
featureUnavailableDesc: "该功能暂未开放。",
|
|
|
|
|
|
confirm: "知道了",
|
2026-05-28 10:36:38 +08:00
|
|
|
|
backToHome: "返回首页",
|
2026-05-26 07:36:53 +08:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
const enDict: Dict = {
|
|
|
|
|
|
brand: "ARK Library",
|
|
|
|
|
|
mainNav: "Site menu",
|
|
|
|
|
|
home: "Home",
|
|
|
|
|
|
all: "All assets",
|
|
|
|
|
|
categories: "Categories",
|
|
|
|
|
|
latest: "Latest",
|
|
|
|
|
|
official: "Official picks",
|
|
|
|
|
|
popular: "Popular",
|
|
|
|
|
|
search: "Search",
|
|
|
|
|
|
searchPlaceholder: "Search resources...",
|
2026-05-30 13:55:52 +08:00
|
|
|
|
searchPanelPlaceholder: "Search assets...",
|
2026-05-26 07:36:53 +08:00
|
|
|
|
searchNow: "Search now",
|
2026-05-28 18:51:55 +08:00
|
|
|
|
searchSubmit: "Search",
|
|
|
|
|
|
cancel: "Cancel",
|
|
|
|
|
|
clear: "Clear",
|
|
|
|
|
|
searchPanelHint:
|
|
|
|
|
|
"Search supports title, category, tags, summary, file type, and body text.",
|
|
|
|
|
|
currentTags: "Available tags",
|
|
|
|
|
|
noTagsAvailable: "No tags available yet.",
|
|
|
|
|
|
tagPostsTitle: "#{{tag}} related posts",
|
|
|
|
|
|
noTagPosts: "No posts with this tag yet.",
|
2026-05-26 07:36:53 +08:00
|
|
|
|
viewAll: "View all",
|
2026-05-29 11:50:27 +08:00
|
|
|
|
backToTop: "Back to top",
|
2026-05-26 07:36:53 +08:00
|
|
|
|
heroTitle: "ARK Official Library",
|
|
|
|
|
|
heroSub:
|
|
|
|
|
|
"Centralize, organize, and manage the ARK library so you can find what you need fast and help the community grow together.",
|
|
|
|
|
|
categorySection: "Categories",
|
|
|
|
|
|
officialSection: "Official recommendations",
|
|
|
|
|
|
latestSection: "Latest updates",
|
2026-05-29 15:42:16 +08:00
|
|
|
|
popularSection: "Popular assets",
|
2026-05-26 07:36:53 +08:00
|
|
|
|
preview: "Preview",
|
|
|
|
|
|
download: "Download",
|
2026-05-27 12:23:06 +08:00
|
|
|
|
downloading: "Downloading…",
|
2026-05-29 11:50:27 +08:00
|
|
|
|
downloadOk: "Download complete",
|
|
|
|
|
|
downloadFail: "Download failed, please retry",
|
2026-05-30 18:14:58 +08:00
|
|
|
|
longPressImageSave: "Long-press image to save",
|
2026-05-29 23:49:59 +08:00
|
|
|
|
showMore: "Show all",
|
|
|
|
|
|
showLess: "Show less",
|
2026-05-26 07:36:53 +08:00
|
|
|
|
share: "Share",
|
|
|
|
|
|
langLabel: "Language",
|
|
|
|
|
|
admin: "Admin",
|
|
|
|
|
|
login: "Sign in",
|
|
|
|
|
|
logout: "Sign out",
|
|
|
|
|
|
email: "Email",
|
|
|
|
|
|
password: "Password",
|
|
|
|
|
|
dashboard: "Dashboard",
|
|
|
|
|
|
resources: "Resources",
|
|
|
|
|
|
newResource: "New resource",
|
|
|
|
|
|
save: "Save",
|
|
|
|
|
|
title: "Title",
|
|
|
|
|
|
description: "Description",
|
|
|
|
|
|
type: "Type",
|
|
|
|
|
|
language: "Language",
|
|
|
|
|
|
category: "Category",
|
|
|
|
|
|
status: "Status",
|
|
|
|
|
|
public: "Public",
|
|
|
|
|
|
downloadable: "Downloadable",
|
|
|
|
|
|
recommended: "Featured",
|
|
|
|
|
|
cover: "Cover image URL",
|
|
|
|
|
|
fileUrl: "File URL",
|
|
|
|
|
|
externalUrl: "External URL",
|
|
|
|
|
|
body: "Text body",
|
|
|
|
|
|
badge: "Badge label",
|
|
|
|
|
|
published: "Published",
|
|
|
|
|
|
draft: "Draft",
|
|
|
|
|
|
archived: "Archived",
|
|
|
|
|
|
noResults: "No results. Try another keyword or browse categories.",
|
|
|
|
|
|
copyLink: "Copy link",
|
|
|
|
|
|
related: "Related",
|
|
|
|
|
|
total: "Total items",
|
|
|
|
|
|
views: "Views",
|
|
|
|
|
|
downloads: "Downloads",
|
2026-05-26 10:03:12 +08:00
|
|
|
|
lang_zh_CN: "Chinese",
|
2026-05-26 07:36:53 +08:00
|
|
|
|
lang_en: "English",
|
|
|
|
|
|
lang_ja: "Japanese",
|
|
|
|
|
|
lang_ko: "Korean",
|
|
|
|
|
|
lang_vi: "Vietnamese",
|
|
|
|
|
|
lang_id: "Indonesian",
|
|
|
|
|
|
lang_ms: "Malay",
|
|
|
|
|
|
filterAll: "All types",
|
|
|
|
|
|
sortPublished: "Published date",
|
|
|
|
|
|
type_ppt: "PPT",
|
|
|
|
|
|
type_music: "Music",
|
|
|
|
|
|
type_video: "Video",
|
|
|
|
|
|
type_image: "Image",
|
|
|
|
|
|
type_pdf: "PDF",
|
|
|
|
|
|
type_link: "Link",
|
|
|
|
|
|
type_text: "Text",
|
|
|
|
|
|
type_archive: "Archive",
|
|
|
|
|
|
type_zip: "ZIP",
|
|
|
|
|
|
adminLoginTitle: "Admin sign in",
|
|
|
|
|
|
adminEditResource: "Edit resource",
|
|
|
|
|
|
adminVideoFileHint:
|
|
|
|
|
|
"Upload a video file (MP4/WebM/MOV, etc.) and set type to Video; the site will autoplay (muted by default — user can unmute).",
|
|
|
|
|
|
adminStatTodayNew: "New today",
|
|
|
|
|
|
adminStatFavorites: "Favorites",
|
|
|
|
|
|
adminMetricDownloads: "Downloads",
|
|
|
|
|
|
adminMetricFavorites: "Favorites",
|
|
|
|
|
|
adminMetricViews: "Views",
|
|
|
|
|
|
edit: "Edit",
|
|
|
|
|
|
backToList: "Back to list",
|
|
|
|
|
|
sortOrderLabel: "Sort order",
|
|
|
|
|
|
previewUrlLabel: "Preview URL",
|
|
|
|
|
|
tagsCommaLabel: "Tags (comma-separated)",
|
|
|
|
|
|
uploadFile: "Upload",
|
|
|
|
|
|
loading: "Loading…",
|
|
|
|
|
|
paginationPrev: "Previous",
|
|
|
|
|
|
paginationNext: "Next",
|
|
|
|
|
|
listRange: "Showing {{from}}–{{to}} of {{total}}",
|
|
|
|
|
|
pageIndicator: "Page {{c}} / {{p}}",
|
|
|
|
|
|
resourceLangFilter: "Resource language",
|
|
|
|
|
|
filterTagClear: "Clear tag",
|
|
|
|
|
|
filterLanguageAll: "All languages",
|
|
|
|
|
|
footerAdminLogin: "Admin sign-in",
|
|
|
|
|
|
adminSearchLogs: "Search logs",
|
|
|
|
|
|
adminMetricShares: "Shares",
|
|
|
|
|
|
adminSearchQuery: "Query",
|
|
|
|
|
|
adminSearchTime: "Time",
|
|
|
|
|
|
adminSearchId: "ID",
|
2026-05-28 10:36:38 +08:00
|
|
|
|
favorites: "My Favorites",
|
|
|
|
|
|
favoritesComingSoon: "Coming Soon",
|
|
|
|
|
|
favoritesComingSoonDesc:
|
|
|
|
|
|
"Sign-in and favorites are in development. Stay tuned.",
|
2026-05-28 16:19:21 +08:00
|
|
|
|
featureUnavailable: "Not available yet",
|
|
|
|
|
|
featureUnavailableDesc: "This feature is not available yet.",
|
|
|
|
|
|
confirm: "Got it",
|
2026-05-28 10:36:38 +08:00
|
|
|
|
backToHome: "Back to Home",
|
2026-05-26 07:36:53 +08:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
const languageNames: Record<Lang, Dict> = {
|
2026-05-26 10:03:12 +08:00
|
|
|
|
"zh-CN": {
|
|
|
|
|
|
lang_zh_CN: "中文",
|
2026-05-16 00:18:22 +08:00
|
|
|
|
lang_en: "English",
|
2026-05-26 07:36:53 +08:00
|
|
|
|
lang_ja: "日本語",
|
|
|
|
|
|
lang_ko: "한국어",
|
|
|
|
|
|
lang_vi: "Tiếng Việt",
|
|
|
|
|
|
lang_id: "Bahasa Indonesia",
|
|
|
|
|
|
lang_ms: "Bahasa Melayu",
|
2026-05-16 00:18:22 +08:00
|
|
|
|
},
|
|
|
|
|
|
en: {
|
2026-05-26 10:03:12 +08:00
|
|
|
|
lang_zh_CN: "Chinese",
|
2026-05-16 00:18:22 +08:00
|
|
|
|
lang_en: "English",
|
2026-05-26 07:36:53 +08:00
|
|
|
|
lang_ja: "Japanese",
|
|
|
|
|
|
lang_ko: "Korean",
|
|
|
|
|
|
lang_vi: "Vietnamese",
|
|
|
|
|
|
lang_id: "Indonesian",
|
|
|
|
|
|
lang_ms: "Malay",
|
|
|
|
|
|
},
|
|
|
|
|
|
ja: {
|
2026-05-28 10:16:38 +08:00
|
|
|
|
brand: "ARK ライブラリー",
|
2026-05-29 23:49:59 +08:00
|
|
|
|
showMore: "すべて表示",
|
|
|
|
|
|
showLess: "閉じる",
|
2026-05-26 10:03:12 +08:00
|
|
|
|
lang_zh_CN: "中国語",
|
2026-05-26 07:36:53 +08:00
|
|
|
|
lang_en: "英語",
|
|
|
|
|
|
lang_ja: "日本語",
|
|
|
|
|
|
lang_ko: "韓国語",
|
|
|
|
|
|
lang_vi: "ベトナム語",
|
|
|
|
|
|
lang_id: "インドネシア語",
|
|
|
|
|
|
lang_ms: "マレー語",
|
2026-05-16 00:18:22 +08:00
|
|
|
|
},
|
2026-05-26 07:36:53 +08:00
|
|
|
|
ko: {
|
2026-05-28 10:16:38 +08:00
|
|
|
|
brand: "ARK 라이브러리",
|
2026-05-29 23:49:59 +08:00
|
|
|
|
showMore: "모두 보기",
|
|
|
|
|
|
showLess: "접기",
|
2026-05-26 10:03:12 +08:00
|
|
|
|
lang_zh_CN: "중국어",
|
2026-05-26 07:36:53 +08:00
|
|
|
|
lang_en: "영어",
|
|
|
|
|
|
lang_ja: "일본어",
|
|
|
|
|
|
lang_ko: "한국어",
|
|
|
|
|
|
lang_vi: "베트남어",
|
|
|
|
|
|
lang_id: "인도네시아어",
|
|
|
|
|
|
lang_ms: "말레이어",
|
|
|
|
|
|
},
|
|
|
|
|
|
vi: {
|
2026-05-28 10:16:38 +08:00
|
|
|
|
brand: "Thư viện ARK",
|
2026-05-29 23:49:59 +08:00
|
|
|
|
showMore: "Xem tất cả",
|
|
|
|
|
|
showLess: "Thu gọn",
|
2026-05-26 10:03:12 +08:00
|
|
|
|
lang_zh_CN: "Tiếng Trung",
|
2026-05-26 07:36:53 +08:00
|
|
|
|
lang_en: "Tiếng Anh",
|
|
|
|
|
|
lang_ja: "Tiếng Nhật",
|
|
|
|
|
|
lang_ko: "Tiếng Hàn",
|
|
|
|
|
|
lang_vi: "Tiếng Việt",
|
|
|
|
|
|
lang_id: "Tiếng Indonesia",
|
|
|
|
|
|
lang_ms: "Tiếng Mã Lai",
|
|
|
|
|
|
},
|
|
|
|
|
|
id: {
|
2026-05-28 10:16:38 +08:00
|
|
|
|
brand: "Perpustakaan ARK",
|
2026-05-29 23:49:59 +08:00
|
|
|
|
showMore: "Lihat semua",
|
|
|
|
|
|
showLess: "Tutup",
|
2026-05-26 10:03:12 +08:00
|
|
|
|
lang_zh_CN: "Bahasa Tionghoa",
|
2026-05-26 07:36:53 +08:00
|
|
|
|
lang_en: "Bahasa Inggris",
|
|
|
|
|
|
lang_ja: "Bahasa Jepang",
|
|
|
|
|
|
lang_ko: "Bahasa Korea",
|
|
|
|
|
|
lang_vi: "Bahasa Vietnam",
|
|
|
|
|
|
lang_id: "Bahasa Indonesia",
|
|
|
|
|
|
lang_ms: "Bahasa Melayu",
|
|
|
|
|
|
},
|
|
|
|
|
|
ms: {
|
2026-05-28 10:16:38 +08:00
|
|
|
|
brand: "Perpustakaan ARK",
|
2026-05-29 23:49:59 +08:00
|
|
|
|
showMore: "Lihat semua",
|
|
|
|
|
|
showLess: "Tutup",
|
2026-05-26 10:03:12 +08:00
|
|
|
|
lang_zh_CN: "Bahasa Cina",
|
2026-05-26 07:36:53 +08:00
|
|
|
|
lang_en: "Bahasa Inggeris",
|
|
|
|
|
|
lang_ja: "Bahasa Jepun",
|
|
|
|
|
|
lang_ko: "Bahasa Korea",
|
|
|
|
|
|
lang_vi: "Bahasa Vietnam",
|
|
|
|
|
|
lang_id: "Bahasa Indonesia",
|
|
|
|
|
|
lang_ms: "Bahasa Melayu",
|
|
|
|
|
|
},
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
const dict: Record<Lang, Dict> = {
|
2026-05-26 10:03:12 +08:00
|
|
|
|
"zh-CN": { ...zhDict, ...languageNames["zh-CN"] },
|
2026-05-26 07:36:53 +08:00
|
|
|
|
en: { ...enDict, ...languageNames.en },
|
|
|
|
|
|
ja: { ...enDict, ...languageNames.ja },
|
|
|
|
|
|
ko: { ...enDict, ...languageNames.ko },
|
|
|
|
|
|
vi: { ...enDict, ...languageNames.vi },
|
|
|
|
|
|
id: { ...enDict, ...languageNames.id },
|
|
|
|
|
|
ms: { ...enDict, ...languageNames.ms },
|
2026-05-16 00:18:22 +08:00
|
|
|
|
};
|
|
|
|
|
|
|
2026-05-26 07:36:53 +08:00
|
|
|
|
/** Fixed locale lookup (admin UI uses Simplified Chinese). */
|
2026-05-16 00:18:22 +08:00
|
|
|
|
export function tLang(lang: Lang, key: string): string {
|
2026-05-26 07:36:53 +08:00
|
|
|
|
return dict[lang][key] || dict.en[key] || key;
|
2026-05-16 00:18:22 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
type Ctx = { lang: Lang; setLang: (l: Lang) => void; t: (k: string) => string };
|
|
|
|
|
|
|
|
|
|
|
|
const I18nCtx = createContext<Ctx | null>(null);
|
|
|
|
|
|
|
|
|
|
|
|
const LANG_KEY = "ark_lang";
|
|
|
|
|
|
|
|
|
|
|
|
export function I18nProvider({ children }: { children: React.ReactNode }) {
|
|
|
|
|
|
const [lang, setLangState] = useState<Lang>(() => {
|
2026-05-26 07:36:53 +08:00
|
|
|
|
const s = localStorage.getItem(LANG_KEY);
|
2026-05-26 10:03:12 +08:00
|
|
|
|
if (s === "zh" || s === "zh-TW") return "zh-CN";
|
2026-05-26 07:36:53 +08:00
|
|
|
|
if (
|
2026-05-26 10:03:12 +08:00
|
|
|
|
s === "zh-CN" ||
|
2026-05-26 07:36:53 +08:00
|
|
|
|
s === "en" ||
|
|
|
|
|
|
s === "ja" ||
|
|
|
|
|
|
s === "ko" ||
|
|
|
|
|
|
s === "vi" ||
|
|
|
|
|
|
s === "id" ||
|
|
|
|
|
|
s === "ms"
|
|
|
|
|
|
)
|
|
|
|
|
|
return s;
|
|
|
|
|
|
return "en";
|
2026-05-16 00:18:22 +08:00
|
|
|
|
});
|
|
|
|
|
|
const setLang = (l: Lang) => {
|
|
|
|
|
|
localStorage.setItem(LANG_KEY, l);
|
|
|
|
|
|
setLangState(l);
|
|
|
|
|
|
};
|
|
|
|
|
|
const t = useCallback(
|
2026-05-26 07:36:53 +08:00
|
|
|
|
(k: string) => dict[lang][k] || dict.en[k] || k,
|
2026-05-16 00:18:22 +08:00
|
|
|
|
[lang],
|
|
|
|
|
|
);
|
|
|
|
|
|
const v = useMemo(() => ({ lang, setLang, t }), [lang, t]);
|
|
|
|
|
|
return <I18nCtx.Provider value={v}>{children}</I18nCtx.Provider>;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
export function useI18n() {
|
|
|
|
|
|
const v = useContext(I18nCtx);
|
|
|
|
|
|
if (!v) throw new Error("I18nProvider missing");
|
|
|
|
|
|
return v;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
export function langQuery(lang: Lang) {
|
2026-05-26 10:03:12 +08:00
|
|
|
|
return lang;
|
2026-05-16 00:18:22 +08:00
|
|
|
|
}
|