@let-value/translate 1.0.9-beta.2 → 1.0.9
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.
- package/dist/index.cjs +52 -10
- package/dist/index.d.cts.map +1 -1
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +52 -10
- package/dist/index.js.map +1 -1
- package/package.json +1 -1
package/dist/index.cjs
CHANGED
|
@@ -39,19 +39,58 @@ function memo(fn) {
|
|
|
39
39
|
});
|
|
40
40
|
}
|
|
41
41
|
function substitute(text, values = []) {
|
|
42
|
-
|
|
42
|
+
const placeholders = [];
|
|
43
|
+
return text.replace(/\$\{([^}]+)\}/g, (match, placeholder) => {
|
|
44
|
+
if (/^\d+$/.test(placeholder)) {
|
|
45
|
+
const index$1 = Number(placeholder);
|
|
46
|
+
return index$1 < values.length ? String(values[index$1]) : match;
|
|
47
|
+
}
|
|
48
|
+
let index = placeholders.indexOf(placeholder);
|
|
49
|
+
if (index === -1) {
|
|
50
|
+
index = placeholders.length;
|
|
51
|
+
placeholders.push(placeholder);
|
|
52
|
+
}
|
|
53
|
+
return index < values.length ? String(values[index]) : match;
|
|
54
|
+
});
|
|
55
|
+
}
|
|
56
|
+
function normalizeMessageId(msgid) {
|
|
57
|
+
const placeholders = [];
|
|
58
|
+
return msgid.replace(/\$\{([^}]+)\}/g, (match, placeholder) => {
|
|
59
|
+
if (/^\d+$/.test(placeholder)) return match;
|
|
60
|
+
let index = placeholders.indexOf(placeholder);
|
|
61
|
+
if (index === -1) {
|
|
62
|
+
index = placeholders.length;
|
|
63
|
+
placeholders.push(placeholder);
|
|
64
|
+
}
|
|
65
|
+
return `\${${index}}`;
|
|
66
|
+
});
|
|
67
|
+
}
|
|
68
|
+
function normalizeTranslations(translations) {
|
|
69
|
+
const normalized = {
|
|
70
|
+
...translations,
|
|
71
|
+
translations: {}
|
|
72
|
+
};
|
|
73
|
+
for (const [context$1, messages] of Object.entries(translations.translations)) {
|
|
74
|
+
normalized.translations[context$1] = {};
|
|
75
|
+
for (const [msgid, entry] of Object.entries(messages)) {
|
|
76
|
+
const normalizedId = normalizeMessageId(msgid);
|
|
77
|
+
normalized.translations[context$1][normalizedId] = entry;
|
|
78
|
+
}
|
|
79
|
+
}
|
|
80
|
+
return normalized;
|
|
43
81
|
}
|
|
44
82
|
const defaultPluralFunc = (n) => n !== 1 ? 1 : 0;
|
|
45
83
|
const pluralFunc = memo(function pluralFunc$1(locale) {
|
|
46
84
|
try {
|
|
47
85
|
const length = Number((0, plural_forms.getNPlurals)(locale));
|
|
86
|
+
if (length === 1) return () => 0;
|
|
48
87
|
const pluralFunc$2 = (0, plural_forms.getPluralFunc)(locale);
|
|
49
88
|
const forms = Array.from({ length }, (_, i) => String(i));
|
|
50
89
|
return (n) => {
|
|
51
90
|
const idx = Number(pluralFunc$2(n, forms));
|
|
52
|
-
if (idx < 0) return 0;
|
|
91
|
+
if (!Number.isFinite(idx) || idx < 0) return 0;
|
|
53
92
|
if (idx >= length) return length - 1;
|
|
54
|
-
return idx;
|
|
93
|
+
return Math.floor(idx);
|
|
55
94
|
};
|
|
56
95
|
} catch {
|
|
57
96
|
return defaultPluralFunc;
|
|
@@ -157,19 +196,19 @@ var LocaleTranslator = class {
|
|
|
157
196
|
translations;
|
|
158
197
|
constructor(locale, translations) {
|
|
159
198
|
this.locale = locale;
|
|
160
|
-
this.translations = translations;
|
|
199
|
+
this.translations = translations ? normalizeTranslations(translations) : void 0;
|
|
161
200
|
}
|
|
162
201
|
translateMessage({ msgid, msgstr, values }, msgctxt = "") {
|
|
163
202
|
const translated = this.translations?.translations?.[msgctxt]?.[msgid];
|
|
164
|
-
const result = translated?.msgstr
|
|
203
|
+
const result = translated?.msgstr?.[0] || msgstr;
|
|
165
204
|
return values ? substitute(result, values) : result;
|
|
166
205
|
}
|
|
167
206
|
translatePlural({ forms, n }, msgctxt = "") {
|
|
168
207
|
const entry = this.translations?.translations?.[msgctxt]?.[forms[0].msgid];
|
|
169
208
|
const index = pluralFunc(this.locale)(n);
|
|
170
209
|
const form = forms[index] ?? forms[forms.length - 1];
|
|
171
|
-
const translated = entry?.msgstr
|
|
172
|
-
const result = translated
|
|
210
|
+
const translated = entry?.msgstr ? entry.msgstr[index] ?? entry.msgstr[entry.msgstr.length - 1] : void 0;
|
|
211
|
+
const result = translated || form.msgstr;
|
|
173
212
|
const usedVals = form.values?.length ? form.values : forms[0].values;
|
|
174
213
|
return usedVals?.length ? substitute(result, usedVals) : result;
|
|
175
214
|
}
|
|
@@ -228,9 +267,12 @@ var Translator = class {
|
|
|
228
267
|
translators = {};
|
|
229
268
|
constructor(translations, parent) {
|
|
230
269
|
this.parent = parent;
|
|
231
|
-
for (const [locale, value] of Object.entries(translations))
|
|
232
|
-
|
|
233
|
-
|
|
270
|
+
for (const [locale, value] of Object.entries(translations)) {
|
|
271
|
+
if (!value) continue;
|
|
272
|
+
if (typeof value === "function") this.loaders[locale] = value;
|
|
273
|
+
else if ("then" in value) this.loaders[locale] = () => value;
|
|
274
|
+
else this.translations[locale] = resolveTranslationModule(value);
|
|
275
|
+
}
|
|
234
276
|
}
|
|
235
277
|
async loadLocale(locale, loader) {
|
|
236
278
|
this.loaders[locale] = loader ?? this.loaders[locale];
|
package/dist/index.d.cts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.d.cts","names":[],"sources":["../src/config.ts","../src/utils.ts","../src/messages.ts","../src/translator.ts"],"sourcesContent":[],"mappings":";;;;UAEiB,aAAA;;;AAAjB;EAA8B,aAAA,EAIX,iBAJW;;;;;;SAcf,EAAA,SAJO,iBAIP,EAAA;;AAGf;;aAHe,QAAQ,OAAO,mBAAmB;;
|
|
1
|
+
{"version":3,"file":"index.d.cts","names":[],"sources":["../src/config.ts","../src/utils.ts","../src/messages.ts","../src/translator.ts"],"sourcesContent":[],"mappings":"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"}
|
package/dist/index.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.d.ts","names":[],"sources":["../src/config.ts","../src/utils.ts","../src/messages.ts","../src/translator.ts"],"sourcesContent":[],"mappings":";;;;UAEiB,aAAA;;;AAAjB;EAA8B,aAAA,EAIX,iBAJW;;;;;;SAcf,EAAA,SAJO,iBAIP,EAAA;;AAGf;;aAHe,QAAQ,OAAO,mBAAmB;;
|
|
1
|
+
{"version":3,"file":"index.d.ts","names":[],"sources":["../src/config.ts","../src/utils.ts","../src/messages.ts","../src/translator.ts"],"sourcesContent":[],"mappings":"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"}
|
package/dist/index.js
CHANGED
|
@@ -15,19 +15,58 @@ function memo(fn) {
|
|
|
15
15
|
});
|
|
16
16
|
}
|
|
17
17
|
function substitute(text, values = []) {
|
|
18
|
-
|
|
18
|
+
const placeholders = [];
|
|
19
|
+
return text.replace(/\$\{([^}]+)\}/g, (match, placeholder) => {
|
|
20
|
+
if (/^\d+$/.test(placeholder)) {
|
|
21
|
+
const index$1 = Number(placeholder);
|
|
22
|
+
return index$1 < values.length ? String(values[index$1]) : match;
|
|
23
|
+
}
|
|
24
|
+
let index = placeholders.indexOf(placeholder);
|
|
25
|
+
if (index === -1) {
|
|
26
|
+
index = placeholders.length;
|
|
27
|
+
placeholders.push(placeholder);
|
|
28
|
+
}
|
|
29
|
+
return index < values.length ? String(values[index]) : match;
|
|
30
|
+
});
|
|
31
|
+
}
|
|
32
|
+
function normalizeMessageId(msgid) {
|
|
33
|
+
const placeholders = [];
|
|
34
|
+
return msgid.replace(/\$\{([^}]+)\}/g, (match, placeholder) => {
|
|
35
|
+
if (/^\d+$/.test(placeholder)) return match;
|
|
36
|
+
let index = placeholders.indexOf(placeholder);
|
|
37
|
+
if (index === -1) {
|
|
38
|
+
index = placeholders.length;
|
|
39
|
+
placeholders.push(placeholder);
|
|
40
|
+
}
|
|
41
|
+
return `\${${index}}`;
|
|
42
|
+
});
|
|
43
|
+
}
|
|
44
|
+
function normalizeTranslations(translations) {
|
|
45
|
+
const normalized = {
|
|
46
|
+
...translations,
|
|
47
|
+
translations: {}
|
|
48
|
+
};
|
|
49
|
+
for (const [context$1, messages] of Object.entries(translations.translations)) {
|
|
50
|
+
normalized.translations[context$1] = {};
|
|
51
|
+
for (const [msgid, entry] of Object.entries(messages)) {
|
|
52
|
+
const normalizedId = normalizeMessageId(msgid);
|
|
53
|
+
normalized.translations[context$1][normalizedId] = entry;
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
return normalized;
|
|
19
57
|
}
|
|
20
58
|
const defaultPluralFunc = (n) => n !== 1 ? 1 : 0;
|
|
21
59
|
const pluralFunc = memo(function pluralFunc$1(locale) {
|
|
22
60
|
try {
|
|
23
61
|
const length = Number(getNPlurals(locale));
|
|
62
|
+
if (length === 1) return () => 0;
|
|
24
63
|
const pluralFunc$2 = getPluralFunc(locale);
|
|
25
64
|
const forms = Array.from({ length }, (_, i) => String(i));
|
|
26
65
|
return (n) => {
|
|
27
66
|
const idx = Number(pluralFunc$2(n, forms));
|
|
28
|
-
if (idx < 0) return 0;
|
|
67
|
+
if (!Number.isFinite(idx) || idx < 0) return 0;
|
|
29
68
|
if (idx >= length) return length - 1;
|
|
30
|
-
return idx;
|
|
69
|
+
return Math.floor(idx);
|
|
31
70
|
};
|
|
32
71
|
} catch {
|
|
33
72
|
return defaultPluralFunc;
|
|
@@ -133,19 +172,19 @@ var LocaleTranslator = class {
|
|
|
133
172
|
translations;
|
|
134
173
|
constructor(locale, translations) {
|
|
135
174
|
this.locale = locale;
|
|
136
|
-
this.translations = translations;
|
|
175
|
+
this.translations = translations ? normalizeTranslations(translations) : void 0;
|
|
137
176
|
}
|
|
138
177
|
translateMessage({ msgid, msgstr, values }, msgctxt = "") {
|
|
139
178
|
const translated = this.translations?.translations?.[msgctxt]?.[msgid];
|
|
140
|
-
const result = translated?.msgstr
|
|
179
|
+
const result = translated?.msgstr?.[0] || msgstr;
|
|
141
180
|
return values ? substitute(result, values) : result;
|
|
142
181
|
}
|
|
143
182
|
translatePlural({ forms, n }, msgctxt = "") {
|
|
144
183
|
const entry = this.translations?.translations?.[msgctxt]?.[forms[0].msgid];
|
|
145
184
|
const index = pluralFunc(this.locale)(n);
|
|
146
185
|
const form = forms[index] ?? forms[forms.length - 1];
|
|
147
|
-
const translated = entry?.msgstr
|
|
148
|
-
const result = translated
|
|
186
|
+
const translated = entry?.msgstr ? entry.msgstr[index] ?? entry.msgstr[entry.msgstr.length - 1] : void 0;
|
|
187
|
+
const result = translated || form.msgstr;
|
|
149
188
|
const usedVals = form.values?.length ? form.values : forms[0].values;
|
|
150
189
|
return usedVals?.length ? substitute(result, usedVals) : result;
|
|
151
190
|
}
|
|
@@ -204,9 +243,12 @@ var Translator = class {
|
|
|
204
243
|
translators = {};
|
|
205
244
|
constructor(translations, parent) {
|
|
206
245
|
this.parent = parent;
|
|
207
|
-
for (const [locale, value] of Object.entries(translations))
|
|
208
|
-
|
|
209
|
-
|
|
246
|
+
for (const [locale, value] of Object.entries(translations)) {
|
|
247
|
+
if (!value) continue;
|
|
248
|
+
if (typeof value === "function") this.loaders[locale] = value;
|
|
249
|
+
else if ("then" in value) this.loaders[locale] = () => value;
|
|
250
|
+
else this.translations[locale] = resolveTranslationModule(value);
|
|
251
|
+
}
|
|
210
252
|
}
|
|
211
253
|
async loadLocale(locale, loader) {
|
|
212
254
|
this.loaders[locale] = loader ?? this.loaders[locale];
|
package/dist/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.js","names":["message","pluralFunc","id","message","args","plural","buildMessage","buildPlural","args","context"],"sources":["../src/utils.ts","../src/messages.ts","../../node_modules/radash/dist/esm/typed.mjs","../../node_modules/radash/dist/esm/object.mjs","../src/translator.ts"],"sourcesContent":["import { getNPlurals, getPluralFunc } from \"plural-forms\";\nimport type { Locale } from \"./config.ts\";\n\n// biome-ignore lint/suspicious/noExplicitAny: true\nexport type IsUnion<T, U = T> = (T extends any ? (x: T) => 0 : never) extends (x: U) => 0 ? false : true;\n\nexport type StrictStaticString<T extends string> = string extends T ? never : IsUnion<T> extends true ? never : T;\n\nexport function assert<T>(value: T, message?: string): asserts value is T {\n if (!value) {\n throw new Error(message || \"Assertion failed\");\n }\n}\n\n// biome-ignore lint/suspicious/noExplicitAny: true\nexport function memo<T extends (...args: any[]) => any>(fn: T): T {\n const cache = new Map<string, ReturnType<T>>();\n return ((...args: Parameters<T>): ReturnType<T> => {\n const key = JSON.stringify(args);\n if (cache.has(key)) {\n // biome-ignore lint/style/noNonNullAssertion: true\n return cache.get(key)!;\n }\n const result = fn(...args);\n cache.set(key, result);\n return result;\n }) as T;\n}\n\nexport function substitute(text: string, values: unknown[] = []): string {\n return text.replace(/\\$\\{(\\d+)\\}/g, (_, i) => String(values[Number(i)]));\n}\n\nconst defaultPluralFunc = (n: number) => (n !== 1 ? 1 : 0);\n\nexport const pluralFunc = memo(function pluralFunc(locale: Locale) {\n try {\n const length = Number(getNPlurals(locale));\n const pluralFunc = getPluralFunc(locale);\n const forms = Array.from({ length }, (_, i) => String(i));\n return (n: number) => {\n const idx = Number(pluralFunc(n, forms));\n if (idx < 0) return 0;\n if (idx >= length) return length - 1;\n return idx;\n };\n } catch {\n return defaultPluralFunc;\n }\n});\n","import { assert, type StrictStaticString } from \"./utils.ts\";\n\nexport interface MessageDescriptor {\n id?: string;\n message?: string;\n}\n\nexport interface Message {\n msgid: string;\n msgstr: string;\n // biome-ignore lint/suspicious/noExplicitAny: true\n values?: any[];\n}\n\nexport interface PluralMessage {\n forms: Message[];\n n: number;\n}\n\nexport interface ContextMessage {\n context: string;\n id: Message;\n}\n\nexport interface ContextPluralMessage {\n context: string;\n id: PluralMessage;\n}\n\nfunction buildFromTemplate(strings: TemplateStringsArray): string {\n let result = \"\";\n for (let i = 0; i < strings.length; i++) {\n result += strings[i];\n if (i < strings.length - 1) result += `\\${${i}}`;\n }\n return result;\n}\n\nexport type MessageArgs<T extends string> =\n | [id: StrictStaticString<T>]\n | [descriptor: MessageDescriptor]\n | [strings: TemplateStringsArray, ...values: unknown[]];\n\nexport type MessageFunction = <T extends string>(...args: MessageArgs<T>) => Message;\n\nexport function message<T extends string>(...args: MessageArgs<T>): Message {\n const [source, ...values] = args as [MessageDescriptor | TemplateStringsArray, ...unknown[]];\n\n if (typeof source === \"string\") {\n return { msgid: source, msgstr: source };\n }\n\n if (typeof source === \"object\" && \"reduce\" in source) {\n const id = buildFromTemplate(source);\n return { msgid: id, msgstr: id, values };\n }\n\n const id = source.id ?? source.message ?? \"\";\n const message = source.message ?? source.id ?? \"\";\n return { msgid: id, msgstr: message };\n}\n\nexport type PluralArgs = [...forms: Message[], n: number];\n\nexport function plural(...args: PluralArgs): PluralMessage {\n assert(args.length > 1, \"At least one plural form and n are required\");\n\n const n = args[args.length - 1] as number;\n assert(typeof n === \"number\", \"The last argument must be a number\");\n\n const forms = args.slice(0, -1) as Message[];\n\n return { forms, n };\n}\n\nexport type PluralFunction = (...args: PluralArgs) => PluralMessage;\nexport type MessageInput<T extends string = string> = [Message] | MessageArgs<T>;\nexport type PluralInput = [PluralMessage] | PluralArgs;\n\nexport type ContextMessageFunction = <T extends string>(...args: MessageArgs<T>) => ContextMessage;\n\nexport type ContextPluralFunction = (...args: PluralArgs) => ContextPluralMessage;\n\nexport interface ContextBuilder {\n message: ContextMessageFunction;\n plural: ContextPluralFunction;\n}\n\nconst baseMessage = message;\nconst basePlural = plural;\n\nexport function context<T extends string>(context: StrictStaticString<T>): ContextBuilder;\nexport function context(strings: TemplateStringsArray, ...values: never[]): ContextBuilder;\nexport function context<T extends string>(\n ...args: [StrictStaticString<T>] | [TemplateStringsArray, ...never[]]\n): ContextBuilder {\n const [source] = args as [StrictStaticString<T> | TemplateStringsArray];\n\n const ctx = typeof source === \"string\" ? source : source[0];\n\n function message<T extends string>(...args: MessageArgs<T>): ContextMessage {\n return { id: baseMessage(...args), context: ctx };\n }\n\n function plural(...forms: PluralArgs): ContextPluralMessage {\n return { id: basePlural(...forms), context: ctx };\n }\n\n return {\n message,\n plural,\n };\n}\n","const isSymbol = (value) => {\n return !!value && value.constructor === Symbol;\n};\nconst isArray = Array.isArray;\nconst isObject = (value) => {\n return !!value && value.constructor === Object;\n};\nconst isPrimitive = (value) => {\n return value === void 0 || value === null || typeof value !== \"object\" && typeof value !== \"function\";\n};\nconst isFunction = (value) => {\n return !!(value && value.constructor && value.call && value.apply);\n};\nconst isString = (value) => {\n return typeof value === \"string\" || value instanceof String;\n};\nconst isInt = (value) => {\n return isNumber(value) && value % 1 === 0;\n};\nconst isFloat = (value) => {\n return isNumber(value) && value % 1 !== 0;\n};\nconst isNumber = (value) => {\n try {\n return Number(value) === value;\n } catch {\n return false;\n }\n};\nconst isDate = (value) => {\n return Object.prototype.toString.call(value) === \"[object Date]\";\n};\nconst isPromise = (value) => {\n if (!value)\n return false;\n if (!value.then)\n return false;\n if (!isFunction(value.then))\n return false;\n return true;\n};\nconst isEmpty = (value) => {\n if (value === true || value === false)\n return true;\n if (value === null || value === void 0)\n return true;\n if (isNumber(value))\n return value === 0;\n if (isDate(value))\n return isNaN(value.getTime());\n if (isFunction(value))\n return false;\n if (isSymbol(value))\n return false;\n const length = value.length;\n if (isNumber(length))\n return length === 0;\n const size = value.size;\n if (isNumber(size))\n return size === 0;\n const keys = Object.keys(value).length;\n return keys === 0;\n};\nconst isEqual = (x, y) => {\n if (Object.is(x, y))\n return true;\n if (x instanceof Date && y instanceof Date) {\n return x.getTime() === y.getTime();\n }\n if (x instanceof RegExp && y instanceof RegExp) {\n return x.toString() === y.toString();\n }\n if (typeof x !== \"object\" || x === null || typeof y !== \"object\" || y === null) {\n return false;\n }\n const keysX = Reflect.ownKeys(x);\n const keysY = Reflect.ownKeys(y);\n if (keysX.length !== keysY.length)\n return false;\n for (let i = 0; i < keysX.length; i++) {\n if (!Reflect.has(y, keysX[i]))\n return false;\n if (!isEqual(x[keysX[i]], y[keysX[i]]))\n return false;\n }\n return true;\n};\n\nexport { isArray, isDate, isEmpty, isEqual, isFloat, isFunction, isInt, isNumber, isObject, isPrimitive, isPromise, isString, isSymbol };\n//# sourceMappingURL=typed.mjs.map\n","import { objectify } from './array.mjs';\nimport { isPrimitive, isObject, isArray } from './typed.mjs';\n\nconst shake = (obj, filter = (x) => x === void 0) => {\n if (!obj)\n return {};\n const keys2 = Object.keys(obj);\n return keys2.reduce((acc, key) => {\n if (filter(obj[key])) {\n return acc;\n } else {\n acc[key] = obj[key];\n return acc;\n }\n }, {});\n};\nconst mapKeys = (obj, mapFunc) => {\n const keys2 = Object.keys(obj);\n return keys2.reduce((acc, key) => {\n acc[mapFunc(key, obj[key])] = obj[key];\n return acc;\n }, {});\n};\nconst mapValues = (obj, mapFunc) => {\n const keys2 = Object.keys(obj);\n return keys2.reduce((acc, key) => {\n acc[key] = mapFunc(obj[key], key);\n return acc;\n }, {});\n};\nconst mapEntries = (obj, toEntry) => {\n if (!obj)\n return {};\n return Object.entries(obj).reduce((acc, [key, value]) => {\n const [newKey, newValue] = toEntry(key, value);\n acc[newKey] = newValue;\n return acc;\n }, {});\n};\nconst invert = (obj) => {\n if (!obj)\n return {};\n const keys2 = Object.keys(obj);\n return keys2.reduce((acc, key) => {\n acc[obj[key]] = key;\n return acc;\n }, {});\n};\nconst lowerize = (obj) => mapKeys(obj, (k) => k.toLowerCase());\nconst upperize = (obj) => mapKeys(obj, (k) => k.toUpperCase());\nconst clone = (obj) => {\n if (isPrimitive(obj)) {\n return obj;\n }\n if (typeof obj === \"function\") {\n return obj.bind({});\n }\n const newObj = new obj.constructor();\n Object.getOwnPropertyNames(obj).forEach((prop) => {\n newObj[prop] = obj[prop];\n });\n return newObj;\n};\nconst listify = (obj, toItem) => {\n if (!obj)\n return [];\n const entries = Object.entries(obj);\n if (entries.length === 0)\n return [];\n return entries.reduce((acc, entry) => {\n acc.push(toItem(entry[0], entry[1]));\n return acc;\n }, []);\n};\nconst pick = (obj, keys2) => {\n if (!obj)\n return {};\n return keys2.reduce((acc, key) => {\n if (Object.prototype.hasOwnProperty.call(obj, key))\n acc[key] = obj[key];\n return acc;\n }, {});\n};\nconst omit = (obj, keys2) => {\n if (!obj)\n return {};\n if (!keys2 || keys2.length === 0)\n return obj;\n return keys2.reduce(\n (acc, key) => {\n delete acc[key];\n return acc;\n },\n { ...obj }\n );\n};\nconst get = (value, path, defaultValue) => {\n const segments = path.split(/[\\.\\[\\]]/g);\n let current = value;\n for (const key of segments) {\n if (current === null)\n return defaultValue;\n if (current === void 0)\n return defaultValue;\n const dequoted = key.replace(/['\"]/g, \"\");\n if (dequoted.trim() === \"\")\n continue;\n current = current[dequoted];\n }\n if (current === void 0)\n return defaultValue;\n return current;\n};\nconst set = (initial, path, value) => {\n if (!initial)\n return {};\n if (!path || value === void 0)\n return initial;\n const segments = path.split(/[\\.\\[\\]]/g).filter((x) => !!x.trim());\n const _set = (node) => {\n if (segments.length > 1) {\n const key = segments.shift();\n const nextIsNum = /^\\d+$/.test(segments[0]);\n node[key] = node[key] === void 0 ? nextIsNum ? [] : {} : node[key];\n _set(node[key]);\n } else {\n node[segments[0]] = value;\n }\n };\n const cloned = clone(initial);\n _set(cloned);\n return cloned;\n};\nconst assign = (initial, override) => {\n if (!initial || !override)\n return initial ?? override ?? {};\n return Object.entries({ ...initial, ...override }).reduce(\n (acc, [key, value]) => {\n return {\n ...acc,\n [key]: (() => {\n if (isObject(initial[key]))\n return assign(initial[key], value);\n return value;\n })()\n };\n },\n {}\n );\n};\nconst keys = (value) => {\n if (!value)\n return [];\n const getKeys = (nested, paths) => {\n if (isObject(nested)) {\n return Object.entries(nested).flatMap(\n ([k, v]) => getKeys(v, [...paths, k])\n );\n }\n if (isArray(nested)) {\n return nested.flatMap((item, i) => getKeys(item, [...paths, `${i}`]));\n }\n return [paths.join(\".\")];\n };\n return getKeys(value, []);\n};\nconst crush = (value) => {\n if (!value)\n return {};\n return objectify(\n keys(value),\n (k) => k,\n (k) => get(value, k)\n );\n};\nconst construct = (obj) => {\n if (!obj)\n return {};\n return Object.keys(obj).reduce((acc, path) => {\n return set(acc, path, obj[path]);\n }, {});\n};\n\nexport { assign, clone, construct, crush, get, invert, keys, listify, lowerize, mapEntries, mapKeys, mapValues, omit, pick, set, shake, upperize };\n//# sourceMappingURL=object.mjs.map\n","import type { GetTextTranslations } from \"gettext-parser\";\nimport { assign } from \"radash\";\nimport type { Locale } from \"./config.ts\";\n\nimport {\n message as buildMessage,\n plural as buildPlural,\n type ContextMessage,\n type ContextPluralMessage,\n type Message,\n type MessageArgs,\n type MessageInput,\n type PluralArgs,\n type PluralInput,\n type PluralMessage,\n} from \"./messages.ts\";\nimport { pluralFunc, type StrictStaticString, substitute } from \"./utils.ts\";\n\ntype TranslationModule = GetTextTranslations | { default: GetTextTranslations };\ntype TranslationLoader = () => Promise<TranslationModule>;\ntype TranslationEntry = TranslationModule | Promise<TranslationModule> | TranslationLoader;\ntype TranslationRecord = Partial<Record<Locale, TranslationEntry>>;\ntype SyncLocaleKeys<T> = {\n [K in keyof T]: T[K] extends GetTextTranslations ? K : never;\n}[keyof T];\n\ntype ContextMessageInput<T extends string> = [ContextMessage] | MessageInput<T>;\ntype ContextPluralInput = [ContextPluralMessage] | PluralInput;\n\nfunction resolveTranslationModule(module: TranslationModule): GetTextTranslations {\n return \"default\" in module ? module.default : module;\n}\n\nexport class LocaleTranslator {\n locale: Locale;\n translations?: GetTextTranslations;\n\n constructor(locale: Locale, translations?: GetTextTranslations) {\n this.locale = locale;\n this.translations = translations;\n }\n\n private translateMessage({ msgid, msgstr, values }: Message, msgctxt = \"\"): string {\n const translated = this.translations?.translations?.[msgctxt]?.[msgid];\n const result = translated?.msgstr ? translated.msgstr[0] : msgstr;\n return values ? substitute(result, values) : result;\n }\n\n private translatePlural({ forms, n }: PluralMessage, msgctxt = \"\"): string {\n const entry = this.translations?.translations?.[msgctxt]?.[forms[0].msgid];\n const index = pluralFunc(this.locale)(n);\n const form = forms[index] ?? forms[forms.length - 1];\n const translated = entry?.msgstr?.[index];\n const result = translated?.length ? translated : form.msgstr;\n const usedVals = form.values?.length ? form.values : forms[0].values;\n return usedVals?.length ? substitute(result, usedVals) : result;\n }\n\n message = <T extends string>(...args: MessageInput<T>): string => {\n const [source] = args;\n if (typeof source === \"object\" && \"msgid\" in source) {\n return this.translateMessage(source);\n }\n return this.translateMessage(buildMessage(...(args as MessageArgs<T>)));\n };\n\n plural = (...args: PluralInput): string => {\n const [source] = args;\n if (typeof source === \"object\" && \"forms\" in source) {\n return this.translatePlural(source);\n }\n return this.translatePlural(buildPlural(...(args as PluralArgs)));\n };\n\n context<T extends string>(\n context: StrictStaticString<T>,\n ): {\n message: <T extends string>(...args: ContextMessageInput<T>) => string;\n plural: (...args: ContextPluralInput) => string;\n };\n context(\n strings: TemplateStringsArray,\n ...values: never[]\n ): {\n message: <T extends string>(...args: ContextMessageInput<T>) => string;\n plural: (...args: ContextPluralInput) => string;\n };\n context<T extends string>(...args: [StrictStaticString<T>] | [TemplateStringsArray, ...never[]]) {\n const [source] = args as [StrictStaticString<T> | TemplateStringsArray];\n\n const ctx = typeof source === \"string\" ? source : source[0];\n\n return {\n message: <T extends string>(...args: ContextMessageInput<T>): string => {\n const [src] = args;\n if (typeof src === \"object\") {\n if (\"id\" in src && \"context\" in src) {\n return this.translateMessage(src.id, src.context);\n }\n if (\"msgid\" in src) {\n return this.translateMessage(src, ctx);\n }\n }\n return this.translateMessage(buildMessage(...(args as MessageArgs<T>)), ctx);\n },\n plural: (...args: ContextPluralInput): string => {\n const [src] = args;\n if (typeof src === \"object\") {\n if (\"id\" in src && \"context\" in src) {\n return this.translatePlural(src.id, src.context);\n }\n if (\"forms\" in src) {\n return this.translatePlural(src, ctx);\n }\n }\n return this.translatePlural(buildPlural(...(args as PluralArgs)), ctx);\n },\n };\n }\n\n gettext<T extends string>(...args: MessageInput<T>): string {\n return this.message(...args);\n }\n\n ngettext(...args: PluralInput): string {\n return this.plural(...args);\n }\n\n pgettext<C extends string, T extends string>(\n context: ContextMessage | StrictStaticString<C>,\n ...args: MessageInput<T> | []\n ): string {\n if (typeof context === \"object\") {\n return this.translateMessage(context.id, context.context);\n }\n return this.context(context).message(...(args as ContextMessageInput<T>));\n }\n\n npgettext<C extends string>(\n context: ContextPluralMessage | StrictStaticString<C>,\n ...args: PluralInput | []\n ): string {\n if (typeof context === \"object\") {\n return this.translatePlural(context.id, context.context);\n }\n return this.context(context).plural(...(args as ContextPluralInput));\n }\n}\n\nexport class Translator<T extends TranslationRecord = TranslationRecord> {\n parent: Translator | undefined;\n loaders: Partial<Record<Locale, TranslationLoader>> = {};\n translations: Partial<Record<Locale, GetTextTranslations>> = {};\n pending: Partial<Record<Locale, Promise<LocaleTranslator>>> = {};\n translators: Partial<Record<Locale, LocaleTranslator>> = {};\n\n constructor(translations: T, parent?: Translator) {\n this.parent = parent;\n for (const [locale, value] of Object.entries(translations) as [Locale, TranslationEntry][]) {\n if (typeof value === \"function\") {\n this.loaders[locale] = value;\n } else if (\"then\" in value) {\n this.loaders[locale] = () => value;\n } else {\n this.translations[locale] = resolveTranslationModule(value);\n }\n }\n }\n\n async loadLocale(locale: Locale, loader?: TranslationLoader): Promise<LocaleTranslator> {\n this.loaders[locale] = loader ?? this.loaders[locale];\n\n return this.fetchLocale(locale as never);\n }\n\n getLocale<L extends SyncLocaleKeys<T>>(locale: L): LocaleTranslator {\n const key = locale as Locale;\n const existing = this.translators[key];\n if (existing) {\n return existing;\n }\n\n const base = this.translations[key];\n if (base) {\n const parentTranslations = this.parent?.getLocale(locale as never)?.translations;\n const translations = assign<GetTextTranslations>(base, parentTranslations as never);\n delete this.translations[key];\n\n const translator = new LocaleTranslator(key, translations);\n this.translators[key] = translator;\n return translator;\n }\n\n if (this.pending[key] || this.loaders[key]) {\n throw new Error(\"async locale cannot be loaded synchronously\");\n }\n\n this.translators[key] ??= new LocaleTranslator(key);\n return this.translators[key];\n }\n\n fetchLocale<L extends keyof T>(locale: L) {\n const key = locale as Locale;\n\n const pending = this.pending[key];\n if (pending) {\n return pending;\n }\n\n const loader = this.loaders[key];\n if (loader) {\n const promise = loader().then((translations) => {\n this.translations[key] ??= resolveTranslationModule(translations);\n delete this.pending[key];\n\n return this.getLocale(key as never);\n });\n this.pending[key] = promise;\n delete this.loaders[key];\n\n return promise;\n }\n\n return this.getLocale(key as never);\n }\n}\n"],"x_google_ignoreList":[2,3],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"index.js","names":["message","placeholders: string[]","index","normalized: GetTextTranslations","context","pluralFunc","id","message","args","plural","buildMessage","buildPlural","args","context"],"sources":["../src/utils.ts","../src/messages.ts","../../node_modules/radash/dist/esm/typed.mjs","../../node_modules/radash/dist/esm/object.mjs","../src/translator.ts"],"sourcesContent":["import type { GetTextTranslations } from \"gettext-parser\";\nimport { getNPlurals, getPluralFunc } from \"plural-forms\";\nimport type { Locale } from \"./config.ts\";\n\n// biome-ignore lint/suspicious/noExplicitAny: true\nexport type IsUnion<T, U = T> = (T extends any ? (x: T) => 0 : never) extends (x: U) => 0 ? false : true;\n\nexport type StrictStaticString<T extends string> = string extends T ? never : IsUnion<T> extends true ? never : T;\n\nexport function assert<T>(value: T, message?: string): asserts value is T {\n if (!value) {\n throw new Error(message || \"Assertion failed\");\n }\n}\n\n// biome-ignore lint/suspicious/noExplicitAny: true\nexport function memo<T extends (...args: any[]) => any>(fn: T): T {\n const cache = new Map<string, ReturnType<T>>();\n return ((...args: Parameters<T>): ReturnType<T> => {\n const key = JSON.stringify(args);\n if (cache.has(key)) {\n // biome-ignore lint/style/noNonNullAssertion: true\n return cache.get(key)!;\n }\n const result = fn(...args);\n cache.set(key, result);\n return result;\n }) as T;\n}\n\nexport function substitute(text: string, values: unknown[] = []): string {\n const placeholders: string[] = [];\n\n return text.replace(/\\$\\{([^}]+)\\}/g, (match, placeholder) => {\n if (/^\\d+$/.test(placeholder)) {\n const index = Number(placeholder);\n return index < values.length ? String(values[index]) : match;\n }\n\n let index = placeholders.indexOf(placeholder);\n if (index === -1) {\n index = placeholders.length;\n placeholders.push(placeholder);\n }\n\n return index < values.length ? String(values[index]) : match;\n });\n}\n\nexport function normalizeMessageId(msgid: string): string {\n const placeholders: string[] = [];\n return msgid.replace(/\\$\\{([^}]+)\\}/g, (match, placeholder) => {\n if (/^\\d+$/.test(placeholder)) {\n return match;\n }\n\n let index = placeholders.indexOf(placeholder);\n if (index === -1) {\n index = placeholders.length;\n placeholders.push(placeholder);\n }\n return `\\${${index}}`;\n });\n}\n\nexport function normalizeTranslations(translations: GetTextTranslations): GetTextTranslations {\n const normalized: GetTextTranslations = {\n ...translations,\n translations: {},\n };\n\n for (const [context, messages] of Object.entries(translations.translations)) {\n normalized.translations[context] = {};\n for (const [msgid, entry] of Object.entries(messages)) {\n const normalizedId = normalizeMessageId(msgid);\n normalized.translations[context][normalizedId] = entry;\n }\n }\n\n return normalized;\n}\n\nconst defaultPluralFunc = (n: number) => (n !== 1 ? 1 : 0);\n\nexport const pluralFunc = memo(function pluralFunc(locale: Locale) {\n try {\n const length = Number(getNPlurals(locale));\n if (length === 1) {\n return () => 0;\n }\n const pluralFunc = getPluralFunc(locale);\n const forms = Array.from({ length }, (_, i) => String(i));\n return (n: number) => {\n const idx = Number(pluralFunc(n, forms));\n if (!Number.isFinite(idx) || idx < 0) return 0;\n if (idx >= length) return length - 1;\n return Math.floor(idx);\n };\n } catch {\n return defaultPluralFunc;\n }\n});\n","import { assert, type StrictStaticString } from \"./utils.ts\";\n\nexport interface MessageDescriptor {\n id?: string;\n message?: string;\n}\n\nexport interface Message {\n msgid: string;\n msgstr: string;\n // biome-ignore lint/suspicious/noExplicitAny: true\n values?: any[];\n}\n\nexport interface PluralMessage {\n forms: Message[];\n n: number;\n}\n\nexport interface ContextMessage {\n context: string;\n id: Message;\n}\n\nexport interface ContextPluralMessage {\n context: string;\n id: PluralMessage;\n}\n\nfunction buildFromTemplate(strings: TemplateStringsArray): string {\n let result = \"\";\n for (let i = 0; i < strings.length; i++) {\n result += strings[i];\n if (i < strings.length - 1) result += `\\${${i}}`;\n }\n return result;\n}\n\nexport type MessageArgs<T extends string> =\n | [id: StrictStaticString<T>]\n | [descriptor: MessageDescriptor]\n | [strings: TemplateStringsArray, ...values: unknown[]];\n\nexport type MessageFunction = <T extends string>(...args: MessageArgs<T>) => Message;\n\nexport function message<T extends string>(...args: MessageArgs<T>): Message {\n const [source, ...values] = args as [MessageDescriptor | TemplateStringsArray, ...unknown[]];\n\n if (typeof source === \"string\") {\n return { msgid: source, msgstr: source };\n }\n\n if (typeof source === \"object\" && \"reduce\" in source) {\n const id = buildFromTemplate(source);\n return { msgid: id, msgstr: id, values };\n }\n\n const id = source.id ?? source.message ?? \"\";\n const message = source.message ?? source.id ?? \"\";\n return { msgid: id, msgstr: message };\n}\n\nexport type PluralArgs = [...forms: Message[], n: number];\n\nexport function plural(...args: PluralArgs): PluralMessage {\n assert(args.length > 1, \"At least one plural form and n are required\");\n\n const n = args[args.length - 1] as number;\n assert(typeof n === \"number\", \"The last argument must be a number\");\n\n const forms = args.slice(0, -1) as Message[];\n\n return { forms, n };\n}\n\nexport type PluralFunction = (...args: PluralArgs) => PluralMessage;\nexport type MessageInput<T extends string = string> = [Message] | MessageArgs<T>;\nexport type PluralInput = [PluralMessage] | PluralArgs;\n\nexport type ContextMessageFunction = <T extends string>(...args: MessageArgs<T>) => ContextMessage;\n\nexport type ContextPluralFunction = (...args: PluralArgs) => ContextPluralMessage;\n\nexport interface ContextBuilder {\n message: ContextMessageFunction;\n plural: ContextPluralFunction;\n}\n\nconst baseMessage = message;\nconst basePlural = plural;\n\nexport function context<T extends string>(context: StrictStaticString<T>): ContextBuilder;\nexport function context(strings: TemplateStringsArray, ...values: never[]): ContextBuilder;\nexport function context<T extends string>(\n ...args: [StrictStaticString<T>] | [TemplateStringsArray, ...never[]]\n): ContextBuilder {\n const [source] = args as [StrictStaticString<T> | TemplateStringsArray];\n\n const ctx = typeof source === \"string\" ? source : source[0];\n\n function message<T extends string>(...args: MessageArgs<T>): ContextMessage {\n return { id: baseMessage(...args), context: ctx };\n }\n\n function plural(...forms: PluralArgs): ContextPluralMessage {\n return { id: basePlural(...forms), context: ctx };\n }\n\n return {\n message,\n plural,\n };\n}\n","const isSymbol = (value) => {\n return !!value && value.constructor === Symbol;\n};\nconst isArray = Array.isArray;\nconst isObject = (value) => {\n return !!value && value.constructor === Object;\n};\nconst isPrimitive = (value) => {\n return value === void 0 || value === null || typeof value !== \"object\" && typeof value !== \"function\";\n};\nconst isFunction = (value) => {\n return !!(value && value.constructor && value.call && value.apply);\n};\nconst isString = (value) => {\n return typeof value === \"string\" || value instanceof String;\n};\nconst isInt = (value) => {\n return isNumber(value) && value % 1 === 0;\n};\nconst isFloat = (value) => {\n return isNumber(value) && value % 1 !== 0;\n};\nconst isNumber = (value) => {\n try {\n return Number(value) === value;\n } catch {\n return false;\n }\n};\nconst isDate = (value) => {\n return Object.prototype.toString.call(value) === \"[object Date]\";\n};\nconst isPromise = (value) => {\n if (!value)\n return false;\n if (!value.then)\n return false;\n if (!isFunction(value.then))\n return false;\n return true;\n};\nconst isEmpty = (value) => {\n if (value === true || value === false)\n return true;\n if (value === null || value === void 0)\n return true;\n if (isNumber(value))\n return value === 0;\n if (isDate(value))\n return isNaN(value.getTime());\n if (isFunction(value))\n return false;\n if (isSymbol(value))\n return false;\n const length = value.length;\n if (isNumber(length))\n return length === 0;\n const size = value.size;\n if (isNumber(size))\n return size === 0;\n const keys = Object.keys(value).length;\n return keys === 0;\n};\nconst isEqual = (x, y) => {\n if (Object.is(x, y))\n return true;\n if (x instanceof Date && y instanceof Date) {\n return x.getTime() === y.getTime();\n }\n if (x instanceof RegExp && y instanceof RegExp) {\n return x.toString() === y.toString();\n }\n if (typeof x !== \"object\" || x === null || typeof y !== \"object\" || y === null) {\n return false;\n }\n const keysX = Reflect.ownKeys(x);\n const keysY = Reflect.ownKeys(y);\n if (keysX.length !== keysY.length)\n return false;\n for (let i = 0; i < keysX.length; i++) {\n if (!Reflect.has(y, keysX[i]))\n return false;\n if (!isEqual(x[keysX[i]], y[keysX[i]]))\n return false;\n }\n return true;\n};\n\nexport { isArray, isDate, isEmpty, isEqual, isFloat, isFunction, isInt, isNumber, isObject, isPrimitive, isPromise, isString, isSymbol };\n//# sourceMappingURL=typed.mjs.map\n","import { objectify } from './array.mjs';\nimport { isPrimitive, isObject, isArray } from './typed.mjs';\n\nconst shake = (obj, filter = (x) => x === void 0) => {\n if (!obj)\n return {};\n const keys2 = Object.keys(obj);\n return keys2.reduce((acc, key) => {\n if (filter(obj[key])) {\n return acc;\n } else {\n acc[key] = obj[key];\n return acc;\n }\n }, {});\n};\nconst mapKeys = (obj, mapFunc) => {\n const keys2 = Object.keys(obj);\n return keys2.reduce((acc, key) => {\n acc[mapFunc(key, obj[key])] = obj[key];\n return acc;\n }, {});\n};\nconst mapValues = (obj, mapFunc) => {\n const keys2 = Object.keys(obj);\n return keys2.reduce((acc, key) => {\n acc[key] = mapFunc(obj[key], key);\n return acc;\n }, {});\n};\nconst mapEntries = (obj, toEntry) => {\n if (!obj)\n return {};\n return Object.entries(obj).reduce((acc, [key, value]) => {\n const [newKey, newValue] = toEntry(key, value);\n acc[newKey] = newValue;\n return acc;\n }, {});\n};\nconst invert = (obj) => {\n if (!obj)\n return {};\n const keys2 = Object.keys(obj);\n return keys2.reduce((acc, key) => {\n acc[obj[key]] = key;\n return acc;\n }, {});\n};\nconst lowerize = (obj) => mapKeys(obj, (k) => k.toLowerCase());\nconst upperize = (obj) => mapKeys(obj, (k) => k.toUpperCase());\nconst clone = (obj) => {\n if (isPrimitive(obj)) {\n return obj;\n }\n if (typeof obj === \"function\") {\n return obj.bind({});\n }\n const newObj = new obj.constructor();\n Object.getOwnPropertyNames(obj).forEach((prop) => {\n newObj[prop] = obj[prop];\n });\n return newObj;\n};\nconst listify = (obj, toItem) => {\n if (!obj)\n return [];\n const entries = Object.entries(obj);\n if (entries.length === 0)\n return [];\n return entries.reduce((acc, entry) => {\n acc.push(toItem(entry[0], entry[1]));\n return acc;\n }, []);\n};\nconst pick = (obj, keys2) => {\n if (!obj)\n return {};\n return keys2.reduce((acc, key) => {\n if (Object.prototype.hasOwnProperty.call(obj, key))\n acc[key] = obj[key];\n return acc;\n }, {});\n};\nconst omit = (obj, keys2) => {\n if (!obj)\n return {};\n if (!keys2 || keys2.length === 0)\n return obj;\n return keys2.reduce(\n (acc, key) => {\n delete acc[key];\n return acc;\n },\n { ...obj }\n );\n};\nconst get = (value, path, defaultValue) => {\n const segments = path.split(/[\\.\\[\\]]/g);\n let current = value;\n for (const key of segments) {\n if (current === null)\n return defaultValue;\n if (current === void 0)\n return defaultValue;\n const dequoted = key.replace(/['\"]/g, \"\");\n if (dequoted.trim() === \"\")\n continue;\n current = current[dequoted];\n }\n if (current === void 0)\n return defaultValue;\n return current;\n};\nconst set = (initial, path, value) => {\n if (!initial)\n return {};\n if (!path || value === void 0)\n return initial;\n const segments = path.split(/[\\.\\[\\]]/g).filter((x) => !!x.trim());\n const _set = (node) => {\n if (segments.length > 1) {\n const key = segments.shift();\n const nextIsNum = /^\\d+$/.test(segments[0]);\n node[key] = node[key] === void 0 ? nextIsNum ? [] : {} : node[key];\n _set(node[key]);\n } else {\n node[segments[0]] = value;\n }\n };\n const cloned = clone(initial);\n _set(cloned);\n return cloned;\n};\nconst assign = (initial, override) => {\n if (!initial || !override)\n return initial ?? override ?? {};\n return Object.entries({ ...initial, ...override }).reduce(\n (acc, [key, value]) => {\n return {\n ...acc,\n [key]: (() => {\n if (isObject(initial[key]))\n return assign(initial[key], value);\n return value;\n })()\n };\n },\n {}\n );\n};\nconst keys = (value) => {\n if (!value)\n return [];\n const getKeys = (nested, paths) => {\n if (isObject(nested)) {\n return Object.entries(nested).flatMap(\n ([k, v]) => getKeys(v, [...paths, k])\n );\n }\n if (isArray(nested)) {\n return nested.flatMap((item, i) => getKeys(item, [...paths, `${i}`]));\n }\n return [paths.join(\".\")];\n };\n return getKeys(value, []);\n};\nconst crush = (value) => {\n if (!value)\n return {};\n return objectify(\n keys(value),\n (k) => k,\n (k) => get(value, k)\n );\n};\nconst construct = (obj) => {\n if (!obj)\n return {};\n return Object.keys(obj).reduce((acc, path) => {\n return set(acc, path, obj[path]);\n }, {});\n};\n\nexport { assign, clone, construct, crush, get, invert, keys, listify, lowerize, mapEntries, mapKeys, mapValues, omit, pick, set, shake, upperize };\n//# sourceMappingURL=object.mjs.map\n","import type { GetTextTranslations } from \"gettext-parser\";\nimport { assign } from \"radash\";\nimport type { Locale } from \"./config.ts\";\n\nimport {\n message as buildMessage,\n plural as buildPlural,\n type ContextMessage,\n type ContextPluralMessage,\n type Message,\n type MessageArgs,\n type MessageInput,\n type PluralArgs,\n type PluralInput,\n type PluralMessage,\n} from \"./messages.ts\";\nimport { normalizeTranslations, pluralFunc, type StrictStaticString, substitute } from \"./utils.ts\";\n\ntype TranslationModule = GetTextTranslations | { default: GetTextTranslations };\ntype TranslationLoader = () => Promise<TranslationModule>;\ntype TranslationEntry = TranslationModule | Promise<TranslationModule> | TranslationLoader;\ntype TranslationRecord = Partial<Record<Locale, TranslationEntry>>;\ntype SyncLocaleKeys<T> = {\n [K in keyof T]: T[K] extends GetTextTranslations ? K : never;\n}[keyof T];\n\ntype ContextMessageInput<T extends string> = [ContextMessage] | MessageInput<T>;\ntype ContextPluralInput = [ContextPluralMessage] | PluralInput;\n\nfunction resolveTranslationModule(module: TranslationModule): GetTextTranslations {\n return \"default\" in module ? module.default : module;\n}\n\nexport class LocaleTranslator {\n locale: Locale;\n translations?: GetTextTranslations;\n\n constructor(locale: Locale, translations?: GetTextTranslations) {\n this.locale = locale;\n this.translations = translations ? normalizeTranslations(translations) : undefined;\n }\n\n private translateMessage({ msgid, msgstr, values }: Message, msgctxt = \"\"): string {\n const translated = this.translations?.translations?.[msgctxt]?.[msgid];\n const result = translated?.msgstr?.[0] || msgstr;\n return values ? substitute(result, values) : result;\n }\n\n private translatePlural({ forms, n }: PluralMessage, msgctxt = \"\"): string {\n const entry = this.translations?.translations?.[msgctxt]?.[forms[0].msgid];\n const index = pluralFunc(this.locale)(n);\n const form = forms[index] ?? forms[forms.length - 1];\n const translated = entry?.msgstr\n ? entry.msgstr[index] ?? entry.msgstr[entry.msgstr.length - 1]\n : undefined;\n const result = translated || form.msgstr;\n const usedVals = form.values?.length ? form.values : forms[0].values;\n return usedVals?.length ? substitute(result, usedVals) : result;\n }\n\n message = <T extends string>(...args: MessageInput<T>): string => {\n const [source] = args;\n if (typeof source === \"object\" && \"msgid\" in source) {\n return this.translateMessage(source);\n }\n return this.translateMessage(buildMessage(...(args as MessageArgs<T>)));\n };\n\n plural = (...args: PluralInput): string => {\n const [source] = args;\n if (typeof source === \"object\" && \"forms\" in source) {\n return this.translatePlural(source);\n }\n return this.translatePlural(buildPlural(...(args as PluralArgs)));\n };\n\n context<T extends string>(\n context: StrictStaticString<T>,\n ): {\n message: <T extends string>(...args: ContextMessageInput<T>) => string;\n plural: (...args: ContextPluralInput) => string;\n };\n context(\n strings: TemplateStringsArray,\n ...values: never[]\n ): {\n message: <T extends string>(...args: ContextMessageInput<T>) => string;\n plural: (...args: ContextPluralInput) => string;\n };\n context<T extends string>(...args: [StrictStaticString<T>] | [TemplateStringsArray, ...never[]]) {\n const [source] = args as [StrictStaticString<T> | TemplateStringsArray];\n\n const ctx = typeof source === \"string\" ? source : source[0];\n\n return {\n message: <T extends string>(...args: ContextMessageInput<T>): string => {\n const [src] = args;\n if (typeof src === \"object\") {\n if (\"id\" in src && \"context\" in src) {\n return this.translateMessage(src.id, src.context);\n }\n if (\"msgid\" in src) {\n return this.translateMessage(src, ctx);\n }\n }\n return this.translateMessage(buildMessage(...(args as MessageArgs<T>)), ctx);\n },\n plural: (...args: ContextPluralInput): string => {\n const [src] = args;\n if (typeof src === \"object\") {\n if (\"id\" in src && \"context\" in src) {\n return this.translatePlural(src.id, src.context);\n }\n if (\"forms\" in src) {\n return this.translatePlural(src, ctx);\n }\n }\n return this.translatePlural(buildPlural(...(args as PluralArgs)), ctx);\n },\n };\n }\n\n gettext<T extends string>(...args: MessageInput<T>): string {\n return this.message(...args);\n }\n\n ngettext(...args: PluralInput): string {\n return this.plural(...args);\n }\n\n pgettext<C extends string, T extends string>(\n context: ContextMessage | StrictStaticString<C>,\n ...args: MessageInput<T> | []\n ): string {\n if (typeof context === \"object\") {\n return this.translateMessage(context.id, context.context);\n }\n return this.context(context).message(...(args as ContextMessageInput<T>));\n }\n\n npgettext<C extends string>(\n context: ContextPluralMessage | StrictStaticString<C>,\n ...args: PluralInput | []\n ): string {\n if (typeof context === \"object\") {\n return this.translatePlural(context.id, context.context);\n }\n return this.context(context).plural(...(args as ContextPluralInput));\n }\n}\n\nexport class Translator<T extends TranslationRecord = TranslationRecord> {\n parent: Translator | undefined;\n loaders: Partial<Record<Locale, TranslationLoader>> = {};\n translations: Partial<Record<Locale, GetTextTranslations>> = {};\n pending: Partial<Record<Locale, Promise<LocaleTranslator>>> = {};\n translators: Partial<Record<Locale, LocaleTranslator>> = {};\n\n constructor(translations: T, parent?: Translator) {\n this.parent = parent;\n for (const [locale, value] of Object.entries(translations) as [Locale, TranslationEntry][]) {\n if (!value) continue;\n if (typeof value === \"function\") {\n this.loaders[locale] = value;\n } else if (\"then\" in value) {\n this.loaders[locale] = () => value;\n } else {\n this.translations[locale] = resolveTranslationModule(value);\n }\n }\n }\n\n async loadLocale(locale: Locale, loader?: TranslationLoader): Promise<LocaleTranslator> {\n this.loaders[locale] = loader ?? this.loaders[locale];\n\n return this.fetchLocale(locale as never);\n }\n\n getLocale<L extends SyncLocaleKeys<T>>(locale: L): LocaleTranslator {\n const key = locale as Locale;\n const existing = this.translators[key];\n if (existing) {\n return existing;\n }\n\n const base = this.translations[key];\n if (base) {\n const parentTranslations = this.parent?.getLocale(locale as never)?.translations;\n const translations = assign<GetTextTranslations>(base, parentTranslations as never);\n delete this.translations[key];\n\n const translator = new LocaleTranslator(key, translations);\n this.translators[key] = translator;\n return translator;\n }\n\n if (this.pending[key] || this.loaders[key]) {\n throw new Error(\"async locale cannot be loaded synchronously\");\n }\n\n this.translators[key] ??= new LocaleTranslator(key);\n return this.translators[key];\n }\n\n fetchLocale<L extends keyof T>(locale: L) {\n const key = locale as Locale;\n\n const pending = this.pending[key];\n if (pending) {\n return pending;\n }\n\n const loader = this.loaders[key];\n if (loader) {\n const promise = loader().then((translations) => {\n this.translations[key] ??= resolveTranslationModule(translations);\n delete this.pending[key];\n\n return this.getLocale(key as never);\n });\n this.pending[key] = promise;\n delete this.loaders[key];\n\n return promise;\n }\n\n return this.getLocale(key as never);\n }\n}\n"],"x_google_ignoreList":[2,3],"mappings":"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"}
|