@liberfi.io/i18n 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,80 @@
1
+ import * as react_i18next from 'react-i18next';
2
+ import { FallbackNs, UseTranslationOptions } from 'react-i18next';
3
+ export * from 'react-i18next';
4
+ import { i18n as i18n$1, FlatNamespace, KeyPrefix } from 'i18next';
5
+ export { createInstance, default as i18next } from 'i18next';
6
+ import * as react_jsx_runtime from 'react/jsx-runtime';
7
+ import * as react from 'react';
8
+ import { PropsWithChildren } from 'react';
9
+ import { L as LocaleCode, a as Language, R as Resources } from './types-DY-Gbo__.mjs';
10
+ export { E as ExtendLocaleMessages, b as LocaleEnum, c as LocaleMessages, e as en } from './types-DY-Gbo__.mjs';
11
+ export { defaultLanguages, defaultLng, defaultNS, i18nCookieKey, i18nLocalStorageKey } from './constant.mjs';
12
+ export { generatePath, getLocalePathFromPathname, parseI18nLang, removeLangPrefix } from './utils.mjs';
13
+
14
+ declare const i18n: i18n$1;
15
+
16
+ type BackendOptions = {
17
+ /**
18
+ * Load url for a given language and namespace
19
+ * @param lang language code
20
+ * @param ns namespace
21
+ * @returns url or urls
22
+ */
23
+ loadPath: (lang: LocaleCode, ns: string) => string | string[];
24
+ };
25
+
26
+ type LocaleContextState = {
27
+ /**
28
+ * Supported languages
29
+ */
30
+ languages: Language[];
31
+ /**
32
+ * Called before language change
33
+ * @param lang - new language code
34
+ * @returns
35
+ */
36
+ beforeLanguageChange: (lang: LocaleCode) => Promise<void>;
37
+ /**
38
+ * Called after language changed
39
+ * @param lang - new language code
40
+ * @returns
41
+ */
42
+ afterLanguageChange: (lang: LocaleCode) => Promise<void>;
43
+ };
44
+ declare const LocaleContext: react.Context<LocaleContextState>;
45
+ declare const useLocaleContext: () => LocaleContextState;
46
+
47
+ type LocaleProviderProps = PropsWithChildren<{
48
+ /** current locale */
49
+ locale?: LocaleCode;
50
+ /** current locale's resource */
51
+ resource?: Record<string, string>;
52
+ /** all synchronously loaded resources */
53
+ resources?: Resources;
54
+ /** supported languages, must be a subset of {@link defaultLanguages}, used when {@link languages} is not provided */
55
+ supportedLanguages?: LocaleCode[];
56
+ /** optional conversion function to modify the detected language code */
57
+ convertDetectedLanguage?: (lang: string) => LocaleCode;
58
+ /** options to load resources asynchronously */
59
+ backend?: BackendOptions;
60
+ } & Partial<LocaleContextState>>;
61
+ declare function LocaleProvider({ children, locale, resource, resources, backend, supportedLanguages, convertDetectedLanguage, languages: languagesProp, beforeLanguageChange, afterLanguageChange, }: LocaleProviderProps): react_jsx_runtime.JSX.Element;
62
+
63
+ type $Tuple<T> = readonly [T?, ...T[]];
64
+ /**
65
+ * Hook to get translation function
66
+ * @param ns
67
+ * @param options
68
+ * @returns
69
+ */
70
+ declare function useTranslation<Ns extends FlatNamespace | $Tuple<FlatNamespace> | undefined = undefined, KPrefix extends KeyPrefix<FallbackNs<Ns>> = undefined>(ns?: Ns, options?: UseTranslationOptions<KPrefix>): react_i18next.UseTranslationResponse<FallbackNs<Ns>, KPrefix>;
71
+
72
+ /**
73
+ * Hooks to get current locale code
74
+ * @returns current locale code
75
+ */
76
+ declare function useLocale(): LocaleCode;
77
+
78
+ declare function useChangeLocale(): (locale: LocaleCode) => Promise<void>;
79
+
80
+ export { Language, LocaleCode, LocaleContext, type LocaleContextState, LocaleProvider, type LocaleProviderProps, Resources, i18n, useChangeLocale, useLocale, useLocaleContext, useTranslation };
@@ -0,0 +1,80 @@
1
+ import * as react_i18next from 'react-i18next';
2
+ import { FallbackNs, UseTranslationOptions } from 'react-i18next';
3
+ export * from 'react-i18next';
4
+ import { i18n as i18n$1, FlatNamespace, KeyPrefix } from 'i18next';
5
+ export { createInstance, default as i18next } from 'i18next';
6
+ import * as react_jsx_runtime from 'react/jsx-runtime';
7
+ import * as react from 'react';
8
+ import { PropsWithChildren } from 'react';
9
+ import { L as LocaleCode, a as Language, R as Resources } from './types-DY-Gbo__.js';
10
+ export { E as ExtendLocaleMessages, b as LocaleEnum, c as LocaleMessages, e as en } from './types-DY-Gbo__.js';
11
+ export { defaultLanguages, defaultLng, defaultNS, i18nCookieKey, i18nLocalStorageKey } from './constant.js';
12
+ export { generatePath, getLocalePathFromPathname, parseI18nLang, removeLangPrefix } from './utils.js';
13
+
14
+ declare const i18n: i18n$1;
15
+
16
+ type BackendOptions = {
17
+ /**
18
+ * Load url for a given language and namespace
19
+ * @param lang language code
20
+ * @param ns namespace
21
+ * @returns url or urls
22
+ */
23
+ loadPath: (lang: LocaleCode, ns: string) => string | string[];
24
+ };
25
+
26
+ type LocaleContextState = {
27
+ /**
28
+ * Supported languages
29
+ */
30
+ languages: Language[];
31
+ /**
32
+ * Called before language change
33
+ * @param lang - new language code
34
+ * @returns
35
+ */
36
+ beforeLanguageChange: (lang: LocaleCode) => Promise<void>;
37
+ /**
38
+ * Called after language changed
39
+ * @param lang - new language code
40
+ * @returns
41
+ */
42
+ afterLanguageChange: (lang: LocaleCode) => Promise<void>;
43
+ };
44
+ declare const LocaleContext: react.Context<LocaleContextState>;
45
+ declare const useLocaleContext: () => LocaleContextState;
46
+
47
+ type LocaleProviderProps = PropsWithChildren<{
48
+ /** current locale */
49
+ locale?: LocaleCode;
50
+ /** current locale's resource */
51
+ resource?: Record<string, string>;
52
+ /** all synchronously loaded resources */
53
+ resources?: Resources;
54
+ /** supported languages, must be a subset of {@link defaultLanguages}, used when {@link languages} is not provided */
55
+ supportedLanguages?: LocaleCode[];
56
+ /** optional conversion function to modify the detected language code */
57
+ convertDetectedLanguage?: (lang: string) => LocaleCode;
58
+ /** options to load resources asynchronously */
59
+ backend?: BackendOptions;
60
+ } & Partial<LocaleContextState>>;
61
+ declare function LocaleProvider({ children, locale, resource, resources, backend, supportedLanguages, convertDetectedLanguage, languages: languagesProp, beforeLanguageChange, afterLanguageChange, }: LocaleProviderProps): react_jsx_runtime.JSX.Element;
62
+
63
+ type $Tuple<T> = readonly [T?, ...T[]];
64
+ /**
65
+ * Hook to get translation function
66
+ * @param ns
67
+ * @param options
68
+ * @returns
69
+ */
70
+ declare function useTranslation<Ns extends FlatNamespace | $Tuple<FlatNamespace> | undefined = undefined, KPrefix extends KeyPrefix<FallbackNs<Ns>> = undefined>(ns?: Ns, options?: UseTranslationOptions<KPrefix>): react_i18next.UseTranslationResponse<FallbackNs<Ns>, KPrefix>;
71
+
72
+ /**
73
+ * Hooks to get current locale code
74
+ * @returns current locale code
75
+ */
76
+ declare function useLocale(): LocaleCode;
77
+
78
+ declare function useChangeLocale(): (locale: LocaleCode) => Promise<void>;
79
+
80
+ export { Language, LocaleCode, LocaleContext, type LocaleContextState, LocaleProvider, type LocaleProviderProps, Resources, i18n, useChangeLocale, useLocale, useLocaleContext, useTranslation };
package/dist/index.js ADDED
@@ -0,0 +1,2 @@
1
+ 'use strict';var reactI18next=require('react-i18next'),i18next=require('i18next'),D=require('i18next-browser-languagedetector'),react=require('react'),jsxRuntime=require('react/jsx-runtime');function _interopDefault(e){return e&&e.__esModule?e:{default:e}}var i18next__default=/*#__PURE__*/_interopDefault(i18next);var D__default=/*#__PURE__*/_interopDefault(D);var p=(o=>(o.en="en",o.zh="zh",o.ja="ja",o.es="es",o.ko="ko",o.vi="vi",o.de="de",o.fr="fr",o.ru="ru",o.id="id",o.tr="tr",o.it="it",o.pt="pt",o.uk="uk",o.pl="pl",o.nl="nl",o))(p||{});var f=[{localCode:"en",displayName:"English"},{localCode:"zh",displayName:"\u4E2D\u6587"},{localCode:"ja",displayName:"\u65E5\u672C\u8A9E"},{localCode:"es",displayName:"Espa\xF1ol"},{localCode:"ko",displayName:"\uD55C\uAD6D\uC5B4"},{localCode:"vi",displayName:"Ti\u1EBFng Vi\u1EC7t"},{localCode:"de",displayName:"Deutsch"},{localCode:"fr",displayName:"Fran\xE7ais"},{localCode:"ru",displayName:"\u0420\u0443\u0441\u0441\u043A\u0438\u0439"},{localCode:"id",displayName:"Bahasa Indonesia"},{localCode:"tr",displayName:"T\xFCrk\xE7e"},{localCode:"it",displayName:"Italiano"},{localCode:"pt",displayName:"Portugu\xEAs"},{localCode:"uk",displayName:"\u0423\u043A\u0440\u0430\u0457\u043D\u0441\u044C\u043A\u0430"},{localCode:"pl",displayName:"Polski"},{localCode:"nl",displayName:"Nederlands"}],x="en",s="translation",T="liberfi_i18nLng",S="liberfi_i18nLng";var b={"common.cancel":"Cancel","common.confirm":"Confirm","common.ok":"OK","common.yes":"Yes","common.no":"No","common.all":"All","common.buy":"Buy","common.sell":"Sell","common.long":"Long","common.short":"Short","common.edit":"Edit","common.save":"Save","common.add":"Add","common.delete":"Delete","common.tips":"Tips","common.max":"Max","common.download":"Download","common.copy":"Copy","common.copy.failed":"Copy failed","common.copy.copied":"Copied","common.share":"Share","common.export":"Export"};var I={"mediaTrack.title":"Media Track","mediaTrack.description":"Media Track"};var R={...b,...I};var O=i18next.createInstance();O.use(D__default.default).use(reactI18next.initReactI18next).init({fallbackLng:x,ns:[s],defaultNS:s,interpolation:{escapeValue:false},detection:{lookupLocalStorage:T,lookupCookie:S,caches:["cookie","localStorage"],order:["cookie","localStorage","htmlTag","navigator"]},resources:{[x]:{[s]:R}}});var a=O;var g=class{constructor(e){this.options=e;this.cache=new Set;}cache;async fetchData(e){try{return await(await fetch(e)).json()}catch{return {}}}async loadLanguage(e,t){if(typeof this.options?.loadPath!="function")return;let n=this.options.loadPath(e,t);if(typeof n=="string"&&(n=[n]),!n.length)return;let i=n.filter(l=>!a.hasResourceBundle(e,t)||!this.cache.has(l)).map(async l=>{let m=await this.fetchData(l);a.addResourceBundle(e,t,m,true,true),this.cache.add(l);});await Promise.all(i);}};var C=react.createContext({languages:[],beforeLanguageChange:()=>Promise.resolve(),afterLanguageChange:()=>Promise.resolve()}),B=()=>{let r=react.useContext(C);if(!r)throw new Error("useLocaleContext must be used within a LocaleProvider");return r};function L(r,e,t){e=e||Object.values(p),t=t||"en";let n=/^([a-z]{2})/i,c=r?.match(n);if(!c)return t;let i=c[1];return e.includes(r)?r:e.includes(i)?i:t}function we(r,e){let t=$(r,e);return t?r.replace(new RegExp(`^/${t}(?=/)`),""):r}function $(r,e){let t=r.split("/")[1];return e=e||Object.values(p),e.includes(t)?t:null}function Te(r){let{path:e,locale:t,search:n}=r,c=n||(typeof window<"u"?window.location.search:""),i=$(e);return i?`${e}${c}`:(i=t||L(a.language),`/${i}${e}${c}`)}function Y({children:r,locale:e,resource:t,resources:n,backend:c,supportedLanguages:i,convertDetectedLanguage:l,languages:m,beforeLanguageChange:h,afterLanguageChange:y}){let[P,k]=react.useState(f),N=react.useRef(new g(c));react.useEffect(()=>{if(n){Object.entries(n).forEach(([o,d])=>{a.addResourceBundle(o,s,d,true,true);});return}t&&e&&a.addResourceBundle(e,s,t,true,true);},[e,t,n]),react.useEffect(()=>{e&&e!==a.language&&a.changeLanguage(e);},[e]),react.useEffect(()=>{Array.isArray(m)?k(m):Array.isArray(i)&&k(i.map(o=>f.find(d=>d.localCode===o)).filter(o=>!!o));},[i,m]),react.useEffect(()=>{(async()=>{let d=typeof l=="function"?l(a.language):L(a.language);await N.current.loadLanguage(d,s),d!==a.language&&await a.changeLanguage(d);})();},[a.language]);let v=react.useCallback(async o=>{await h?.(o),await N.current.loadLanguage(o,s);},[h]),w=react.useCallback(async o=>{y?.(o);},[y]),F=react.useMemo(()=>({languages:P,beforeLanguageChange:v,afterLanguageChange:w}),[P,v,w]);return jsxRuntime.jsx(C.Provider,{value:F,children:jsxRuntime.jsx(reactI18next.I18nextProvider,{i18n:a,defaultNS:s,children:r})})}function Q(r,e){let t=react.useContext(reactI18next.I18nContext);return reactI18next.useTranslation(r,{i18n:t?.i18n||a,...e})}function E(){let[r,e]=react.useState(a.language);return react.useEffect(()=>(a.on("languageChanged",e),()=>{a.off("languageChanged",e);}),[a]),r}function oe(){let{beforeLanguageChange:r,afterLanguageChange:e}=B();return react.useCallback(async n=>{await r(n),await a.changeLanguage(n),await e(n);},[r,e,a])}Object.defineProperty(exports,"createInstance",{enumerable:true,get:function(){return i18next.createInstance}});Object.defineProperty(exports,"i18next",{enumerable:true,get:function(){return i18next__default.default}});exports.LocaleContext=C;exports.LocaleEnum=p;exports.LocaleProvider=Y;exports.defaultLanguages=f;exports.defaultLng=x;exports.defaultNS=s;exports.en=R;exports.generatePath=Te;exports.getLocalePathFromPathname=$;exports.i18n=a;exports.i18nCookieKey=S;exports.i18nLocalStorageKey=T;exports.parseI18nLang=L;exports.removeLangPrefix=we;exports.useChangeLocale=oe;exports.useLocale=E;exports.useLocaleContext=B;exports.useTranslation=Q;Object.keys(reactI18next).forEach(function(k){if(k!=='default'&&!Object.prototype.hasOwnProperty.call(exports,k))Object.defineProperty(exports,k,{enumerable:true,get:function(){return reactI18next[k]}})});//# sourceMappingURL=index.js.map
2
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/types.ts","../src/constant.ts","../src/locale/module/common.ts","../src/locale/module/mediaTrack.ts","../src/locale/en.ts","../src/i18n.ts","../src/backend.ts","../src/context.ts","../src/utils.ts","../src/provider.tsx","../src/useTranslation.ts","../src/useLocale.ts","../src/useChangeLocale.ts"],"names":["LocaleEnum","defaultLanguages","defaultLng","defaultNS","i18nLocalStorageKey","i18nCookieKey","common","mediaTrack","en","i18n","createInstance","LanguageDetector","initReactI18next","i18n_default","Backend","options","url","lang","ns","paths","promises","path","data","LocaleContext","createContext","useLocaleContext","context","useContext","parseI18nLang","localeCodes","defaultLang","regex","match","matchLang","removeLangPrefix","pathname","localePath","getLocalePathFromPathname","locale","generatePath","params","search","searchUrl","LocaleProvider","children","resource","resources","backend","supportedLanguages","convertDetectedLanguage","languagesProp","beforeLanguageChange","afterLanguageChange","languages","setLanguages","useState","backendRef","useRef","useEffect","messages","localCode","l","item","beforeLanguageChangeHandler","useCallback","afterLanguageChangeHandler","contextValue","useMemo","jsx","I18nextProvider","useTranslation","I18nContext","_useTranslation","useLocale","localeCode","setLocaleCode","useChangeLocale"],"mappings":"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","file":"index.js","sourcesContent":["// import the original type declarations\nimport \"i18next\";\n// import all namespaces (for the default language, only)\nimport { en } from \"./locale/en\";\n\nexport enum LocaleEnum {\n /** English */\n en = \"en\",\n /** Chinese */\n zh = \"zh\",\n /** Japanese */\n ja = \"ja\",\n /** Spanish */\n es = \"es\",\n /** Korean */\n ko = \"ko\",\n /** Vietnamese */\n vi = \"vi\",\n /** German */\n de = \"de\",\n /** French */\n fr = \"fr\",\n /** Russian */\n ru = \"ru\",\n /** Indonesian */\n id = \"id\",\n /** Turkish */\n tr = \"tr\",\n /** Italian */\n it = \"it\",\n /** Portuguese */\n pt = \"pt\",\n /** Ukrainian */\n uk = \"uk\",\n /** Polish */\n pl = \"pl\",\n /** Dutch */\n nl = \"nl\",\n}\n\nexport type LocaleCode = keyof typeof LocaleEnum | (string & {});\n\nexport type Language = {\n localCode: LocaleCode;\n displayName: string;\n};\n\nexport type ExtendLocaleMessages = Record<`extend.${string}`, string>;\n\nexport type LocaleMessages = typeof en & ExtendLocaleMessages;\n\n// eslint-disable-next-line @typescript-eslint/no-empty-object-type\nexport type Resources<T extends {} = {}> = {\n [key in LocaleCode]?: Partial<LocaleMessages & T>;\n};\n\n// https://www.i18next.com/overview/typescript#create-a-declaration-file\n// Enhance the input parameter intelliSense for the t function.\ndeclare module \"i18next\" {\n // Extend CustomTypeOptions\n interface CustomTypeOptions {\n // custom namespace type, if you changed it\n defaultNS: \"translation\";\n // custom resources type\n resources: {\n translation: LocaleMessages;\n };\n }\n}\n","import { Language, LocaleEnum } from \"./types\";\n\nexport const defaultLanguages: Language[] = [\n { localCode: LocaleEnum.en, displayName: \"English\" }, // English\n { localCode: LocaleEnum.zh, displayName: \"中文\" }, // Chinese\n { localCode: LocaleEnum.ja, displayName: \"日本語\" }, // Japanese\n { localCode: LocaleEnum.es, displayName: \"Español\" }, // Spanish\n { localCode: LocaleEnum.ko, displayName: \"한국어\" }, // Korean\n { localCode: LocaleEnum.vi, displayName: \"Tiếng Việt\" }, // Vietnamese\n { localCode: LocaleEnum.de, displayName: \"Deutsch\" }, // German\n { localCode: LocaleEnum.fr, displayName: \"Français\" }, // French\n { localCode: LocaleEnum.ru, displayName: \"Русский\" }, // Russian\n { localCode: LocaleEnum.id, displayName: \"Bahasa Indonesia\" }, // Indonesian\n { localCode: LocaleEnum.tr, displayName: \"Türkçe\" }, // Turkish\n { localCode: LocaleEnum.it, displayName: \"Italiano\" }, // Italian\n { localCode: LocaleEnum.pt, displayName: \"Português\" }, // Portuguese\n { localCode: LocaleEnum.uk, displayName: \"Українська\" }, // Ukrainian\n { localCode: LocaleEnum.pl, displayName: \"Polski\" }, // Polish\n { localCode: LocaleEnum.nl, displayName: \"Nederlands\" }, // Dutch\n];\n\nexport const defaultLng = LocaleEnum.en;\n\nexport const defaultNS = \"translation\";\n\nexport const i18nLocalStorageKey = \"liberfi_i18nLng\";\n\nexport const i18nCookieKey = \"liberfi_i18nLng\";\n","export const common = {\n \"common.cancel\": \"Cancel\",\n \"common.confirm\": \"Confirm\",\n \"common.ok\": \"OK\",\n \"common.yes\": \"Yes\",\n \"common.no\": \"No\",\n \"common.all\": \"All\",\n \"common.buy\": \"Buy\",\n \"common.sell\": \"Sell\",\n \"common.long\": \"Long\",\n \"common.short\": \"Short\",\n \"common.edit\": \"Edit\",\n \"common.save\": \"Save\",\n \"common.add\": \"Add\",\n \"common.delete\": \"Delete\",\n \"common.tips\": \"Tips\",\n \"common.max\": \"Max\",\n \"common.download\": \"Download\",\n \"common.copy\": \"Copy\",\n \"common.copy.failed\": \"Copy failed\",\n \"common.copy.copied\": \"Copied\",\n \"common.share\": \"Share\",\n \"common.export\": \"Export\",\n};\n","export const mediaTrack = {\n \"mediaTrack.title\": \"Media Track\",\n \"mediaTrack.description\": \"Media Track\",\n};\n","import { common } from \"./module/common\";\nimport { mediaTrack } from \"./module/mediaTrack\";\n\nexport const en = {\n ...common,\n ...mediaTrack,\n};\n","import { initReactI18next } from \"react-i18next\";\nimport { i18n as I18nInstance, createInstance } from \"i18next\";\nimport LanguageDetector from \"i18next-browser-languagedetector\";\nimport {\n defaultLng,\n defaultNS,\n i18nCookieKey,\n i18nLocalStorageKey,\n} from \"./constant\";\nimport { en } from \"./locale/en\";\n\nconst i18n: I18nInstance = createInstance();\n\ni18n\n .use(LanguageDetector)\n .use(initReactI18next) // bind react-i18next to the instance\n .init({\n fallbackLng: defaultLng,\n ns: [defaultNS],\n defaultNS,\n interpolation: {\n escapeValue: false, // not needed for react!!\n },\n detection: {\n lookupLocalStorage: i18nLocalStorageKey,\n lookupCookie: i18nCookieKey,\n caches: [\"cookie\", \"localStorage\"],\n order: [\"cookie\", \"localStorage\", \"htmlTag\", \"navigator\"],\n },\n resources: {\n [defaultLng]: { [defaultNS]: en },\n },\n });\n\nexport default i18n;\n","import i18n from \"./i18n\";\nimport { LocaleCode } from \"./types\";\n\nexport type BackendOptions = {\n /**\n * Load url for a given language and namespace\n * @param lang language code\n * @param ns namespace\n * @returns url or urls\n */\n loadPath: (lang: LocaleCode, ns: string) => string | string[];\n};\n\n/**\n * Async resources loading through HTTP, resources must be placed in the site's public directory\n */\nexport class Backend {\n private readonly cache: Set<string>;\n\n constructor(private readonly options?: BackendOptions) {\n this.cache = new Set();\n }\n\n async fetchData(url: string): Promise<Record<string, string>> {\n try {\n const res = await fetch(url);\n return await res.json();\n } catch (error) {\n console.warn(\n `Failed to fetch locale resource bundle from ${url}:`,\n error,\n );\n return {};\n }\n }\n\n async loadLanguage(lang: LocaleCode, ns: string) {\n if (typeof this.options?.loadPath !== \"function\") return;\n\n // get load url paths\n let paths = this.options.loadPath(lang, ns);\n if (typeof paths === \"string\") paths = [paths];\n if (!paths.length) return;\n\n // filter out the paths that have already been loaded\n const urls = paths.filter((path) => {\n const loaded = i18n.hasResourceBundle(lang, ns);\n return !loaded || !this.cache.has(path);\n });\n\n const promises = urls.map(async (url) => {\n const data = await this.fetchData(url);\n i18n.addResourceBundle(lang, ns, data, true, true);\n this.cache.add(url);\n });\n await Promise.all(promises);\n }\n}\n","import { createContext, useContext } from \"react\";\nimport { Language, LocaleCode } from \"./types\";\n\nexport type LocaleContextState = {\n /**\n * Supported languages\n */\n languages: Language[];\n\n /**\n * Called before language change\n * @param lang - new language code\n * @returns\n */\n beforeLanguageChange: (lang: LocaleCode) => Promise<void>;\n /**\n * Called after language changed\n * @param lang - new language code\n * @returns\n */\n afterLanguageChange: (lang: LocaleCode) => Promise<void>;\n};\n\nexport const LocaleContext = createContext<LocaleContextState>({\n languages: [],\n beforeLanguageChange: () => Promise.resolve(),\n afterLanguageChange: () => Promise.resolve(),\n});\n\nexport const useLocaleContext = () => {\n const context = useContext(LocaleContext);\n if (!context) {\n throw new Error(\"useLocaleContext must be used within a LocaleProvider\");\n }\n return context;\n};\n","import i18n from \"./i18n\";\nimport { type LocaleCode, LocaleEnum } from \"./types\";\n\n/**\n * transform browser language to i18n locale codes\n * @param lang - browser language\n * @param localeCodes - locale codes to check\n * @param defaultLang - default locale code\n * @example\n * parseI18nLang('en-US') => 'en'\n * parseI18nLang('zh-CN') => 'zh'\n * parseI18nLang('zh-TW') => 'zh'\n * parseI18nLang('ja') => 'ja'\n * */\nexport function parseI18nLang(\n lang: string,\n localeCodes?: LocaleCode[],\n defaultLang?: LocaleCode,\n) {\n localeCodes = localeCodes || Object.values(LocaleEnum);\n defaultLang = defaultLang || LocaleEnum.en;\n\n const regex = /^([a-z]{2})/i;\n const match = lang?.match(regex);\n\n if (!match) {\n return defaultLang;\n }\n\n const matchLang = match[1];\n\n if (localeCodes.includes(lang)) {\n return lang;\n }\n\n if (localeCodes.includes(matchLang)) {\n return matchLang;\n }\n\n return defaultLang;\n}\n\n/**\n * remove lang prefix from pathname\n * @param pathname - pathname to remove lang prefix\n * @param localeCodes - locale codes to check\n * @example\n * removeLangPrefix('/en/perp/PERP_ETH_USDC') => '/perp/PERP_ETH_USDC'\n * removeLangPrefix('/en/markets') => '/markets'\n * removeLangPrefix('/perp/PERP_ETH_USDC') => '/perp/PERP_ETH_USDC'\n * removeLangPrefix('/markets') => '/markets'\n */\nexport function removeLangPrefix(pathname: string, localeCodes?: string[]) {\n const localePath = getLocalePathFromPathname(pathname, localeCodes);\n\n return localePath\n ? pathname.replace(new RegExp(`^/${localePath}(?=/)`), \"\")\n : pathname;\n}\n\n/**\n * get locale path from pathname\n * @param pathname - pathname to get locale path\n * @param localeCodes - locale codes to check\n * @example\n * getLocalePathFromPathname('/en/perp/PERP_ETH_USDC') => 'en'\n * getLocalePathFromPathname('/perp/PERP_ETH_USDC') => null\n * getLocalePathFromPathname('/en/markets') => 'en'\n * getLocalePathFromPathname('/markets') => null\n */\nexport function getLocalePathFromPathname(\n pathname: string,\n localeCodes?: string[],\n) {\n const locale = pathname.split(\"/\")[1];\n localeCodes = localeCodes || Object.values(LocaleEnum);\n return localeCodes.includes(locale as LocaleEnum) ? locale : null;\n}\n\n/**\n * Generate a localized path with proper locale prefix and search parameters\n *\n * This function ensures that the returned path includes the appropriate locale prefix.\n * If the path already contains a valid locale prefix, it returns the path as-is.\n * Otherwise, it prepends the specified locale or falls back to the current i18n language.\n *\n * @param params - Configuration object for path generation\n * @param params.path - The base pathname (e.g., '/markets', '/perp/PERP_ETH_USDC')\n * @param params.locale - Optional locale code to use as prefix. If not provided, uses i18n.language\n * @param params.search - Optional search query string. If not provided, uses window.location.search\n *\n * @returns A complete URL path with locale prefix and search parameters\n *\n * @example\n * generatePath({ path: '/markets' }) => '/en/markets?tab=spot'\n * generatePath({ path: '/en/markets', search: '?tab=futures' }) => '/en/markets?tab=futures'\n * generatePath({ path: '/perp/PERP_ETH_USDC', locale: 'zh' }) => '/zh/perp/PERP_ETH_USDC'\n * generatePath({ path: '/en/perp/PERP_ETH_USDC' }) => '/en/perp/PERP_ETH_USDC'\n */\nexport function generatePath(params: {\n path: string;\n locale?: string;\n search?: string;\n}) {\n const { path, locale, search } = params;\n const searchUrl =\n search || (typeof window !== \"undefined\" ? window.location.search : \"\");\n\n let localePath = getLocalePathFromPathname(path);\n\n // If path already contains a valid locale prefix, return it unchanged\n if (localePath) {\n return `${path}${searchUrl}`;\n }\n\n // Use provided locale or fall back to current i18n language\n localePath = locale || parseI18nLang(i18n.language);\n\n // Prepend locale prefix to path\n return `/${localePath}${path}${searchUrl}`;\n}\n","import {\n PropsWithChildren,\n useCallback,\n useEffect,\n useMemo,\n useRef,\n useState,\n} from \"react\";\nimport { I18nextProvider } from \"react-i18next\";\nimport { Backend, BackendOptions } from \"./backend\";\nimport { defaultLanguages, defaultNS } from \"./constant\";\nimport { LocaleContext, LocaleContextState } from \"./context\";\nimport i18n from \"./i18n\";\nimport { LocaleCode, Resources, Language } from \"./types\";\nimport { parseI18nLang } from \"./utils\";\n\nexport type LocaleProviderProps = PropsWithChildren<\n {\n /** current locale */\n locale?: LocaleCode;\n /** current locale's resource */\n resource?: Record<string, string>;\n /** all synchronously loaded resources */\n resources?: Resources;\n /** supported languages, must be a subset of {@link defaultLanguages}, used when {@link languages} is not provided */\n supportedLanguages?: LocaleCode[];\n /** optional conversion function to modify the detected language code */\n convertDetectedLanguage?: (lang: string) => LocaleCode;\n /** options to load resources asynchronously */\n backend?: BackendOptions;\n } & Partial<LocaleContextState>\n>;\n\nexport function LocaleProvider({\n children,\n locale,\n resource,\n resources,\n backend,\n supportedLanguages,\n convertDetectedLanguage,\n languages: languagesProp,\n beforeLanguageChange,\n afterLanguageChange,\n}: LocaleProviderProps) {\n // calculated supported languages\n const [languages, setLanguages] = useState<Language[]>(defaultLanguages);\n\n // backend instance to load resources asynchronously\n const backendRef = useRef(new Backend(backend!));\n\n // load resources synchronously\n useEffect(() => {\n if (resources) {\n Object.entries(resources).forEach(([locale, messages]) => {\n i18n.addResourceBundle(locale, defaultNS, messages, true, true);\n });\n return;\n }\n if (resource && locale) {\n i18n.addResourceBundle(locale, defaultNS, resource, true, true);\n }\n }, [locale, resource, resources]);\n\n // change language when locale changed\n useEffect(() => {\n if (locale && locale !== i18n.language) {\n i18n.changeLanguage(locale);\n }\n }, [locale]);\n\n // calculate supported languages\n useEffect(() => {\n if (Array.isArray(languagesProp)) {\n setLanguages(languagesProp);\n } else if (Array.isArray(supportedLanguages)) {\n setLanguages(\n supportedLanguages\n .map((localCode) =>\n defaultLanguages.find((l) => l.localCode === localCode),\n )\n .filter((item) => !!item),\n );\n }\n }, [supportedLanguages, languagesProp]);\n\n // if browser language is not a valid language, change language\n useEffect(() => {\n const fixLanguage = async () => {\n const lang =\n typeof convertDetectedLanguage === \"function\"\n ? convertDetectedLanguage(i18n.language)\n : parseI18nLang(i18n.language);\n await backendRef.current.loadLanguage(lang, defaultNS);\n if (lang !== i18n.language) {\n await i18n.changeLanguage(lang);\n }\n };\n fixLanguage();\n }, [i18n.language]);\n\n const beforeLanguageChangeHandler = useCallback(\n async (lang: LocaleCode) => {\n await beforeLanguageChange?.(lang);\n // load language before language changed\n await backendRef.current.loadLanguage(lang, defaultNS);\n },\n [beforeLanguageChange],\n );\n\n const afterLanguageChangeHandler = useCallback(\n async (lang: LocaleCode) => {\n afterLanguageChange?.(lang);\n },\n [afterLanguageChange],\n );\n\n const contextValue = useMemo<LocaleContextState>(() => {\n return {\n languages,\n beforeLanguageChange: beforeLanguageChangeHandler,\n afterLanguageChange: afterLanguageChangeHandler,\n };\n }, [languages, beforeLanguageChangeHandler, afterLanguageChangeHandler]);\n\n return (\n <LocaleContext.Provider value={contextValue}>\n <I18nextProvider i18n={i18n} defaultNS={defaultNS}>\n {children}\n </I18nextProvider>\n </LocaleContext.Provider>\n );\n}\n","import { useContext } from \"react\";\nimport {\n FallbackNs,\n useTranslation as _useTranslation,\n UseTranslationOptions,\n I18nContext,\n} from \"react-i18next\";\nimport { FlatNamespace, KeyPrefix } from \"i18next\";\nimport i18n from \"./i18n\";\n\ntype $Tuple<T> = readonly [T?, ...T[]];\n\n/**\n * Hook to get translation function\n * @param ns\n * @param options\n * @returns\n */\nexport function useTranslation<\n Ns extends FlatNamespace | $Tuple<FlatNamespace> | undefined = undefined,\n KPrefix extends KeyPrefix<FallbackNs<Ns>> = undefined,\n>(ns?: Ns, options?: UseTranslationOptions<KPrefix>) {\n const context = useContext(I18nContext);\n return _useTranslation(ns, { i18n: context?.i18n || i18n, ...options });\n}\n","import { useEffect, useState } from \"react\";\nimport i18n from \"./i18n\";\nimport { LocaleCode } from \"./types\";\n\n/**\n * Hooks to get current locale code\n * @returns current locale code\n */\nexport function useLocale() {\n const [localeCode, setLocaleCode] = useState<LocaleCode>(i18n.language);\n\n useEffect(() => {\n i18n.on(\"languageChanged\", setLocaleCode);\n return () => {\n i18n.off(\"languageChanged\", setLocaleCode);\n };\n }, [i18n]);\n\n return localeCode;\n}\n","import { useCallback } from \"react\";\nimport { useLocaleContext } from \"./context\";\nimport i18n from \"./i18n\";\nimport { LocaleCode } from \"./types\";\n\nexport function useChangeLocale() {\n const { beforeLanguageChange, afterLanguageChange } = useLocaleContext();\n\n const changeLocale = useCallback(\n async (locale: LocaleCode) => {\n await beforeLanguageChange(locale);\n await i18n.changeLanguage(locale);\n await afterLanguageChange(locale);\n },\n [beforeLanguageChange, afterLanguageChange, i18n],\n );\n\n return changeLocale;\n}\n"]}
package/dist/index.mjs ADDED
@@ -0,0 +1,2 @@
1
+ import {initReactI18next,I18nextProvider,I18nContext,useTranslation}from'react-i18next';export*from'react-i18next';import {createInstance}from'i18next';export{createInstance,default as i18next}from'i18next';import D from'i18next-browser-languagedetector';import {createContext,useContext,useState,useRef,useEffect,useCallback,useMemo}from'react';import {jsx}from'react/jsx-runtime';var p=(o=>(o.en="en",o.zh="zh",o.ja="ja",o.es="es",o.ko="ko",o.vi="vi",o.de="de",o.fr="fr",o.ru="ru",o.id="id",o.tr="tr",o.it="it",o.pt="pt",o.uk="uk",o.pl="pl",o.nl="nl",o))(p||{});var f=[{localCode:"en",displayName:"English"},{localCode:"zh",displayName:"\u4E2D\u6587"},{localCode:"ja",displayName:"\u65E5\u672C\u8A9E"},{localCode:"es",displayName:"Espa\xF1ol"},{localCode:"ko",displayName:"\uD55C\uAD6D\uC5B4"},{localCode:"vi",displayName:"Ti\u1EBFng Vi\u1EC7t"},{localCode:"de",displayName:"Deutsch"},{localCode:"fr",displayName:"Fran\xE7ais"},{localCode:"ru",displayName:"\u0420\u0443\u0441\u0441\u043A\u0438\u0439"},{localCode:"id",displayName:"Bahasa Indonesia"},{localCode:"tr",displayName:"T\xFCrk\xE7e"},{localCode:"it",displayName:"Italiano"},{localCode:"pt",displayName:"Portugu\xEAs"},{localCode:"uk",displayName:"\u0423\u043A\u0440\u0430\u0457\u043D\u0441\u044C\u043A\u0430"},{localCode:"pl",displayName:"Polski"},{localCode:"nl",displayName:"Nederlands"}],x="en",s="translation",T="liberfi_i18nLng",S="liberfi_i18nLng";var b={"common.cancel":"Cancel","common.confirm":"Confirm","common.ok":"OK","common.yes":"Yes","common.no":"No","common.all":"All","common.buy":"Buy","common.sell":"Sell","common.long":"Long","common.short":"Short","common.edit":"Edit","common.save":"Save","common.add":"Add","common.delete":"Delete","common.tips":"Tips","common.max":"Max","common.download":"Download","common.copy":"Copy","common.copy.failed":"Copy failed","common.copy.copied":"Copied","common.share":"Share","common.export":"Export"};var I={"mediaTrack.title":"Media Track","mediaTrack.description":"Media Track"};var R={...b,...I};var O=createInstance();O.use(D).use(initReactI18next).init({fallbackLng:x,ns:[s],defaultNS:s,interpolation:{escapeValue:false},detection:{lookupLocalStorage:T,lookupCookie:S,caches:["cookie","localStorage"],order:["cookie","localStorage","htmlTag","navigator"]},resources:{[x]:{[s]:R}}});var a=O;var g=class{constructor(e){this.options=e;this.cache=new Set;}cache;async fetchData(e){try{return await(await fetch(e)).json()}catch{return {}}}async loadLanguage(e,t){if(typeof this.options?.loadPath!="function")return;let n=this.options.loadPath(e,t);if(typeof n=="string"&&(n=[n]),!n.length)return;let i=n.filter(l=>!a.hasResourceBundle(e,t)||!this.cache.has(l)).map(async l=>{let m=await this.fetchData(l);a.addResourceBundle(e,t,m,true,true),this.cache.add(l);});await Promise.all(i);}};var C=createContext({languages:[],beforeLanguageChange:()=>Promise.resolve(),afterLanguageChange:()=>Promise.resolve()}),B=()=>{let r=useContext(C);if(!r)throw new Error("useLocaleContext must be used within a LocaleProvider");return r};function L(r,e,t){e=e||Object.values(p),t=t||"en";let n=/^([a-z]{2})/i,c=r?.match(n);if(!c)return t;let i=c[1];return e.includes(r)?r:e.includes(i)?i:t}function we(r,e){let t=$(r,e);return t?r.replace(new RegExp(`^/${t}(?=/)`),""):r}function $(r,e){let t=r.split("/")[1];return e=e||Object.values(p),e.includes(t)?t:null}function Te(r){let{path:e,locale:t,search:n}=r,c=n||(typeof window<"u"?window.location.search:""),i=$(e);return i?`${e}${c}`:(i=t||L(a.language),`/${i}${e}${c}`)}function Y({children:r,locale:e,resource:t,resources:n,backend:c,supportedLanguages:i,convertDetectedLanguage:l,languages:m,beforeLanguageChange:h,afterLanguageChange:y}){let[P,k]=useState(f),N=useRef(new g(c));useEffect(()=>{if(n){Object.entries(n).forEach(([o,d])=>{a.addResourceBundle(o,s,d,true,true);});return}t&&e&&a.addResourceBundle(e,s,t,true,true);},[e,t,n]),useEffect(()=>{e&&e!==a.language&&a.changeLanguage(e);},[e]),useEffect(()=>{Array.isArray(m)?k(m):Array.isArray(i)&&k(i.map(o=>f.find(d=>d.localCode===o)).filter(o=>!!o));},[i,m]),useEffect(()=>{(async()=>{let d=typeof l=="function"?l(a.language):L(a.language);await N.current.loadLanguage(d,s),d!==a.language&&await a.changeLanguage(d);})();},[a.language]);let v=useCallback(async o=>{await h?.(o),await N.current.loadLanguage(o,s);},[h]),w=useCallback(async o=>{y?.(o);},[y]),F=useMemo(()=>({languages:P,beforeLanguageChange:v,afterLanguageChange:w}),[P,v,w]);return jsx(C.Provider,{value:F,children:jsx(I18nextProvider,{i18n:a,defaultNS:s,children:r})})}function Q(r,e){let t=useContext(I18nContext);return useTranslation(r,{i18n:t?.i18n||a,...e})}function E(){let[r,e]=useState(a.language);return useEffect(()=>(a.on("languageChanged",e),()=>{a.off("languageChanged",e);}),[a]),r}function oe(){let{beforeLanguageChange:r,afterLanguageChange:e}=B();return useCallback(async n=>{await r(n),await a.changeLanguage(n),await e(n);},[r,e,a])}export{C as LocaleContext,p as LocaleEnum,Y as LocaleProvider,f as defaultLanguages,x as defaultLng,s as defaultNS,R as en,Te as generatePath,$ as getLocalePathFromPathname,a as i18n,S as i18nCookieKey,T as i18nLocalStorageKey,L as parseI18nLang,we as removeLangPrefix,oe as useChangeLocale,E as useLocale,B as useLocaleContext,Q as useTranslation};//# sourceMappingURL=index.mjs.map
2
+ //# sourceMappingURL=index.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/types.ts","../src/constant.ts","../src/locale/module/common.ts","../src/locale/module/mediaTrack.ts","../src/locale/en.ts","../src/i18n.ts","../src/backend.ts","../src/context.ts","../src/utils.ts","../src/provider.tsx","../src/useTranslation.ts","../src/useLocale.ts","../src/useChangeLocale.ts"],"names":["LocaleEnum","defaultLanguages","defaultLng","defaultNS","i18nLocalStorageKey","i18nCookieKey","common","mediaTrack","en","i18n","createInstance","LanguageDetector","initReactI18next","i18n_default","Backend","options","url","lang","ns","paths","promises","path","data","LocaleContext","createContext","useLocaleContext","context","useContext","parseI18nLang","localeCodes","defaultLang","regex","match","matchLang","removeLangPrefix","pathname","localePath","getLocalePathFromPathname","locale","generatePath","params","search","searchUrl","LocaleProvider","children","resource","resources","backend","supportedLanguages","convertDetectedLanguage","languagesProp","beforeLanguageChange","afterLanguageChange","languages","setLanguages","useState","backendRef","useRef","useEffect","messages","localCode","l","item","beforeLanguageChangeHandler","useCallback","afterLanguageChangeHandler","contextValue","useMemo","jsx","I18nextProvider","useTranslation","I18nContext","_useTranslation","useLocale","localeCode","setLocaleCode","useChangeLocale"],"mappings":"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","file":"index.mjs","sourcesContent":["// import the original type declarations\nimport \"i18next\";\n// import all namespaces (for the default language, only)\nimport { en } from \"./locale/en\";\n\nexport enum LocaleEnum {\n /** English */\n en = \"en\",\n /** Chinese */\n zh = \"zh\",\n /** Japanese */\n ja = \"ja\",\n /** Spanish */\n es = \"es\",\n /** Korean */\n ko = \"ko\",\n /** Vietnamese */\n vi = \"vi\",\n /** German */\n de = \"de\",\n /** French */\n fr = \"fr\",\n /** Russian */\n ru = \"ru\",\n /** Indonesian */\n id = \"id\",\n /** Turkish */\n tr = \"tr\",\n /** Italian */\n it = \"it\",\n /** Portuguese */\n pt = \"pt\",\n /** Ukrainian */\n uk = \"uk\",\n /** Polish */\n pl = \"pl\",\n /** Dutch */\n nl = \"nl\",\n}\n\nexport type LocaleCode = keyof typeof LocaleEnum | (string & {});\n\nexport type Language = {\n localCode: LocaleCode;\n displayName: string;\n};\n\nexport type ExtendLocaleMessages = Record<`extend.${string}`, string>;\n\nexport type LocaleMessages = typeof en & ExtendLocaleMessages;\n\n// eslint-disable-next-line @typescript-eslint/no-empty-object-type\nexport type Resources<T extends {} = {}> = {\n [key in LocaleCode]?: Partial<LocaleMessages & T>;\n};\n\n// https://www.i18next.com/overview/typescript#create-a-declaration-file\n// Enhance the input parameter intelliSense for the t function.\ndeclare module \"i18next\" {\n // Extend CustomTypeOptions\n interface CustomTypeOptions {\n // custom namespace type, if you changed it\n defaultNS: \"translation\";\n // custom resources type\n resources: {\n translation: LocaleMessages;\n };\n }\n}\n","import { Language, LocaleEnum } from \"./types\";\n\nexport const defaultLanguages: Language[] = [\n { localCode: LocaleEnum.en, displayName: \"English\" }, // English\n { localCode: LocaleEnum.zh, displayName: \"中文\" }, // Chinese\n { localCode: LocaleEnum.ja, displayName: \"日本語\" }, // Japanese\n { localCode: LocaleEnum.es, displayName: \"Español\" }, // Spanish\n { localCode: LocaleEnum.ko, displayName: \"한국어\" }, // Korean\n { localCode: LocaleEnum.vi, displayName: \"Tiếng Việt\" }, // Vietnamese\n { localCode: LocaleEnum.de, displayName: \"Deutsch\" }, // German\n { localCode: LocaleEnum.fr, displayName: \"Français\" }, // French\n { localCode: LocaleEnum.ru, displayName: \"Русский\" }, // Russian\n { localCode: LocaleEnum.id, displayName: \"Bahasa Indonesia\" }, // Indonesian\n { localCode: LocaleEnum.tr, displayName: \"Türkçe\" }, // Turkish\n { localCode: LocaleEnum.it, displayName: \"Italiano\" }, // Italian\n { localCode: LocaleEnum.pt, displayName: \"Português\" }, // Portuguese\n { localCode: LocaleEnum.uk, displayName: \"Українська\" }, // Ukrainian\n { localCode: LocaleEnum.pl, displayName: \"Polski\" }, // Polish\n { localCode: LocaleEnum.nl, displayName: \"Nederlands\" }, // Dutch\n];\n\nexport const defaultLng = LocaleEnum.en;\n\nexport const defaultNS = \"translation\";\n\nexport const i18nLocalStorageKey = \"liberfi_i18nLng\";\n\nexport const i18nCookieKey = \"liberfi_i18nLng\";\n","export const common = {\n \"common.cancel\": \"Cancel\",\n \"common.confirm\": \"Confirm\",\n \"common.ok\": \"OK\",\n \"common.yes\": \"Yes\",\n \"common.no\": \"No\",\n \"common.all\": \"All\",\n \"common.buy\": \"Buy\",\n \"common.sell\": \"Sell\",\n \"common.long\": \"Long\",\n \"common.short\": \"Short\",\n \"common.edit\": \"Edit\",\n \"common.save\": \"Save\",\n \"common.add\": \"Add\",\n \"common.delete\": \"Delete\",\n \"common.tips\": \"Tips\",\n \"common.max\": \"Max\",\n \"common.download\": \"Download\",\n \"common.copy\": \"Copy\",\n \"common.copy.failed\": \"Copy failed\",\n \"common.copy.copied\": \"Copied\",\n \"common.share\": \"Share\",\n \"common.export\": \"Export\",\n};\n","export const mediaTrack = {\n \"mediaTrack.title\": \"Media Track\",\n \"mediaTrack.description\": \"Media Track\",\n};\n","import { common } from \"./module/common\";\nimport { mediaTrack } from \"./module/mediaTrack\";\n\nexport const en = {\n ...common,\n ...mediaTrack,\n};\n","import { initReactI18next } from \"react-i18next\";\nimport { i18n as I18nInstance, createInstance } from \"i18next\";\nimport LanguageDetector from \"i18next-browser-languagedetector\";\nimport {\n defaultLng,\n defaultNS,\n i18nCookieKey,\n i18nLocalStorageKey,\n} from \"./constant\";\nimport { en } from \"./locale/en\";\n\nconst i18n: I18nInstance = createInstance();\n\ni18n\n .use(LanguageDetector)\n .use(initReactI18next) // bind react-i18next to the instance\n .init({\n fallbackLng: defaultLng,\n ns: [defaultNS],\n defaultNS,\n interpolation: {\n escapeValue: false, // not needed for react!!\n },\n detection: {\n lookupLocalStorage: i18nLocalStorageKey,\n lookupCookie: i18nCookieKey,\n caches: [\"cookie\", \"localStorage\"],\n order: [\"cookie\", \"localStorage\", \"htmlTag\", \"navigator\"],\n },\n resources: {\n [defaultLng]: { [defaultNS]: en },\n },\n });\n\nexport default i18n;\n","import i18n from \"./i18n\";\nimport { LocaleCode } from \"./types\";\n\nexport type BackendOptions = {\n /**\n * Load url for a given language and namespace\n * @param lang language code\n * @param ns namespace\n * @returns url or urls\n */\n loadPath: (lang: LocaleCode, ns: string) => string | string[];\n};\n\n/**\n * Async resources loading through HTTP, resources must be placed in the site's public directory\n */\nexport class Backend {\n private readonly cache: Set<string>;\n\n constructor(private readonly options?: BackendOptions) {\n this.cache = new Set();\n }\n\n async fetchData(url: string): Promise<Record<string, string>> {\n try {\n const res = await fetch(url);\n return await res.json();\n } catch (error) {\n console.warn(\n `Failed to fetch locale resource bundle from ${url}:`,\n error,\n );\n return {};\n }\n }\n\n async loadLanguage(lang: LocaleCode, ns: string) {\n if (typeof this.options?.loadPath !== \"function\") return;\n\n // get load url paths\n let paths = this.options.loadPath(lang, ns);\n if (typeof paths === \"string\") paths = [paths];\n if (!paths.length) return;\n\n // filter out the paths that have already been loaded\n const urls = paths.filter((path) => {\n const loaded = i18n.hasResourceBundle(lang, ns);\n return !loaded || !this.cache.has(path);\n });\n\n const promises = urls.map(async (url) => {\n const data = await this.fetchData(url);\n i18n.addResourceBundle(lang, ns, data, true, true);\n this.cache.add(url);\n });\n await Promise.all(promises);\n }\n}\n","import { createContext, useContext } from \"react\";\nimport { Language, LocaleCode } from \"./types\";\n\nexport type LocaleContextState = {\n /**\n * Supported languages\n */\n languages: Language[];\n\n /**\n * Called before language change\n * @param lang - new language code\n * @returns\n */\n beforeLanguageChange: (lang: LocaleCode) => Promise<void>;\n /**\n * Called after language changed\n * @param lang - new language code\n * @returns\n */\n afterLanguageChange: (lang: LocaleCode) => Promise<void>;\n};\n\nexport const LocaleContext = createContext<LocaleContextState>({\n languages: [],\n beforeLanguageChange: () => Promise.resolve(),\n afterLanguageChange: () => Promise.resolve(),\n});\n\nexport const useLocaleContext = () => {\n const context = useContext(LocaleContext);\n if (!context) {\n throw new Error(\"useLocaleContext must be used within a LocaleProvider\");\n }\n return context;\n};\n","import i18n from \"./i18n\";\nimport { type LocaleCode, LocaleEnum } from \"./types\";\n\n/**\n * transform browser language to i18n locale codes\n * @param lang - browser language\n * @param localeCodes - locale codes to check\n * @param defaultLang - default locale code\n * @example\n * parseI18nLang('en-US') => 'en'\n * parseI18nLang('zh-CN') => 'zh'\n * parseI18nLang('zh-TW') => 'zh'\n * parseI18nLang('ja') => 'ja'\n * */\nexport function parseI18nLang(\n lang: string,\n localeCodes?: LocaleCode[],\n defaultLang?: LocaleCode,\n) {\n localeCodes = localeCodes || Object.values(LocaleEnum);\n defaultLang = defaultLang || LocaleEnum.en;\n\n const regex = /^([a-z]{2})/i;\n const match = lang?.match(regex);\n\n if (!match) {\n return defaultLang;\n }\n\n const matchLang = match[1];\n\n if (localeCodes.includes(lang)) {\n return lang;\n }\n\n if (localeCodes.includes(matchLang)) {\n return matchLang;\n }\n\n return defaultLang;\n}\n\n/**\n * remove lang prefix from pathname\n * @param pathname - pathname to remove lang prefix\n * @param localeCodes - locale codes to check\n * @example\n * removeLangPrefix('/en/perp/PERP_ETH_USDC') => '/perp/PERP_ETH_USDC'\n * removeLangPrefix('/en/markets') => '/markets'\n * removeLangPrefix('/perp/PERP_ETH_USDC') => '/perp/PERP_ETH_USDC'\n * removeLangPrefix('/markets') => '/markets'\n */\nexport function removeLangPrefix(pathname: string, localeCodes?: string[]) {\n const localePath = getLocalePathFromPathname(pathname, localeCodes);\n\n return localePath\n ? pathname.replace(new RegExp(`^/${localePath}(?=/)`), \"\")\n : pathname;\n}\n\n/**\n * get locale path from pathname\n * @param pathname - pathname to get locale path\n * @param localeCodes - locale codes to check\n * @example\n * getLocalePathFromPathname('/en/perp/PERP_ETH_USDC') => 'en'\n * getLocalePathFromPathname('/perp/PERP_ETH_USDC') => null\n * getLocalePathFromPathname('/en/markets') => 'en'\n * getLocalePathFromPathname('/markets') => null\n */\nexport function getLocalePathFromPathname(\n pathname: string,\n localeCodes?: string[],\n) {\n const locale = pathname.split(\"/\")[1];\n localeCodes = localeCodes || Object.values(LocaleEnum);\n return localeCodes.includes(locale as LocaleEnum) ? locale : null;\n}\n\n/**\n * Generate a localized path with proper locale prefix and search parameters\n *\n * This function ensures that the returned path includes the appropriate locale prefix.\n * If the path already contains a valid locale prefix, it returns the path as-is.\n * Otherwise, it prepends the specified locale or falls back to the current i18n language.\n *\n * @param params - Configuration object for path generation\n * @param params.path - The base pathname (e.g., '/markets', '/perp/PERP_ETH_USDC')\n * @param params.locale - Optional locale code to use as prefix. If not provided, uses i18n.language\n * @param params.search - Optional search query string. If not provided, uses window.location.search\n *\n * @returns A complete URL path with locale prefix and search parameters\n *\n * @example\n * generatePath({ path: '/markets' }) => '/en/markets?tab=spot'\n * generatePath({ path: '/en/markets', search: '?tab=futures' }) => '/en/markets?tab=futures'\n * generatePath({ path: '/perp/PERP_ETH_USDC', locale: 'zh' }) => '/zh/perp/PERP_ETH_USDC'\n * generatePath({ path: '/en/perp/PERP_ETH_USDC' }) => '/en/perp/PERP_ETH_USDC'\n */\nexport function generatePath(params: {\n path: string;\n locale?: string;\n search?: string;\n}) {\n const { path, locale, search } = params;\n const searchUrl =\n search || (typeof window !== \"undefined\" ? window.location.search : \"\");\n\n let localePath = getLocalePathFromPathname(path);\n\n // If path already contains a valid locale prefix, return it unchanged\n if (localePath) {\n return `${path}${searchUrl}`;\n }\n\n // Use provided locale or fall back to current i18n language\n localePath = locale || parseI18nLang(i18n.language);\n\n // Prepend locale prefix to path\n return `/${localePath}${path}${searchUrl}`;\n}\n","import {\n PropsWithChildren,\n useCallback,\n useEffect,\n useMemo,\n useRef,\n useState,\n} from \"react\";\nimport { I18nextProvider } from \"react-i18next\";\nimport { Backend, BackendOptions } from \"./backend\";\nimport { defaultLanguages, defaultNS } from \"./constant\";\nimport { LocaleContext, LocaleContextState } from \"./context\";\nimport i18n from \"./i18n\";\nimport { LocaleCode, Resources, Language } from \"./types\";\nimport { parseI18nLang } from \"./utils\";\n\nexport type LocaleProviderProps = PropsWithChildren<\n {\n /** current locale */\n locale?: LocaleCode;\n /** current locale's resource */\n resource?: Record<string, string>;\n /** all synchronously loaded resources */\n resources?: Resources;\n /** supported languages, must be a subset of {@link defaultLanguages}, used when {@link languages} is not provided */\n supportedLanguages?: LocaleCode[];\n /** optional conversion function to modify the detected language code */\n convertDetectedLanguage?: (lang: string) => LocaleCode;\n /** options to load resources asynchronously */\n backend?: BackendOptions;\n } & Partial<LocaleContextState>\n>;\n\nexport function LocaleProvider({\n children,\n locale,\n resource,\n resources,\n backend,\n supportedLanguages,\n convertDetectedLanguage,\n languages: languagesProp,\n beforeLanguageChange,\n afterLanguageChange,\n}: LocaleProviderProps) {\n // calculated supported languages\n const [languages, setLanguages] = useState<Language[]>(defaultLanguages);\n\n // backend instance to load resources asynchronously\n const backendRef = useRef(new Backend(backend!));\n\n // load resources synchronously\n useEffect(() => {\n if (resources) {\n Object.entries(resources).forEach(([locale, messages]) => {\n i18n.addResourceBundle(locale, defaultNS, messages, true, true);\n });\n return;\n }\n if (resource && locale) {\n i18n.addResourceBundle(locale, defaultNS, resource, true, true);\n }\n }, [locale, resource, resources]);\n\n // change language when locale changed\n useEffect(() => {\n if (locale && locale !== i18n.language) {\n i18n.changeLanguage(locale);\n }\n }, [locale]);\n\n // calculate supported languages\n useEffect(() => {\n if (Array.isArray(languagesProp)) {\n setLanguages(languagesProp);\n } else if (Array.isArray(supportedLanguages)) {\n setLanguages(\n supportedLanguages\n .map((localCode) =>\n defaultLanguages.find((l) => l.localCode === localCode),\n )\n .filter((item) => !!item),\n );\n }\n }, [supportedLanguages, languagesProp]);\n\n // if browser language is not a valid language, change language\n useEffect(() => {\n const fixLanguage = async () => {\n const lang =\n typeof convertDetectedLanguage === \"function\"\n ? convertDetectedLanguage(i18n.language)\n : parseI18nLang(i18n.language);\n await backendRef.current.loadLanguage(lang, defaultNS);\n if (lang !== i18n.language) {\n await i18n.changeLanguage(lang);\n }\n };\n fixLanguage();\n }, [i18n.language]);\n\n const beforeLanguageChangeHandler = useCallback(\n async (lang: LocaleCode) => {\n await beforeLanguageChange?.(lang);\n // load language before language changed\n await backendRef.current.loadLanguage(lang, defaultNS);\n },\n [beforeLanguageChange],\n );\n\n const afterLanguageChangeHandler = useCallback(\n async (lang: LocaleCode) => {\n afterLanguageChange?.(lang);\n },\n [afterLanguageChange],\n );\n\n const contextValue = useMemo<LocaleContextState>(() => {\n return {\n languages,\n beforeLanguageChange: beforeLanguageChangeHandler,\n afterLanguageChange: afterLanguageChangeHandler,\n };\n }, [languages, beforeLanguageChangeHandler, afterLanguageChangeHandler]);\n\n return (\n <LocaleContext.Provider value={contextValue}>\n <I18nextProvider i18n={i18n} defaultNS={defaultNS}>\n {children}\n </I18nextProvider>\n </LocaleContext.Provider>\n );\n}\n","import { useContext } from \"react\";\nimport {\n FallbackNs,\n useTranslation as _useTranslation,\n UseTranslationOptions,\n I18nContext,\n} from \"react-i18next\";\nimport { FlatNamespace, KeyPrefix } from \"i18next\";\nimport i18n from \"./i18n\";\n\ntype $Tuple<T> = readonly [T?, ...T[]];\n\n/**\n * Hook to get translation function\n * @param ns\n * @param options\n * @returns\n */\nexport function useTranslation<\n Ns extends FlatNamespace | $Tuple<FlatNamespace> | undefined = undefined,\n KPrefix extends KeyPrefix<FallbackNs<Ns>> = undefined,\n>(ns?: Ns, options?: UseTranslationOptions<KPrefix>) {\n const context = useContext(I18nContext);\n return _useTranslation(ns, { i18n: context?.i18n || i18n, ...options });\n}\n","import { useEffect, useState } from \"react\";\nimport i18n from \"./i18n\";\nimport { LocaleCode } from \"./types\";\n\n/**\n * Hooks to get current locale code\n * @returns current locale code\n */\nexport function useLocale() {\n const [localeCode, setLocaleCode] = useState<LocaleCode>(i18n.language);\n\n useEffect(() => {\n i18n.on(\"languageChanged\", setLocaleCode);\n return () => {\n i18n.off(\"languageChanged\", setLocaleCode);\n };\n }, [i18n]);\n\n return localeCode;\n}\n","import { useCallback } from \"react\";\nimport { useLocaleContext } from \"./context\";\nimport i18n from \"./i18n\";\nimport { LocaleCode } from \"./types\";\n\nexport function useChangeLocale() {\n const { beforeLanguageChange, afterLanguageChange } = useLocaleContext();\n\n const changeLocale = useCallback(\n async (locale: LocaleCode) => {\n await beforeLanguageChange(locale);\n await i18n.changeLanguage(locale);\n await afterLanguageChange(locale);\n },\n [beforeLanguageChange, afterLanguageChange, i18n],\n );\n\n return changeLocale;\n}\n"]}
@@ -0,0 +1,25 @@
1
+ ,en
2
+ "common.cancel","Cancel"
3
+ "common.confirm","Confirm"
4
+ "common.ok","OK"
5
+ "common.yes","Yes"
6
+ "common.no","No"
7
+ "common.all","All"
8
+ "common.buy","Buy"
9
+ "common.sell","Sell"
10
+ "common.long","Long"
11
+ "common.short","Short"
12
+ "common.edit","Edit"
13
+ "common.save","Save"
14
+ "common.add","Add"
15
+ "common.delete","Delete"
16
+ "common.tips","Tips"
17
+ "common.max","Max"
18
+ "common.download","Download"
19
+ "common.copy","Copy"
20
+ "common.copy.failed","Copy failed"
21
+ "common.copy.copied","Copied"
22
+ "common.share","Share"
23
+ "common.export","Export"
24
+ "mediaTrack.title","Media Track"
25
+ "mediaTrack.description","Media Track"
@@ -0,0 +1,26 @@
1
+ {
2
+ "common.cancel": "Cancel",
3
+ "common.confirm": "Confirm",
4
+ "common.ok": "OK",
5
+ "common.yes": "Yes",
6
+ "common.no": "No",
7
+ "common.all": "All",
8
+ "common.buy": "Buy",
9
+ "common.sell": "Sell",
10
+ "common.long": "Long",
11
+ "common.short": "Short",
12
+ "common.edit": "Edit",
13
+ "common.save": "Save",
14
+ "common.add": "Add",
15
+ "common.delete": "Delete",
16
+ "common.tips": "Tips",
17
+ "common.max": "Max",
18
+ "common.download": "Download",
19
+ "common.copy": "Copy",
20
+ "common.copy.failed": "Copy failed",
21
+ "common.copy.copied": "Copied",
22
+ "common.share": "Share",
23
+ "common.export": "Export",
24
+ "mediaTrack.title": "Media Track",
25
+ "mediaTrack.description": "Media Track"
26
+ }
@@ -0,0 +1,81 @@
1
+ declare const en: {
2
+ "mediaTrack.title": string;
3
+ "mediaTrack.description": string;
4
+ "common.cancel": string;
5
+ "common.confirm": string;
6
+ "common.ok": string;
7
+ "common.yes": string;
8
+ "common.no": string;
9
+ "common.all": string;
10
+ "common.buy": string;
11
+ "common.sell": string;
12
+ "common.long": string;
13
+ "common.short": string;
14
+ "common.edit": string;
15
+ "common.save": string;
16
+ "common.add": string;
17
+ "common.delete": string;
18
+ "common.tips": string;
19
+ "common.max": string;
20
+ "common.download": string;
21
+ "common.copy": string;
22
+ "common.copy.failed": string;
23
+ "common.copy.copied": string;
24
+ "common.share": string;
25
+ "common.export": string;
26
+ };
27
+
28
+ declare enum LocaleEnum {
29
+ /** English */
30
+ en = "en",
31
+ /** Chinese */
32
+ zh = "zh",
33
+ /** Japanese */
34
+ ja = "ja",
35
+ /** Spanish */
36
+ es = "es",
37
+ /** Korean */
38
+ ko = "ko",
39
+ /** Vietnamese */
40
+ vi = "vi",
41
+ /** German */
42
+ de = "de",
43
+ /** French */
44
+ fr = "fr",
45
+ /** Russian */
46
+ ru = "ru",
47
+ /** Indonesian */
48
+ id = "id",
49
+ /** Turkish */
50
+ tr = "tr",
51
+ /** Italian */
52
+ it = "it",
53
+ /** Portuguese */
54
+ pt = "pt",
55
+ /** Ukrainian */
56
+ uk = "uk",
57
+ /** Polish */
58
+ pl = "pl",
59
+ /** Dutch */
60
+ nl = "nl"
61
+ }
62
+ type LocaleCode = keyof typeof LocaleEnum | (string & {});
63
+ type Language = {
64
+ localCode: LocaleCode;
65
+ displayName: string;
66
+ };
67
+ type ExtendLocaleMessages = Record<`extend.${string}`, string>;
68
+ type LocaleMessages = typeof en & ExtendLocaleMessages;
69
+ type Resources<T extends {} = {}> = {
70
+ [key in LocaleCode]?: Partial<LocaleMessages & T>;
71
+ };
72
+ declare module "i18next" {
73
+ interface CustomTypeOptions {
74
+ defaultNS: "translation";
75
+ resources: {
76
+ translation: LocaleMessages;
77
+ };
78
+ }
79
+ }
80
+
81
+ export { type ExtendLocaleMessages as E, type LocaleCode as L, type Resources as R, type Language as a, LocaleEnum as b, type LocaleMessages as c, en as e };
@@ -0,0 +1,81 @@
1
+ declare const en: {
2
+ "mediaTrack.title": string;
3
+ "mediaTrack.description": string;
4
+ "common.cancel": string;
5
+ "common.confirm": string;
6
+ "common.ok": string;
7
+ "common.yes": string;
8
+ "common.no": string;
9
+ "common.all": string;
10
+ "common.buy": string;
11
+ "common.sell": string;
12
+ "common.long": string;
13
+ "common.short": string;
14
+ "common.edit": string;
15
+ "common.save": string;
16
+ "common.add": string;
17
+ "common.delete": string;
18
+ "common.tips": string;
19
+ "common.max": string;
20
+ "common.download": string;
21
+ "common.copy": string;
22
+ "common.copy.failed": string;
23
+ "common.copy.copied": string;
24
+ "common.share": string;
25
+ "common.export": string;
26
+ };
27
+
28
+ declare enum LocaleEnum {
29
+ /** English */
30
+ en = "en",
31
+ /** Chinese */
32
+ zh = "zh",
33
+ /** Japanese */
34
+ ja = "ja",
35
+ /** Spanish */
36
+ es = "es",
37
+ /** Korean */
38
+ ko = "ko",
39
+ /** Vietnamese */
40
+ vi = "vi",
41
+ /** German */
42
+ de = "de",
43
+ /** French */
44
+ fr = "fr",
45
+ /** Russian */
46
+ ru = "ru",
47
+ /** Indonesian */
48
+ id = "id",
49
+ /** Turkish */
50
+ tr = "tr",
51
+ /** Italian */
52
+ it = "it",
53
+ /** Portuguese */
54
+ pt = "pt",
55
+ /** Ukrainian */
56
+ uk = "uk",
57
+ /** Polish */
58
+ pl = "pl",
59
+ /** Dutch */
60
+ nl = "nl"
61
+ }
62
+ type LocaleCode = keyof typeof LocaleEnum | (string & {});
63
+ type Language = {
64
+ localCode: LocaleCode;
65
+ displayName: string;
66
+ };
67
+ type ExtendLocaleMessages = Record<`extend.${string}`, string>;
68
+ type LocaleMessages = typeof en & ExtendLocaleMessages;
69
+ type Resources<T extends {} = {}> = {
70
+ [key in LocaleCode]?: Partial<LocaleMessages & T>;
71
+ };
72
+ declare module "i18next" {
73
+ interface CustomTypeOptions {
74
+ defaultNS: "translation";
75
+ resources: {
76
+ translation: LocaleMessages;
77
+ };
78
+ }
79
+ }
80
+
81
+ export { type ExtendLocaleMessages as E, type LocaleCode as L, type Resources as R, type Language as a, LocaleEnum as b, type LocaleMessages as c, en as e };
@@ -0,0 +1,63 @@
1
+ import { L as LocaleCode } from './types-DY-Gbo__.mjs';
2
+
3
+ /**
4
+ * transform browser language to i18n locale codes
5
+ * @param lang - browser language
6
+ * @param localeCodes - locale codes to check
7
+ * @param defaultLang - default locale code
8
+ * @example
9
+ * parseI18nLang('en-US') => 'en'
10
+ * parseI18nLang('zh-CN') => 'zh'
11
+ * parseI18nLang('zh-TW') => 'zh'
12
+ * parseI18nLang('ja') => 'ja'
13
+ * */
14
+ declare function parseI18nLang(lang: string, localeCodes?: LocaleCode[], defaultLang?: LocaleCode): string;
15
+ /**
16
+ * remove lang prefix from pathname
17
+ * @param pathname - pathname to remove lang prefix
18
+ * @param localeCodes - locale codes to check
19
+ * @example
20
+ * removeLangPrefix('/en/perp/PERP_ETH_USDC') => '/perp/PERP_ETH_USDC'
21
+ * removeLangPrefix('/en/markets') => '/markets'
22
+ * removeLangPrefix('/perp/PERP_ETH_USDC') => '/perp/PERP_ETH_USDC'
23
+ * removeLangPrefix('/markets') => '/markets'
24
+ */
25
+ declare function removeLangPrefix(pathname: string, localeCodes?: string[]): string;
26
+ /**
27
+ * get locale path from pathname
28
+ * @param pathname - pathname to get locale path
29
+ * @param localeCodes - locale codes to check
30
+ * @example
31
+ * getLocalePathFromPathname('/en/perp/PERP_ETH_USDC') => 'en'
32
+ * getLocalePathFromPathname('/perp/PERP_ETH_USDC') => null
33
+ * getLocalePathFromPathname('/en/markets') => 'en'
34
+ * getLocalePathFromPathname('/markets') => null
35
+ */
36
+ declare function getLocalePathFromPathname(pathname: string, localeCodes?: string[]): string | null;
37
+ /**
38
+ * Generate a localized path with proper locale prefix and search parameters
39
+ *
40
+ * This function ensures that the returned path includes the appropriate locale prefix.
41
+ * If the path already contains a valid locale prefix, it returns the path as-is.
42
+ * Otherwise, it prepends the specified locale or falls back to the current i18n language.
43
+ *
44
+ * @param params - Configuration object for path generation
45
+ * @param params.path - The base pathname (e.g., '/markets', '/perp/PERP_ETH_USDC')
46
+ * @param params.locale - Optional locale code to use as prefix. If not provided, uses i18n.language
47
+ * @param params.search - Optional search query string. If not provided, uses window.location.search
48
+ *
49
+ * @returns A complete URL path with locale prefix and search parameters
50
+ *
51
+ * @example
52
+ * generatePath({ path: '/markets' }) => '/en/markets?tab=spot'
53
+ * generatePath({ path: '/en/markets', search: '?tab=futures' }) => '/en/markets?tab=futures'
54
+ * generatePath({ path: '/perp/PERP_ETH_USDC', locale: 'zh' }) => '/zh/perp/PERP_ETH_USDC'
55
+ * generatePath({ path: '/en/perp/PERP_ETH_USDC' }) => '/en/perp/PERP_ETH_USDC'
56
+ */
57
+ declare function generatePath(params: {
58
+ path: string;
59
+ locale?: string;
60
+ search?: string;
61
+ }): string;
62
+
63
+ export { generatePath, getLocalePathFromPathname, parseI18nLang, removeLangPrefix };