@fluenti/cli 0.2.1 → 0.3.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.
- package/dist/ai-provider.d.ts +14 -0
- package/dist/ai-provider.d.ts.map +1 -0
- package/dist/catalog.d.ts +1 -1
- package/dist/catalog.d.ts.map +1 -1
- package/dist/check.d.ts +47 -0
- package/dist/check.d.ts.map +1 -0
- package/dist/cli.cjs +12 -9
- package/dist/cli.cjs.map +1 -1
- package/dist/cli.js +631 -194
- package/dist/cli.js.map +1 -1
- package/dist/compile-BZGNcCK1.js +192 -0
- package/dist/compile-BZGNcCK1.js.map +1 -0
- package/dist/compile-DoeVtT7S.cjs +8 -0
- package/dist/compile-DoeVtT7S.cjs.map +1 -0
- package/dist/compile-cache.d.ts +26 -0
- package/dist/compile-cache.d.ts.map +1 -0
- package/dist/compile-runner.d.ts +4 -1
- package/dist/compile-runner.d.ts.map +1 -1
- package/dist/compile-worker.cjs +2 -0
- package/dist/compile-worker.cjs.map +1 -0
- package/dist/compile-worker.d.ts +18 -0
- package/dist/compile-worker.d.ts.map +1 -0
- package/dist/compile-worker.js +14 -0
- package/dist/compile-worker.js.map +1 -0
- package/dist/compile.d.ts.map +1 -1
- package/dist/config-loader.d.ts +1 -6
- package/dist/config-loader.d.ts.map +1 -1
- package/dist/config.d.ts +2 -2
- package/dist/config.d.ts.map +1 -1
- package/dist/extract-cache-1XVokN31.js +304 -0
- package/dist/extract-cache-1XVokN31.js.map +1 -0
- package/dist/extract-cache-BVyzqc__.cjs +10 -0
- package/dist/extract-cache-BVyzqc__.cjs.map +1 -0
- package/dist/extract-cache.d.ts +33 -0
- package/dist/extract-cache.d.ts.map +1 -0
- package/dist/extract-runner.d.ts +12 -0
- package/dist/extract-runner.d.ts.map +1 -0
- package/dist/glossary.d.ts +5 -0
- package/dist/glossary.d.ts.map +1 -0
- package/dist/index.cjs +1 -1
- package/dist/index.cjs.map +1 -1
- package/dist/index.d.ts +5 -1
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +117 -25
- package/dist/index.js.map +1 -1
- package/dist/lint.d.ts +36 -0
- package/dist/lint.d.ts.map +1 -0
- package/dist/parallel-compile.d.ts +26 -0
- package/dist/parallel-compile.d.ts.map +1 -0
- package/dist/translate.d.ts.map +1 -1
- package/dist/{tsx-extractor-CcFjsYI-.js → tsx-extractor-B9fnGNTG.js} +61 -53
- package/dist/tsx-extractor-B9fnGNTG.js.map +1 -0
- package/dist/tsx-extractor-Bi9AXMhH.cjs +2 -0
- package/dist/tsx-extractor-Bi9AXMhH.cjs.map +1 -0
- package/dist/tsx-extractor.d.ts +2 -2
- package/dist/tsx-extractor.d.ts.map +1 -1
- package/dist/validation.d.ts +16 -0
- package/dist/validation.d.ts.map +1 -0
- package/dist/vue-extractor.cjs +2 -2
- package/dist/vue-extractor.cjs.map +1 -1
- package/dist/vue-extractor.d.ts +2 -2
- package/dist/vue-extractor.d.ts.map +1 -1
- package/dist/vue-extractor.js +42 -42
- package/dist/vue-extractor.js.map +1 -1
- package/llms-full.txt +297 -0
- package/llms.txt +86 -0
- package/package.json +4 -3
- package/dist/config-loader-BgAoTfxH.js +0 -387
- package/dist/config-loader-BgAoTfxH.js.map +0 -1
- package/dist/config-loader-D3RGkK_r.cjs +0 -16
- package/dist/config-loader-D3RGkK_r.cjs.map +0 -1
- package/dist/tsx-extractor-CcFjsYI-.js.map +0 -1
- package/dist/tsx-extractor-D__s_cP8.cjs +0 -2
- package/dist/tsx-extractor-D__s_cP8.cjs.map +0 -1
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { createMessageId as e, isSourceNode as t, parseSourceModule as n, walkSourceAst as r } from "@fluenti/core/
|
|
1
|
+
import { createMessageId as e, isSourceNode as t, parseSourceModule as n, walkSourceAst as r } from "@fluenti/core/transform";
|
|
2
2
|
//#region src/tsx-extractor.ts
|
|
3
3
|
var i = new Set([
|
|
4
4
|
"@fluenti/react",
|
|
@@ -44,25 +44,27 @@ function s(e, t, n) {
|
|
|
44
44
|
}
|
|
45
45
|
};
|
|
46
46
|
}
|
|
47
|
-
function c(t) {
|
|
48
|
-
if (t.message) return
|
|
49
|
-
|
|
47
|
+
function c(t, n) {
|
|
48
|
+
if (!t.message) return;
|
|
49
|
+
let r = n ?? e;
|
|
50
|
+
return {
|
|
51
|
+
id: t.id ?? r(t.message, t.context),
|
|
50
52
|
message: t.message,
|
|
51
53
|
...t.context === void 0 ? {} : { context: t.context },
|
|
52
54
|
...t.comment === void 0 ? {} : { comment: t.comment }
|
|
53
55
|
};
|
|
54
56
|
}
|
|
55
|
-
function l(e) {
|
|
56
|
-
if (e.type === "StringLiteral") return c({ message: e.value });
|
|
57
|
+
function l(e, t) {
|
|
58
|
+
if (e.type === "StringLiteral") return c({ message: e.value }, t);
|
|
57
59
|
if (e.type === "TemplateLiteral") {
|
|
58
|
-
let
|
|
59
|
-
return
|
|
60
|
+
let n = e;
|
|
61
|
+
return n.expressions.length === 0 ? c({ message: n.quasis.map((e) => e.value.cooked ?? e.value.raw).join("") }, t) : void 0;
|
|
60
62
|
}
|
|
61
63
|
if (e.type !== "ObjectExpression") return;
|
|
62
|
-
let
|
|
63
|
-
for (let
|
|
64
|
-
if (
|
|
65
|
-
let e =
|
|
64
|
+
let n = {};
|
|
65
|
+
for (let t of e.properties) {
|
|
66
|
+
if (t.type !== "ObjectProperty") continue;
|
|
67
|
+
let e = t;
|
|
66
68
|
if (e.computed || !T(e.key)) continue;
|
|
67
69
|
let r = e.key.name;
|
|
68
70
|
if (![
|
|
@@ -72,9 +74,9 @@ function l(e) {
|
|
|
72
74
|
"comment"
|
|
73
75
|
].includes(r)) continue;
|
|
74
76
|
let i = p(e.value);
|
|
75
|
-
i !== void 0 && (
|
|
77
|
+
i !== void 0 && (n[r] = i);
|
|
76
78
|
}
|
|
77
|
-
if (
|
|
79
|
+
if (n.message) return c(n, t);
|
|
78
80
|
}
|
|
79
81
|
function u(e) {
|
|
80
82
|
let t = [
|
|
@@ -180,67 +182,73 @@ function g(e, t) {
|
|
|
180
182
|
}
|
|
181
183
|
return n;
|
|
182
184
|
}
|
|
183
|
-
function _(t, n) {
|
|
184
|
-
let
|
|
185
|
+
function _(t, n, r) {
|
|
186
|
+
let i = o(n.quasi.quasis.map((e) => e.value.cooked ?? e.value.raw), n.quasi.expressions.map((e) => e.start == null || e.end == null ? "" : t.slice(e.start, e.end)));
|
|
185
187
|
return {
|
|
186
|
-
id:
|
|
187
|
-
message:
|
|
188
|
+
id: (r ?? e)(i),
|
|
189
|
+
message: i
|
|
188
190
|
};
|
|
189
191
|
}
|
|
190
192
|
function v(e) {
|
|
191
|
-
let t = /* @__PURE__ */ new
|
|
192
|
-
for (let e of n) if (b(e) && i.has(e.source.value)) for (let n of e.specifiers)
|
|
193
|
+
let t = /* @__PURE__ */ new Map(), n = Array.isArray(e.body) ? e.body : [];
|
|
194
|
+
for (let e of n) if (b(e) && i.has(e.source.value)) for (let n of e.specifiers) {
|
|
195
|
+
if (!x(n)) continue;
|
|
196
|
+
let e = S(n);
|
|
197
|
+
(e === "t" || e === "msg") && t.set(n.local.name, e);
|
|
198
|
+
}
|
|
193
199
|
return t;
|
|
194
200
|
}
|
|
195
|
-
function y(t, i) {
|
|
196
|
-
let
|
|
197
|
-
if (!
|
|
198
|
-
let
|
|
199
|
-
return r(
|
|
201
|
+
function y(t, i, a) {
|
|
202
|
+
let o = n(t);
|
|
203
|
+
if (!o) return [];
|
|
204
|
+
let c = [], f = v(o);
|
|
205
|
+
return r(o, (n) => {
|
|
200
206
|
if (n.type === "TaggedTemplateExpression") {
|
|
201
207
|
let e = n;
|
|
202
|
-
if (T(e.tag) && (e.tag.name === "t" ||
|
|
203
|
-
let n =
|
|
204
|
-
|
|
208
|
+
if (T(e.tag) && (e.tag.name === "t" || f.has(e.tag.name))) {
|
|
209
|
+
let n = _(t, e, a);
|
|
210
|
+
f.get(e.tag.name) === "msg" && (n.comment = "msg tagged template");
|
|
211
|
+
let r = s(n, i, e);
|
|
212
|
+
r && c.push(r);
|
|
205
213
|
}
|
|
206
214
|
return;
|
|
207
215
|
}
|
|
208
216
|
if (n.type === "CallExpression") {
|
|
209
|
-
let e = n;
|
|
210
|
-
if (T(e.callee) && (e.callee.name === "t" ||
|
|
211
|
-
if (
|
|
212
|
-
let t = e.arguments[0] ? l(e.arguments[0]) : void 0, n = t ? s(t, i, e) : void 0;
|
|
213
|
-
n &&
|
|
217
|
+
let e = n, t = f.get(e.callee && T(e.callee) ? e.callee.name : "");
|
|
218
|
+
if (T(e.callee) && (e.callee.name === "t" || f.has(e.callee.name) && t === "t")) {
|
|
219
|
+
if (f.has(e.callee.name) && e.arguments[0]?.type !== "ObjectExpression") return;
|
|
220
|
+
let t = e.arguments[0] ? l(e.arguments[0], a) : void 0, n = t ? s(t, i, e) : void 0;
|
|
221
|
+
n && c.push(n);
|
|
214
222
|
}
|
|
215
223
|
return;
|
|
216
224
|
}
|
|
217
225
|
if (n.type !== "JSXElement") return;
|
|
218
|
-
let r = n,
|
|
219
|
-
if (
|
|
220
|
-
let e = h(
|
|
226
|
+
let r = n, o = r.openingElement, m = C(o.name);
|
|
227
|
+
if (m === "Trans") {
|
|
228
|
+
let e = h(o, "message"), t = h(o, "id"), n = h(o, "context"), l = h(o, "comment"), u = e?.value ? w({
|
|
221
229
|
id: t?.value ? p(t.value) : void 0,
|
|
222
230
|
message: p(e.value),
|
|
223
231
|
context: n?.value ? p(n.value) : void 0,
|
|
224
|
-
comment:
|
|
225
|
-
}) : w({
|
|
232
|
+
comment: l?.value ? p(l.value) : void 0
|
|
233
|
+
}, a) : w({
|
|
226
234
|
id: t?.value ? p(t.value) : void 0,
|
|
227
235
|
message: d(r.children),
|
|
228
236
|
context: n?.value ? p(n.value) : void 0,
|
|
229
|
-
comment:
|
|
230
|
-
}),
|
|
231
|
-
|
|
237
|
+
comment: l?.value ? p(l.value) : void 0
|
|
238
|
+
}, a), f = u ? s(u, i, r) : void 0;
|
|
239
|
+
f && c.push(f);
|
|
232
240
|
return;
|
|
233
241
|
}
|
|
234
|
-
if (
|
|
235
|
-
let n = g(
|
|
236
|
-
if (!
|
|
237
|
-
let
|
|
238
|
-
id: n.id ??
|
|
239
|
-
message:
|
|
242
|
+
if (m === "Plural") {
|
|
243
|
+
let n = g(o, t), l = u(n);
|
|
244
|
+
if (!l) return;
|
|
245
|
+
let d = a ?? e, f = s({
|
|
246
|
+
id: n.id ?? d(l),
|
|
247
|
+
message: l
|
|
240
248
|
}, i, r);
|
|
241
|
-
|
|
249
|
+
f && c.push(f);
|
|
242
250
|
}
|
|
243
|
-
}),
|
|
251
|
+
}), c;
|
|
244
252
|
}
|
|
245
253
|
function b(e) {
|
|
246
254
|
return t(e) && e.type === "ImportDeclaration";
|
|
@@ -255,9 +263,9 @@ function S(e) {
|
|
|
255
263
|
function C(e) {
|
|
256
264
|
if (e.type === "JSXIdentifier") return String(e.name);
|
|
257
265
|
}
|
|
258
|
-
function w(e) {
|
|
259
|
-
let
|
|
260
|
-
return e.id !== void 0 && (
|
|
266
|
+
function w(e, t) {
|
|
267
|
+
let n = {};
|
|
268
|
+
return e.id !== void 0 && (n.id = e.id), e.message !== void 0 && (n.message = e.message), e.context !== void 0 && (n.context = e.context), e.comment !== void 0 && (n.comment = e.comment), c(n, t);
|
|
261
269
|
}
|
|
262
270
|
function T(e) {
|
|
263
271
|
return t(e) && e.type === "Identifier";
|
|
@@ -265,4 +273,4 @@ function T(e) {
|
|
|
265
273
|
//#endregion
|
|
266
274
|
export { y as t };
|
|
267
275
|
|
|
268
|
-
//# sourceMappingURL=tsx-extractor-
|
|
276
|
+
//# sourceMappingURL=tsx-extractor-B9fnGNTG.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"tsx-extractor-B9fnGNTG.js","names":[],"sources":["../src/tsx-extractor.ts"],"sourcesContent":["import type { ExtractedMessage } from '@fluenti/core/internal'\nimport {\n createMessageId,\n isSourceNode,\n parseSourceModule,\n walkSourceAst,\n type SourceNode,\n} from '@fluenti/core/transform'\n\ninterface IdentifierNode extends SourceNode {\n type: 'Identifier'\n name: string\n}\n\ninterface StringLiteralNode extends SourceNode {\n type: 'StringLiteral'\n value: string\n}\n\ninterface NumericLiteralNode extends SourceNode {\n type: 'NumericLiteral'\n value: number\n}\n\ninterface TemplateElementNode extends SourceNode {\n type: 'TemplateElement'\n value: { raw: string; cooked: string | null }\n}\n\ninterface TemplateLiteralNode extends SourceNode {\n type: 'TemplateLiteral'\n quasis: TemplateElementNode[]\n expressions: SourceNode[]\n}\n\ninterface TaggedTemplateExpressionNode extends SourceNode {\n type: 'TaggedTemplateExpression'\n tag: SourceNode\n quasi: TemplateLiteralNode\n}\n\ninterface CallExpressionNode extends SourceNode {\n type: 'CallExpression'\n callee: SourceNode\n arguments: SourceNode[]\n}\n\ninterface ImportDeclarationNode extends SourceNode {\n type: 'ImportDeclaration'\n source: StringLiteralNode\n specifiers: SourceNode[]\n}\n\ninterface ImportSpecifierNode extends SourceNode {\n type: 'ImportSpecifier'\n imported: IdentifierNode | StringLiteralNode\n local: IdentifierNode\n}\n\ninterface ObjectExpressionNode extends SourceNode {\n type: 'ObjectExpression'\n properties: SourceNode[]\n}\n\ninterface ObjectPropertyNode extends SourceNode {\n type: 'ObjectProperty'\n key: SourceNode\n value: SourceNode\n computed?: boolean\n}\n\ninterface JSXElementNode extends SourceNode {\n type: 'JSXElement'\n openingElement: JSXOpeningElementNode\n children: SourceNode[]\n}\n\ninterface JSXFragmentNode extends SourceNode {\n type: 'JSXFragment'\n children: SourceNode[]\n}\n\ninterface JSXOpeningElementNode extends SourceNode {\n type: 'JSXOpeningElement'\n name: SourceNode\n attributes: SourceNode[]\n}\n\ninterface JSXAttributeNode extends SourceNode {\n type: 'JSXAttribute'\n name: SourceNode\n value?: SourceNode | null\n}\n\ninterface JSXExpressionContainerNode extends SourceNode {\n type: 'JSXExpressionContainer'\n expression: SourceNode\n}\n\ninterface JSXTextNode extends SourceNode {\n type: 'JSXText'\n value: string\n}\n\ninterface ExtractedDescriptor {\n id: string\n message?: string\n context?: string\n comment?: string\n}\n\nconst DIRECT_T_SOURCES = new Set([\n '@fluenti/react',\n '@fluenti/vue',\n '@fluenti/solid',\n '@fluenti/next',\n])\n\nfunction classifyExpression(expr: string): string {\n const trimmed = expr.trim()\n // Simple identifier: name, count\n if (/^[a-zA-Z_$][a-zA-Z0-9_$]*$/.test(trimmed)) {\n return trimmed\n }\n // Dotted path: user.name → name\n if (/^[a-zA-Z_$][a-zA-Z0-9_$.]*$/.test(trimmed) && !trimmed.endsWith('.')) {\n const parts = trimmed.split('.')\n return parts[parts.length - 1]!\n }\n // Function call: fun() → fun, obj.method() → obj_method\n const callMatch = trimmed.match(/^([a-zA-Z_$][a-zA-Z0-9_$.]*)\\s*\\(/)\n if (callMatch) {\n return callMatch[1]!.replace(/\\./g, '_')\n }\n return ''\n}\n\nfunction buildICUFromTemplate(\n strings: readonly string[],\n expressions: readonly string[],\n): string {\n let result = ''\n let positionalIndex = 0\n\n for (let index = 0; index < strings.length; index++) {\n result += strings[index]!\n if (index >= expressions.length) continue\n\n const name = classifyExpression(expressions[index]!)\n if (name === '') {\n result += `{arg${positionalIndex}}`\n positionalIndex++\n continue\n }\n\n result += `{${name}}`\n }\n\n return result\n}\n\nfunction createExtractedMessage(\n descriptor: ExtractedDescriptor,\n filename: string,\n node: SourceNode,\n): ExtractedMessage | undefined {\n if (!descriptor.message) {\n return undefined\n }\n\n const line = node.loc?.start.line ?? 1\n const column = (node.loc?.start.column ?? 0) + 1\n\n return {\n id: descriptor.id,\n message: descriptor.message,\n ...(descriptor.context !== undefined ? { context: descriptor.context } : {}),\n ...(descriptor.comment !== undefined ? { comment: descriptor.comment } : {}),\n origin: { file: filename, line, column },\n }\n}\n\nfunction descriptorFromStaticParts(\n parts: {\n id?: string\n message?: string\n context?: string\n comment?: string\n },\n idGenerator?: (message: string, context?: string) => string,\n): ExtractedDescriptor | undefined {\n if (!parts.message) {\n return undefined\n }\n\n const generateId = idGenerator ?? createMessageId\n\n return {\n id: parts.id ?? generateId(parts.message, parts.context),\n message: parts.message,\n ...(parts.context !== undefined ? { context: parts.context } : {}),\n ...(parts.comment !== undefined ? { comment: parts.comment } : {}),\n }\n}\n\nfunction extractDescriptorFromCallArgument(\n argument: SourceNode,\n idGenerator?: (message: string, context?: string) => string,\n): ExtractedDescriptor | undefined {\n if (argument.type === 'StringLiteral') {\n return descriptorFromStaticParts({ message: (argument as StringLiteralNode).value }, idGenerator)\n }\n\n if (argument.type === 'TemplateLiteral') {\n const template = argument as TemplateLiteralNode\n if (template.expressions.length === 0) {\n const message = template.quasis.map((quasi) => quasi.value.cooked ?? quasi.value.raw).join('')\n return descriptorFromStaticParts({ message }, idGenerator)\n }\n return undefined\n }\n\n if (argument.type !== 'ObjectExpression') {\n return undefined\n }\n\n const staticParts: { id?: string; message?: string; context?: string; comment?: string } = {}\n for (const property of (argument as ObjectExpressionNode).properties) {\n if (property.type !== 'ObjectProperty') continue\n\n const objectProperty = property as ObjectPropertyNode\n if (objectProperty.computed || !isIdentifier(objectProperty.key)) continue\n\n const key = objectProperty.key.name\n if (!['id', 'message', 'context', 'comment'].includes(key)) continue\n\n const value = readStaticStringValue(objectProperty.value)\n if (value === undefined) continue\n staticParts[key as keyof typeof staticParts] = value\n }\n\n if (!staticParts.message) {\n return undefined\n }\n\n return descriptorFromStaticParts(staticParts, idGenerator)\n}\n\nfunction buildPluralICU(props: Record<string, string>): string {\n const categories = ['zero', 'one', 'two', 'few', 'many', 'other'] as const\n const countVar = props['value'] ?? props['count'] ?? 'count'\n const options: string[] = []\n const offset = props['offset']\n\n for (const category of categories) {\n const value = props[category]\n if (value === undefined) continue\n const key = category === 'zero' ? '=0' : category\n options.push(`${key} {${value}}`)\n }\n\n if (options.length === 0) {\n return ''\n }\n\n const offsetPrefix = offset ? `offset:${offset} ` : ''\n return `{${countVar}, plural, ${offsetPrefix}${options.join(' ')}}`\n}\n\nfunction extractRichTextMessage(children: readonly SourceNode[]): string | undefined {\n let nextIndex = 0\n\n function render(nodes: readonly SourceNode[]): string | undefined {\n let message = ''\n\n for (const node of nodes) {\n if (node.type === 'JSXText') {\n message += normalizeJsxText((node as JSXTextNode).value)\n continue\n }\n\n if (node.type === 'JSXElement') {\n const idx = nextIndex++\n const inner = render((node as JSXElementNode).children)\n if (inner === undefined) return undefined\n message += `<${idx}>${inner}</${idx}>`\n continue\n }\n\n if (node.type === 'JSXFragment') {\n const inner = render((node as JSXFragmentNode).children)\n if (inner === undefined) return undefined\n message += inner\n continue\n }\n\n if (node.type === 'JSXExpressionContainer') {\n const expression = (node as JSXExpressionContainerNode).expression\n if (expression.type === 'StringLiteral') {\n message += (expression as StringLiteralNode).value\n continue\n }\n if (expression.type === 'NumericLiteral') {\n message += String((expression as NumericLiteralNode).value)\n continue\n }\n return undefined\n }\n }\n\n return message\n }\n\n const message = render(children)\n if (message === undefined) return undefined\n\n const normalized = message.replace(/\\s+/g, ' ').trim()\n return normalized || undefined\n}\n\nfunction normalizeJsxText(value: string): string {\n return value.replace(/\\s+/g, ' ')\n}\n\nfunction readStaticStringValue(node: SourceNode): string | undefined {\n if (node.type === 'StringLiteral') {\n return (node as StringLiteralNode).value\n }\n\n if (node.type === 'NumericLiteral') {\n return String((node as NumericLiteralNode).value)\n }\n\n if (node.type === 'JSXExpressionContainer') {\n return readStaticStringValue((node as JSXExpressionContainerNode).expression)\n }\n\n if (node.type === 'TemplateLiteral') {\n const template = node as TemplateLiteralNode\n if (template.expressions.length === 0) {\n return template.quasis.map((quasi) => quasi.value.cooked ?? quasi.value.raw).join('')\n }\n }\n\n return undefined\n}\n\nfunction readExpressionSource(node: SourceNode, code: string): string | undefined {\n if (node.start == null || node.end == null) {\n return undefined\n }\n\n if (node.type === 'JSXExpressionContainer') {\n return readExpressionSource((node as JSXExpressionContainerNode).expression, code)\n }\n\n return code.slice(node.start, node.end).trim()\n}\n\nfunction getJsxAttribute(\n openingElement: JSXOpeningElementNode,\n name: string,\n): JSXAttributeNode | undefined {\n for (const attribute of openingElement.attributes) {\n if (attribute.type !== 'JSXAttribute') continue\n\n const jsxAttribute = attribute as JSXAttributeNode\n if (jsxAttribute.name.type === 'JSXIdentifier' && jsxAttribute.name['name'] === name) {\n return jsxAttribute\n }\n }\n\n return undefined\n}\n\nfunction extractPluralProps(\n openingElement: JSXOpeningElementNode,\n code: string,\n): Record<string, string> {\n const props: Record<string, string> = {}\n const propNames = ['id', 'value', 'count', 'offset', 'zero', 'one', 'two', 'few', 'many', 'other']\n\n for (const name of propNames) {\n const attribute = getJsxAttribute(openingElement, name)\n if (!attribute?.value) continue\n\n const staticValue = readStaticStringValue(attribute.value)\n if (staticValue !== undefined) {\n props[name] = staticValue\n continue\n }\n\n const exprValue = readExpressionSource(attribute.value, code)\n if (exprValue !== undefined && (name === 'value' || name === 'count' || name === 'offset')) {\n props[name] = exprValue\n }\n }\n\n return props\n}\n\nfunction extractTaggedTemplateMessage(\n code: string,\n node: TaggedTemplateExpressionNode,\n idGenerator?: (message: string, context?: string) => string,\n): ExtractedDescriptor {\n const strings = node.quasi.quasis.map((quasi) => quasi.value.cooked ?? quasi.value.raw)\n const expressions = node.quasi.expressions.map((expression) => {\n if (expression.start == null || expression.end == null) {\n return ''\n }\n return code.slice(expression.start, expression.end)\n })\n const message = buildICUFromTemplate(strings, expressions)\n const generateId = idGenerator ?? createMessageId\n\n return {\n id: generateId(message),\n message,\n }\n}\n\nfunction collectDirectBindings(ast: SourceNode): Map<string, 't' | 'msg'> {\n const bindings = new Map<string, 't' | 'msg'>()\n const body = Array.isArray(ast['body']) ? ast['body'] : []\n\n for (const entry of body) {\n if (!isImportDeclaration(entry)) continue\n if (!DIRECT_T_SOURCES.has(entry.source.value)) continue\n\n for (const specifier of entry.specifiers) {\n if (!isImportSpecifier(specifier)) continue\n const importedName = readImportedName(specifier)\n if (importedName === 't' || importedName === 'msg') {\n bindings.set(specifier.local.name, importedName as 't' | 'msg')\n }\n }\n }\n\n return bindings\n}\n\nexport function extractFromTsx(\n code: string,\n filename: string,\n idGenerator?: (message: string, context?: string) => string,\n): ExtractedMessage[] {\n const ast = parseSourceModule(code)\n if (!ast) {\n return []\n }\n\n const messages: ExtractedMessage[] = []\n const directBindings = collectDirectBindings(ast)\n\n walkSourceAst(ast, (node: SourceNode) => {\n if (node.type === 'TaggedTemplateExpression') {\n const tagged = node as TaggedTemplateExpressionNode\n if (\n isIdentifier(tagged.tag)\n && (tagged.tag.name === 't' || directBindings.has(tagged.tag.name))\n ) {\n const descriptor = extractTaggedTemplateMessage(code, tagged, idGenerator)\n const bindingType = directBindings.get(tagged.tag.name)\n if (bindingType === 'msg') {\n descriptor.comment = 'msg tagged template'\n }\n const extracted = createExtractedMessage(\n descriptor,\n filename,\n tagged,\n )\n if (extracted) {\n messages.push(extracted)\n }\n }\n return\n }\n\n if (node.type === 'CallExpression') {\n const call = node as CallExpressionNode\n const callBindingType = directBindings.get(call.callee && isIdentifier(call.callee) ? call.callee.name : '')\n if (isIdentifier(call.callee) && (call.callee.name === 't' || (directBindings.has(call.callee.name) && callBindingType === 't'))) {\n if (directBindings.has(call.callee.name) && call.arguments[0]?.type !== 'ObjectExpression') {\n return\n }\n const descriptor = call.arguments[0] ? extractDescriptorFromCallArgument(call.arguments[0], idGenerator) : undefined\n const extracted = descriptor\n ? createExtractedMessage(descriptor, filename, call)\n : undefined\n if (extracted) {\n messages.push(extracted)\n }\n }\n return\n }\n\n if (node.type !== 'JSXElement') {\n return\n }\n\n const element = node as JSXElementNode\n const openingElement = element.openingElement\n const elementName = readJsxElementName(openingElement.name)\n\n if (elementName === 'Trans') {\n const messageAttr = getJsxAttribute(openingElement, 'message')\n const idAttr = getJsxAttribute(openingElement, 'id')\n const contextAttr = getJsxAttribute(openingElement, 'context')\n const commentAttr = getJsxAttribute(openingElement, 'comment')\n\n const descriptor = messageAttr?.value\n ? buildStaticTransDescriptor({\n id: idAttr?.value ? readStaticStringValue(idAttr.value) : undefined,\n message: readStaticStringValue(messageAttr.value),\n context: contextAttr?.value ? readStaticStringValue(contextAttr.value) : undefined,\n comment: commentAttr?.value ? readStaticStringValue(commentAttr.value) : undefined,\n }, idGenerator)\n : buildStaticTransDescriptor({\n id: idAttr?.value ? readStaticStringValue(idAttr.value) : undefined,\n message: extractRichTextMessage(element.children),\n context: contextAttr?.value ? readStaticStringValue(contextAttr.value) : undefined,\n comment: commentAttr?.value ? readStaticStringValue(commentAttr.value) : undefined,\n }, idGenerator)\n\n const extracted = descriptor\n ? createExtractedMessage(descriptor, filename, element)\n : undefined\n if (extracted) {\n messages.push(extracted)\n }\n return\n }\n\n if (elementName === 'Plural') {\n const props = extractPluralProps(openingElement, code)\n const message = buildPluralICU(props)\n if (!message) {\n return\n }\n\n const generateId = idGenerator ?? createMessageId\n const extracted = createExtractedMessage(\n {\n id: props['id'] ?? generateId(message),\n message,\n },\n filename,\n element,\n )\n if (extracted) {\n messages.push(extracted)\n }\n }\n })\n\n return messages\n}\n\nfunction isImportDeclaration(node: unknown): node is ImportDeclarationNode {\n return isSourceNode(node) && node.type === 'ImportDeclaration'\n}\n\nfunction isImportSpecifier(node: unknown): node is ImportSpecifierNode {\n return isSourceNode(node) && node.type === 'ImportSpecifier'\n}\n\nfunction readImportedName(specifier: ImportSpecifierNode): string | undefined {\n if (specifier.imported.type === 'Identifier') {\n return (specifier.imported as IdentifierNode).name\n }\n if (specifier.imported.type === 'StringLiteral') {\n return (specifier.imported as StringLiteralNode).value\n }\n return undefined\n}\n\nfunction readJsxElementName(node: SourceNode): string | undefined {\n if (node.type === 'JSXIdentifier') {\n return String(node['name'])\n }\n return undefined\n}\n\nfunction buildStaticTransDescriptor(\n parts: {\n id: string | undefined\n message: string | undefined\n context: string | undefined\n comment: string | undefined\n },\n idGenerator?: (message: string, context?: string) => string,\n): ExtractedDescriptor | undefined {\n const payload: {\n id?: string\n message?: string\n context?: string\n comment?: string\n } = {}\n\n if (parts.id !== undefined) payload.id = parts.id\n if (parts.message !== undefined) payload.message = parts.message\n if (parts.context !== undefined) payload.context = parts.context\n if (parts.comment !== undefined) payload.comment = parts.comment\n\n return descriptorFromStaticParts(payload, idGenerator)\n}\n\nfunction isIdentifier(node: unknown): node is IdentifierNode {\n return isSourceNode(node) && node.type === 'Identifier'\n}\n"],"mappings":"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"}
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
require(`./extract-cache-BVyzqc__.cjs`);let e=require(`@fluenti/core/transform`);var t=new Set([`@fluenti/react`,`@fluenti/vue`,`@fluenti/solid`,`@fluenti/next`]);function n(e){let t=e.trim();if(/^[a-zA-Z_$][a-zA-Z0-9_$]*$/.test(t))return t;if(/^[a-zA-Z_$][a-zA-Z0-9_$.]*$/.test(t)&&!t.endsWith(`.`)){let e=t.split(`.`);return e[e.length-1]}let n=t.match(/^([a-zA-Z_$][a-zA-Z0-9_$.]*)\s*\(/);return n?n[1].replace(/\./g,`_`):``}function r(e,t){let r=``,i=0;for(let a=0;a<e.length;a++){if(r+=e[a],a>=t.length)continue;let o=n(t[a]);if(o===``){r+=`{arg${i}}`,i++;continue}r+=`{${o}}`}return r}function i(e,t,n){if(!e.message)return;let r=n.loc?.start.line??1,i=(n.loc?.start.column??0)+1;return{id:e.id,message:e.message,...e.context===void 0?{}:{context:e.context},...e.comment===void 0?{}:{comment:e.comment},origin:{file:t,line:r,column:i}}}function a(t,n){if(!t.message)return;let r=n??e.createMessageId;return{id:t.id??r(t.message,t.context),message:t.message,...t.context===void 0?{}:{context:t.context},...t.comment===void 0?{}:{comment:t.comment}}}function o(e,t){if(e.type===`StringLiteral`)return a({message:e.value},t);if(e.type===`TemplateLiteral`){let n=e;return n.expressions.length===0?a({message:n.quasis.map(e=>e.value.cooked??e.value.raw).join(``)},t):void 0}if(e.type!==`ObjectExpression`)return;let n={};for(let t of e.properties){if(t.type!==`ObjectProperty`)continue;let e=t;if(e.computed||!S(e.key))continue;let r=e.key.name;if(![`id`,`message`,`context`,`comment`].includes(r))continue;let i=u(e.value);i!==void 0&&(n[r]=i)}if(n.message)return a(n,t)}function s(e){let t=[`zero`,`one`,`two`,`few`,`many`,`other`],n=e.value??e.count??`count`,r=[],i=e.offset;for(let n of t){let t=e[n];if(t===void 0)continue;let i=n===`zero`?`=0`:n;r.push(`${i} {${t}}`)}return r.length===0?``:`{${n}, plural, ${i?`offset:${i} `:``}${r.join(` `)}}`}function c(e){let t=0;function n(e){let r=``;for(let i of e){if(i.type===`JSXText`){r+=l(i.value);continue}if(i.type===`JSXElement`){let e=t++,a=n(i.children);if(a===void 0)return;r+=`<${e}>${a}</${e}>`;continue}if(i.type===`JSXFragment`){let e=n(i.children);if(e===void 0)return;r+=e;continue}if(i.type===`JSXExpressionContainer`){let e=i.expression;if(e.type===`StringLiteral`){r+=e.value;continue}if(e.type===`NumericLiteral`){r+=String(e.value);continue}return}}return r}let r=n(e);if(r!==void 0)return r.replace(/\s+/g,` `).trim()||void 0}function l(e){return e.replace(/\s+/g,` `)}function u(e){if(e.type===`StringLiteral`)return e.value;if(e.type===`NumericLiteral`)return String(e.value);if(e.type===`JSXExpressionContainer`)return u(e.expression);if(e.type===`TemplateLiteral`){let t=e;if(t.expressions.length===0)return t.quasis.map(e=>e.value.cooked??e.value.raw).join(``)}}function d(e,t){if(!(e.start==null||e.end==null))return e.type===`JSXExpressionContainer`?d(e.expression,t):t.slice(e.start,e.end).trim()}function f(e,t){for(let n of e.attributes){if(n.type!==`JSXAttribute`)continue;let e=n;if(e.name.type===`JSXIdentifier`&&e.name.name===t)return e}}function p(e,t){let n={};for(let r of[`id`,`value`,`count`,`offset`,`zero`,`one`,`two`,`few`,`many`,`other`]){let i=f(e,r);if(!i?.value)continue;let a=u(i.value);if(a!==void 0){n[r]=a;continue}let o=d(i.value,t);o!==void 0&&(r===`value`||r===`count`||r===`offset`)&&(n[r]=o)}return n}function m(t,n,i){let a=r(n.quasi.quasis.map(e=>e.value.cooked??e.value.raw),n.quasi.expressions.map(e=>e.start==null||e.end==null?``:t.slice(e.start,e.end)));return{id:(i??e.createMessageId)(a),message:a}}function h(e){let n=new Map,r=Array.isArray(e.body)?e.body:[];for(let e of r)if(_(e)&&t.has(e.source.value))for(let t of e.specifiers){if(!v(t))continue;let e=y(t);(e===`t`||e===`msg`)&&n.set(t.local.name,e)}return n}function g(t,n,r){let a=(0,e.parseSourceModule)(t);if(!a)return[];let l=[],d=h(a);return(0,e.walkSourceAst)(a,a=>{if(a.type===`TaggedTemplateExpression`){let e=a;if(S(e.tag)&&(e.tag.name===`t`||d.has(e.tag.name))){let a=m(t,e,r);d.get(e.tag.name)===`msg`&&(a.comment=`msg tagged template`);let o=i(a,n,e);o&&l.push(o)}return}if(a.type===`CallExpression`){let e=a,t=d.get(e.callee&&S(e.callee)?e.callee.name:``);if(S(e.callee)&&(e.callee.name===`t`||d.has(e.callee.name)&&t===`t`)){if(d.has(e.callee.name)&&e.arguments[0]?.type!==`ObjectExpression`)return;let t=e.arguments[0]?o(e.arguments[0],r):void 0,a=t?i(t,n,e):void 0;a&&l.push(a)}return}if(a.type!==`JSXElement`)return;let h=a,g=h.openingElement,_=b(g.name);if(_===`Trans`){let e=f(g,`message`),t=f(g,`id`),a=f(g,`context`),o=f(g,`comment`),s=e?.value?x({id:t?.value?u(t.value):void 0,message:u(e.value),context:a?.value?u(a.value):void 0,comment:o?.value?u(o.value):void 0},r):x({id:t?.value?u(t.value):void 0,message:c(h.children),context:a?.value?u(a.value):void 0,comment:o?.value?u(o.value):void 0},r),d=s?i(s,n,h):void 0;d&&l.push(d);return}if(_===`Plural`){let a=p(g,t),o=s(a);if(!o)return;let c=r??e.createMessageId,u=i({id:a.id??c(o),message:o},n,h);u&&l.push(u)}}),l}function _(t){return(0,e.isSourceNode)(t)&&t.type===`ImportDeclaration`}function v(t){return(0,e.isSourceNode)(t)&&t.type===`ImportSpecifier`}function y(e){if(e.imported.type===`Identifier`)return e.imported.name;if(e.imported.type===`StringLiteral`)return e.imported.value}function b(e){if(e.type===`JSXIdentifier`)return String(e.name)}function x(e,t){let n={};return e.id!==void 0&&(n.id=e.id),e.message!==void 0&&(n.message=e.message),e.context!==void 0&&(n.context=e.context),e.comment!==void 0&&(n.comment=e.comment),a(n,t)}function S(t){return(0,e.isSourceNode)(t)&&t.type===`Identifier`}Object.defineProperty(exports,`t`,{enumerable:!0,get:function(){return g}});
|
|
2
|
+
//# sourceMappingURL=tsx-extractor-Bi9AXMhH.cjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"tsx-extractor-Bi9AXMhH.cjs","names":[],"sources":["../src/tsx-extractor.ts"],"sourcesContent":["import type { ExtractedMessage } from '@fluenti/core/internal'\nimport {\n createMessageId,\n isSourceNode,\n parseSourceModule,\n walkSourceAst,\n type SourceNode,\n} from '@fluenti/core/transform'\n\ninterface IdentifierNode extends SourceNode {\n type: 'Identifier'\n name: string\n}\n\ninterface StringLiteralNode extends SourceNode {\n type: 'StringLiteral'\n value: string\n}\n\ninterface NumericLiteralNode extends SourceNode {\n type: 'NumericLiteral'\n value: number\n}\n\ninterface TemplateElementNode extends SourceNode {\n type: 'TemplateElement'\n value: { raw: string; cooked: string | null }\n}\n\ninterface TemplateLiteralNode extends SourceNode {\n type: 'TemplateLiteral'\n quasis: TemplateElementNode[]\n expressions: SourceNode[]\n}\n\ninterface TaggedTemplateExpressionNode extends SourceNode {\n type: 'TaggedTemplateExpression'\n tag: SourceNode\n quasi: TemplateLiteralNode\n}\n\ninterface CallExpressionNode extends SourceNode {\n type: 'CallExpression'\n callee: SourceNode\n arguments: SourceNode[]\n}\n\ninterface ImportDeclarationNode extends SourceNode {\n type: 'ImportDeclaration'\n source: StringLiteralNode\n specifiers: SourceNode[]\n}\n\ninterface ImportSpecifierNode extends SourceNode {\n type: 'ImportSpecifier'\n imported: IdentifierNode | StringLiteralNode\n local: IdentifierNode\n}\n\ninterface ObjectExpressionNode extends SourceNode {\n type: 'ObjectExpression'\n properties: SourceNode[]\n}\n\ninterface ObjectPropertyNode extends SourceNode {\n type: 'ObjectProperty'\n key: SourceNode\n value: SourceNode\n computed?: boolean\n}\n\ninterface JSXElementNode extends SourceNode {\n type: 'JSXElement'\n openingElement: JSXOpeningElementNode\n children: SourceNode[]\n}\n\ninterface JSXFragmentNode extends SourceNode {\n type: 'JSXFragment'\n children: SourceNode[]\n}\n\ninterface JSXOpeningElementNode extends SourceNode {\n type: 'JSXOpeningElement'\n name: SourceNode\n attributes: SourceNode[]\n}\n\ninterface JSXAttributeNode extends SourceNode {\n type: 'JSXAttribute'\n name: SourceNode\n value?: SourceNode | null\n}\n\ninterface JSXExpressionContainerNode extends SourceNode {\n type: 'JSXExpressionContainer'\n expression: SourceNode\n}\n\ninterface JSXTextNode extends SourceNode {\n type: 'JSXText'\n value: string\n}\n\ninterface ExtractedDescriptor {\n id: string\n message?: string\n context?: string\n comment?: string\n}\n\nconst DIRECT_T_SOURCES = new Set([\n '@fluenti/react',\n '@fluenti/vue',\n '@fluenti/solid',\n '@fluenti/next',\n])\n\nfunction classifyExpression(expr: string): string {\n const trimmed = expr.trim()\n // Simple identifier: name, count\n if (/^[a-zA-Z_$][a-zA-Z0-9_$]*$/.test(trimmed)) {\n return trimmed\n }\n // Dotted path: user.name → name\n if (/^[a-zA-Z_$][a-zA-Z0-9_$.]*$/.test(trimmed) && !trimmed.endsWith('.')) {\n const parts = trimmed.split('.')\n return parts[parts.length - 1]!\n }\n // Function call: fun() → fun, obj.method() → obj_method\n const callMatch = trimmed.match(/^([a-zA-Z_$][a-zA-Z0-9_$.]*)\\s*\\(/)\n if (callMatch) {\n return callMatch[1]!.replace(/\\./g, '_')\n }\n return ''\n}\n\nfunction buildICUFromTemplate(\n strings: readonly string[],\n expressions: readonly string[],\n): string {\n let result = ''\n let positionalIndex = 0\n\n for (let index = 0; index < strings.length; index++) {\n result += strings[index]!\n if (index >= expressions.length) continue\n\n const name = classifyExpression(expressions[index]!)\n if (name === '') {\n result += `{arg${positionalIndex}}`\n positionalIndex++\n continue\n }\n\n result += `{${name}}`\n }\n\n return result\n}\n\nfunction createExtractedMessage(\n descriptor: ExtractedDescriptor,\n filename: string,\n node: SourceNode,\n): ExtractedMessage | undefined {\n if (!descriptor.message) {\n return undefined\n }\n\n const line = node.loc?.start.line ?? 1\n const column = (node.loc?.start.column ?? 0) + 1\n\n return {\n id: descriptor.id,\n message: descriptor.message,\n ...(descriptor.context !== undefined ? { context: descriptor.context } : {}),\n ...(descriptor.comment !== undefined ? { comment: descriptor.comment } : {}),\n origin: { file: filename, line, column },\n }\n}\n\nfunction descriptorFromStaticParts(\n parts: {\n id?: string\n message?: string\n context?: string\n comment?: string\n },\n idGenerator?: (message: string, context?: string) => string,\n): ExtractedDescriptor | undefined {\n if (!parts.message) {\n return undefined\n }\n\n const generateId = idGenerator ?? createMessageId\n\n return {\n id: parts.id ?? generateId(parts.message, parts.context),\n message: parts.message,\n ...(parts.context !== undefined ? { context: parts.context } : {}),\n ...(parts.comment !== undefined ? { comment: parts.comment } : {}),\n }\n}\n\nfunction extractDescriptorFromCallArgument(\n argument: SourceNode,\n idGenerator?: (message: string, context?: string) => string,\n): ExtractedDescriptor | undefined {\n if (argument.type === 'StringLiteral') {\n return descriptorFromStaticParts({ message: (argument as StringLiteralNode).value }, idGenerator)\n }\n\n if (argument.type === 'TemplateLiteral') {\n const template = argument as TemplateLiteralNode\n if (template.expressions.length === 0) {\n const message = template.quasis.map((quasi) => quasi.value.cooked ?? quasi.value.raw).join('')\n return descriptorFromStaticParts({ message }, idGenerator)\n }\n return undefined\n }\n\n if (argument.type !== 'ObjectExpression') {\n return undefined\n }\n\n const staticParts: { id?: string; message?: string; context?: string; comment?: string } = {}\n for (const property of (argument as ObjectExpressionNode).properties) {\n if (property.type !== 'ObjectProperty') continue\n\n const objectProperty = property as ObjectPropertyNode\n if (objectProperty.computed || !isIdentifier(objectProperty.key)) continue\n\n const key = objectProperty.key.name\n if (!['id', 'message', 'context', 'comment'].includes(key)) continue\n\n const value = readStaticStringValue(objectProperty.value)\n if (value === undefined) continue\n staticParts[key as keyof typeof staticParts] = value\n }\n\n if (!staticParts.message) {\n return undefined\n }\n\n return descriptorFromStaticParts(staticParts, idGenerator)\n}\n\nfunction buildPluralICU(props: Record<string, string>): string {\n const categories = ['zero', 'one', 'two', 'few', 'many', 'other'] as const\n const countVar = props['value'] ?? props['count'] ?? 'count'\n const options: string[] = []\n const offset = props['offset']\n\n for (const category of categories) {\n const value = props[category]\n if (value === undefined) continue\n const key = category === 'zero' ? '=0' : category\n options.push(`${key} {${value}}`)\n }\n\n if (options.length === 0) {\n return ''\n }\n\n const offsetPrefix = offset ? `offset:${offset} ` : ''\n return `{${countVar}, plural, ${offsetPrefix}${options.join(' ')}}`\n}\n\nfunction extractRichTextMessage(children: readonly SourceNode[]): string | undefined {\n let nextIndex = 0\n\n function render(nodes: readonly SourceNode[]): string | undefined {\n let message = ''\n\n for (const node of nodes) {\n if (node.type === 'JSXText') {\n message += normalizeJsxText((node as JSXTextNode).value)\n continue\n }\n\n if (node.type === 'JSXElement') {\n const idx = nextIndex++\n const inner = render((node as JSXElementNode).children)\n if (inner === undefined) return undefined\n message += `<${idx}>${inner}</${idx}>`\n continue\n }\n\n if (node.type === 'JSXFragment') {\n const inner = render((node as JSXFragmentNode).children)\n if (inner === undefined) return undefined\n message += inner\n continue\n }\n\n if (node.type === 'JSXExpressionContainer') {\n const expression = (node as JSXExpressionContainerNode).expression\n if (expression.type === 'StringLiteral') {\n message += (expression as StringLiteralNode).value\n continue\n }\n if (expression.type === 'NumericLiteral') {\n message += String((expression as NumericLiteralNode).value)\n continue\n }\n return undefined\n }\n }\n\n return message\n }\n\n const message = render(children)\n if (message === undefined) return undefined\n\n const normalized = message.replace(/\\s+/g, ' ').trim()\n return normalized || undefined\n}\n\nfunction normalizeJsxText(value: string): string {\n return value.replace(/\\s+/g, ' ')\n}\n\nfunction readStaticStringValue(node: SourceNode): string | undefined {\n if (node.type === 'StringLiteral') {\n return (node as StringLiteralNode).value\n }\n\n if (node.type === 'NumericLiteral') {\n return String((node as NumericLiteralNode).value)\n }\n\n if (node.type === 'JSXExpressionContainer') {\n return readStaticStringValue((node as JSXExpressionContainerNode).expression)\n }\n\n if (node.type === 'TemplateLiteral') {\n const template = node as TemplateLiteralNode\n if (template.expressions.length === 0) {\n return template.quasis.map((quasi) => quasi.value.cooked ?? quasi.value.raw).join('')\n }\n }\n\n return undefined\n}\n\nfunction readExpressionSource(node: SourceNode, code: string): string | undefined {\n if (node.start == null || node.end == null) {\n return undefined\n }\n\n if (node.type === 'JSXExpressionContainer') {\n return readExpressionSource((node as JSXExpressionContainerNode).expression, code)\n }\n\n return code.slice(node.start, node.end).trim()\n}\n\nfunction getJsxAttribute(\n openingElement: JSXOpeningElementNode,\n name: string,\n): JSXAttributeNode | undefined {\n for (const attribute of openingElement.attributes) {\n if (attribute.type !== 'JSXAttribute') continue\n\n const jsxAttribute = attribute as JSXAttributeNode\n if (jsxAttribute.name.type === 'JSXIdentifier' && jsxAttribute.name['name'] === name) {\n return jsxAttribute\n }\n }\n\n return undefined\n}\n\nfunction extractPluralProps(\n openingElement: JSXOpeningElementNode,\n code: string,\n): Record<string, string> {\n const props: Record<string, string> = {}\n const propNames = ['id', 'value', 'count', 'offset', 'zero', 'one', 'two', 'few', 'many', 'other']\n\n for (const name of propNames) {\n const attribute = getJsxAttribute(openingElement, name)\n if (!attribute?.value) continue\n\n const staticValue = readStaticStringValue(attribute.value)\n if (staticValue !== undefined) {\n props[name] = staticValue\n continue\n }\n\n const exprValue = readExpressionSource(attribute.value, code)\n if (exprValue !== undefined && (name === 'value' || name === 'count' || name === 'offset')) {\n props[name] = exprValue\n }\n }\n\n return props\n}\n\nfunction extractTaggedTemplateMessage(\n code: string,\n node: TaggedTemplateExpressionNode,\n idGenerator?: (message: string, context?: string) => string,\n): ExtractedDescriptor {\n const strings = node.quasi.quasis.map((quasi) => quasi.value.cooked ?? quasi.value.raw)\n const expressions = node.quasi.expressions.map((expression) => {\n if (expression.start == null || expression.end == null) {\n return ''\n }\n return code.slice(expression.start, expression.end)\n })\n const message = buildICUFromTemplate(strings, expressions)\n const generateId = idGenerator ?? createMessageId\n\n return {\n id: generateId(message),\n message,\n }\n}\n\nfunction collectDirectBindings(ast: SourceNode): Map<string, 't' | 'msg'> {\n const bindings = new Map<string, 't' | 'msg'>()\n const body = Array.isArray(ast['body']) ? ast['body'] : []\n\n for (const entry of body) {\n if (!isImportDeclaration(entry)) continue\n if (!DIRECT_T_SOURCES.has(entry.source.value)) continue\n\n for (const specifier of entry.specifiers) {\n if (!isImportSpecifier(specifier)) continue\n const importedName = readImportedName(specifier)\n if (importedName === 't' || importedName === 'msg') {\n bindings.set(specifier.local.name, importedName as 't' | 'msg')\n }\n }\n }\n\n return bindings\n}\n\nexport function extractFromTsx(\n code: string,\n filename: string,\n idGenerator?: (message: string, context?: string) => string,\n): ExtractedMessage[] {\n const ast = parseSourceModule(code)\n if (!ast) {\n return []\n }\n\n const messages: ExtractedMessage[] = []\n const directBindings = collectDirectBindings(ast)\n\n walkSourceAst(ast, (node: SourceNode) => {\n if (node.type === 'TaggedTemplateExpression') {\n const tagged = node as TaggedTemplateExpressionNode\n if (\n isIdentifier(tagged.tag)\n && (tagged.tag.name === 't' || directBindings.has(tagged.tag.name))\n ) {\n const descriptor = extractTaggedTemplateMessage(code, tagged, idGenerator)\n const bindingType = directBindings.get(tagged.tag.name)\n if (bindingType === 'msg') {\n descriptor.comment = 'msg tagged template'\n }\n const extracted = createExtractedMessage(\n descriptor,\n filename,\n tagged,\n )\n if (extracted) {\n messages.push(extracted)\n }\n }\n return\n }\n\n if (node.type === 'CallExpression') {\n const call = node as CallExpressionNode\n const callBindingType = directBindings.get(call.callee && isIdentifier(call.callee) ? call.callee.name : '')\n if (isIdentifier(call.callee) && (call.callee.name === 't' || (directBindings.has(call.callee.name) && callBindingType === 't'))) {\n if (directBindings.has(call.callee.name) && call.arguments[0]?.type !== 'ObjectExpression') {\n return\n }\n const descriptor = call.arguments[0] ? extractDescriptorFromCallArgument(call.arguments[0], idGenerator) : undefined\n const extracted = descriptor\n ? createExtractedMessage(descriptor, filename, call)\n : undefined\n if (extracted) {\n messages.push(extracted)\n }\n }\n return\n }\n\n if (node.type !== 'JSXElement') {\n return\n }\n\n const element = node as JSXElementNode\n const openingElement = element.openingElement\n const elementName = readJsxElementName(openingElement.name)\n\n if (elementName === 'Trans') {\n const messageAttr = getJsxAttribute(openingElement, 'message')\n const idAttr = getJsxAttribute(openingElement, 'id')\n const contextAttr = getJsxAttribute(openingElement, 'context')\n const commentAttr = getJsxAttribute(openingElement, 'comment')\n\n const descriptor = messageAttr?.value\n ? buildStaticTransDescriptor({\n id: idAttr?.value ? readStaticStringValue(idAttr.value) : undefined,\n message: readStaticStringValue(messageAttr.value),\n context: contextAttr?.value ? readStaticStringValue(contextAttr.value) : undefined,\n comment: commentAttr?.value ? readStaticStringValue(commentAttr.value) : undefined,\n }, idGenerator)\n : buildStaticTransDescriptor({\n id: idAttr?.value ? readStaticStringValue(idAttr.value) : undefined,\n message: extractRichTextMessage(element.children),\n context: contextAttr?.value ? readStaticStringValue(contextAttr.value) : undefined,\n comment: commentAttr?.value ? readStaticStringValue(commentAttr.value) : undefined,\n }, idGenerator)\n\n const extracted = descriptor\n ? createExtractedMessage(descriptor, filename, element)\n : undefined\n if (extracted) {\n messages.push(extracted)\n }\n return\n }\n\n if (elementName === 'Plural') {\n const props = extractPluralProps(openingElement, code)\n const message = buildPluralICU(props)\n if (!message) {\n return\n }\n\n const generateId = idGenerator ?? createMessageId\n const extracted = createExtractedMessage(\n {\n id: props['id'] ?? generateId(message),\n message,\n },\n filename,\n element,\n )\n if (extracted) {\n messages.push(extracted)\n }\n }\n })\n\n return messages\n}\n\nfunction isImportDeclaration(node: unknown): node is ImportDeclarationNode {\n return isSourceNode(node) && node.type === 'ImportDeclaration'\n}\n\nfunction isImportSpecifier(node: unknown): node is ImportSpecifierNode {\n return isSourceNode(node) && node.type === 'ImportSpecifier'\n}\n\nfunction readImportedName(specifier: ImportSpecifierNode): string | undefined {\n if (specifier.imported.type === 'Identifier') {\n return (specifier.imported as IdentifierNode).name\n }\n if (specifier.imported.type === 'StringLiteral') {\n return (specifier.imported as StringLiteralNode).value\n }\n return undefined\n}\n\nfunction readJsxElementName(node: SourceNode): string | undefined {\n if (node.type === 'JSXIdentifier') {\n return String(node['name'])\n }\n return undefined\n}\n\nfunction buildStaticTransDescriptor(\n parts: {\n id: string | undefined\n message: string | undefined\n context: string | undefined\n comment: string | undefined\n },\n idGenerator?: (message: string, context?: string) => string,\n): ExtractedDescriptor | undefined {\n const payload: {\n id?: string\n message?: string\n context?: string\n comment?: string\n } = {}\n\n if (parts.id !== undefined) payload.id = parts.id\n if (parts.message !== undefined) payload.message = parts.message\n if (parts.context !== undefined) payload.context = parts.context\n if (parts.comment !== undefined) payload.comment = parts.comment\n\n return descriptorFromStaticParts(payload, idGenerator)\n}\n\nfunction isIdentifier(node: unknown): node is IdentifierNode {\n return isSourceNode(node) && node.type === 'Identifier'\n}\n"],"mappings":"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"}
|
package/dist/tsx-extractor.d.ts
CHANGED
|
@@ -1,3 +1,3 @@
|
|
|
1
|
-
import { ExtractedMessage } from '@fluenti/core';
|
|
2
|
-
export declare function extractFromTsx(code: string, filename: string): ExtractedMessage[];
|
|
1
|
+
import { ExtractedMessage } from '@fluenti/core/internal';
|
|
2
|
+
export declare function extractFromTsx(code: string, filename: string, idGenerator?: (message: string, context?: string) => string): ExtractedMessage[];
|
|
3
3
|
//# sourceMappingURL=tsx-extractor.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"tsx-extractor.d.ts","sourceRoot":"","sources":["../src/tsx-extractor.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,
|
|
1
|
+
{"version":3,"file":"tsx-extractor.d.ts","sourceRoot":"","sources":["../src/tsx-extractor.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,wBAAwB,CAAA;AA0b9D,wBAAgB,cAAc,CAC5B,IAAI,EAAE,MAAM,EACZ,QAAQ,EAAE,MAAM,EAChB,WAAW,CAAC,EAAE,CAAC,OAAO,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE,MAAM,KAAK,MAAM,GAC1D,gBAAgB,EAAE,CA+GpB"}
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
export interface ValidationResult {
|
|
2
|
+
valid: boolean;
|
|
3
|
+
missingPlaceholders: string[];
|
|
4
|
+
extraPlaceholders: string[];
|
|
5
|
+
missingHtmlTags: string[];
|
|
6
|
+
extraHtmlTags: string[];
|
|
7
|
+
}
|
|
8
|
+
/** Extract unique ICU placeholder names from a message, sorted alphabetically */
|
|
9
|
+
export declare function extractPlaceholders(message: string): string[];
|
|
10
|
+
/** Extract unique HTML tag names from a message, lowercased and sorted */
|
|
11
|
+
export declare function extractHtmlTags(message: string): string[];
|
|
12
|
+
/** Validate that a translation preserves placeholders and HTML tags from the source */
|
|
13
|
+
export declare function validateTranslation(source: string, translation: string): ValidationResult;
|
|
14
|
+
/** Validate a batch of translations, returning only invalid entries */
|
|
15
|
+
export declare function validateBatch(sources: Record<string, string>, translations: Record<string, string>): Record<string, ValidationResult>;
|
|
16
|
+
//# sourceMappingURL=validation.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"validation.d.ts","sourceRoot":"","sources":["../src/validation.ts"],"names":[],"mappings":"AAAA,MAAM,WAAW,gBAAgB;IAC/B,KAAK,EAAE,OAAO,CAAA;IACd,mBAAmB,EAAE,MAAM,EAAE,CAAA;IAC7B,iBAAiB,EAAE,MAAM,EAAE,CAAA;IAC3B,eAAe,EAAE,MAAM,EAAE,CAAA;IACzB,aAAa,EAAE,MAAM,EAAE,CAAA;CACxB;AAED,iFAAiF;AACjF,wBAAgB,mBAAmB,CAAC,OAAO,EAAE,MAAM,GAAG,MAAM,EAAE,CAQ7D;AAED,0EAA0E;AAC1E,wBAAgB,eAAe,CAAC,OAAO,EAAE,MAAM,GAAG,MAAM,EAAE,CAQzD;AAED,uFAAuF;AACvF,wBAAgB,mBAAmB,CAAC,MAAM,EAAE,MAAM,EAAE,WAAW,EAAE,MAAM,GAAG,gBAAgB,CAqBzF;AAED,uEAAuE;AACvE,wBAAgB,aAAa,CAC3B,OAAO,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,EAC/B,YAAY,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,GACnC,MAAM,CAAC,MAAM,EAAE,gBAAgB,CAAC,CAclC"}
|
package/dist/vue-extractor.cjs
CHANGED
|
@@ -1,3 +1,3 @@
|
|
|
1
|
-
Object.defineProperty(exports,Symbol.toStringTag,{value:`Module`}),require(`./
|
|
2
|
-
`).length-1;for(let e of
|
|
1
|
+
Object.defineProperty(exports,Symbol.toStringTag,{value:`Module`}),require(`./extract-cache-BVyzqc__.cjs`);const e=require(`./tsx-extractor-Bi9AXMhH.cjs`);let t=require(`@fluenti/core/transform`),n=require(`@vue/compiler-sfc`);var r=1,i=2,a=7,o=6;function s(e){return e.filter(e=>e.type===i).map(e=>(e.content??``).trim()).join(``)}function c(e,t){let n=e.split(`|`).map(e=>e.trim()),r=[`one`,`other`,`zero`,`few`,`many`],i=[];if(n.length===2)i.push(`one {${n[0]}}`),i.push(`other {${n[1]}}`);else for(let e=0;e<n.length&&e<r.length;e++)i.push(`${r[e]} {${n[e]}}`);return`{${t}, plural, ${i.join(` `)}}`}function l(e){let t=e.count??`count`,n=[`zero`,`one`,`two`,`few`,`many`,`other`],r=[],i=e.offset;for(let t of n)if(e[t]!==void 0){let n=t===`zero`?`=0`:t;r.push(`${n} {${e[t]}}`)}return r.length===0?``:`{${t}, plural, ${i?`offset:${i} `:``}${r.join(` `)}}`}function u(e,n,i,p){if(e.type===r){let r=e.props?.find(e=>e.type===a&&f(e)===`t`);if(r){let a=new Set([`plural`]),o=(r.modifiers??[]).map(e=>typeof e==`string`?e:e.content),l=o.includes(`plural`),u=o.filter(e=>!a.has(e)),d=r.arg?.content,f=d?[d,...u].join(`.`):void 0,m=s(e.children??[]);if(l){let e=c(m,r.exp?.content??`count`),a=f??(p??t.createMessageId)(e);i.push({id:a,message:e,origin:{file:n,line:r.loc.start.line,column:r.loc.start.column}})}else if(m){let e=f??(p??t.createMessageId)(m);i.push({id:e,message:m,origin:{file:n,line:r.loc.start.line,column:r.loc.start.column}})}}if(e.tag===`Trans`){let r=e.props?.find(e=>e.type===o&&f(e)===`message`),a=e.props?.find(e=>e.type===o&&f(e)===`id`),s=e.props?.find(e=>e.type===o&&f(e)===`context`),c=e.props?.find(e=>e.type===o&&f(e)===`comment`),l=s?.value?.content,u=c?.value?.content;if(r?.value){let o=r.value.content,s=p??t.createMessageId,c=a?.value?.content??s(o,l);i.push({id:c,message:o,...l===void 0?{}:{context:l},...u===void 0?{}:{comment:u},origin:{file:n,line:e.loc.start.line,column:e.loc.start.column}})}else if(e.children&&e.children.length>0){let r=d(e.children);if(r.message){let o=p??t.createMessageId,s=a?.value?.content??o(r.message,l);i.push({id:s,message:r.message,...l===void 0?{}:{context:l},...u===void 0?{}:{comment:u},origin:{file:n,line:e.loc.start.line,column:e.loc.start.column}})}}}if(e.tag===`Plural`){let r={},s,c;for(let t of e.props??[])t.type===o&&t.value&&(r[f(t)]=t.value.content),t.type===a&&f(t)===`bind`&&t.arg?.content===`value`&&t.exp&&(s=t.exp.content),t.type===a&&f(t)===`bind`&&t.arg?.content===`offset`&&t.exp&&(c=t.exp.content);let u=s??r.count??`count`,d=c??r.offset,m=l({...r,count:u,...d===void 0?{}:{offset:d}});if(m){let a=r.id??(p??t.createMessageId)(m);i.push({id:a,message:m,origin:{file:n,line:e.loc.start.line,column:e.loc.start.column}})}}}if(e.children)for(let t of e.children)u(t,n,i,p)}function d(e){let t=0,n=!1;return{message:e.map(e=>{if(e.type===i)return(e.content??``).trim()?e.content??``:``;if(e.type===r&&e.tag){n=!0;let r=t++;return`<${r}>${d(e.children??[]).message}</${r}>`}return``}).join(``).trim(),hasElements:n}}function f(e){return typeof e.name==`string`?e.name:e.name.content}function p(t,n,r){let i=[],a=/\{\{([\s\S]*?)\}\}/g,o;for(;(o=a.exec(t))!==null;){let a=o[1]?.trim();if(!a)continue;let s=e.t(a,n,r);if(s.length===0)continue;let c=t.slice(0,o.index).split(`
|
|
2
|
+
`).length-1;for(let e of s)i.push({...e,origin:{...e.origin,line:e.origin.line+c}})}return i}function m(t,r,i){let a=[],{descriptor:o}=(0,n.parse)(t,{filename:r});if(o.template?.ast&&u(o.template.ast,r,a,i),o.template?.content){let t=e.t(o.template.content,r,i),n=o.template.loc.start.line-1,s=new Set(a.map(e=>e.id));for(let e of t)s.has(e.id)||a.push({...e,origin:{...e.origin,line:e.origin.line+n}});let c=p(o.template.content,r,i);for(let e of c)s.has(e.id)||a.push({...e,origin:{...e.origin,line:e.origin.line+n}})}if(o.scriptSetup?.content){let t=e.t(o.scriptSetup.content,r,i),n=o.scriptSetup.loc.start.line-1;for(let e of t)a.push({...e,origin:{...e.origin,line:e.origin.line+n}})}if(o.script?.content){let t=e.t(o.script.content,r,i),n=o.script.loc.start.line-1;for(let e of t)a.push({...e,origin:{...e.origin,line:e.origin.line+n}})}return a}exports.extractFromVue=m;
|
|
3
3
|
//# sourceMappingURL=vue-extractor.cjs.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"vue-extractor.cjs","names":[],"sources":["../src/vue-extractor.ts"],"sourcesContent":["import type { ExtractedMessage } from '@fluenti/core'\nimport { parse as parseSFC } from '@vue/compiler-sfc'\nimport { createMessageId } from '@fluenti/core/internal'\nimport { extractFromTsx } from './tsx-extractor'\n\n// Vue template AST node types\nconst ELEMENT_NODE = 1\nconst TEXT_NODE = 2\nconst DIRECTIVE_PROP = 7\nconst ATTRIBUTE_PROP = 6\n\ninterface LocInfo {\n line: number\n column: number\n offset: number\n}\n\ninterface SourceLoc {\n start: LocInfo\n end: LocInfo\n source: string\n}\n\ninterface TemplateNode {\n type: number\n tag?: string\n tagType?: number\n props?: TemplateProp[]\n children?: TemplateNode[]\n content?: string\n loc: SourceLoc\n}\n\ninterface TemplateProp {\n type: number\n name: string | { content: string }\n rawName?: string\n arg?: { content: string; isStatic: boolean }\n exp?: { content: string }\n modifiers?: Array<{ content: string } | string>\n value?: { content: string }\n nameLoc?: SourceLoc\n loc: SourceLoc\n}\n\nfunction getTextContent(children: TemplateNode[]): string {\n return children\n .filter((c) => c.type === TEXT_NODE)\n .map((c) => (c.content ?? '').trim())\n .join('')\n}\n\nfunction buildPluralICUFromPipe(text: string, countVar: string): string {\n const forms = text.split('|').map((s) => s.trim())\n const categories = ['one', 'other', 'zero', 'few', 'many']\n const options: string[] = []\n\n if (forms.length === 2) {\n options.push(`one {${forms[0]}}`)\n options.push(`other {${forms[1]}}`)\n } else {\n for (let i = 0; i < forms.length && i < categories.length; i++) {\n options.push(`${categories[i]} {${forms[i]}}`)\n }\n }\n\n return `{${countVar}, plural, ${options.join(' ')}}`\n}\n\nfunction buildPluralICUFromProps(props: Record<string, string>): string {\n const countVar = props['count'] ?? 'count'\n const categories = ['zero', 'one', 'two', 'few', 'many', 'other']\n const options: string[] = []\n const offset = props['offset']\n\n for (const cat of categories) {\n if (props[cat] !== undefined) {\n const key = cat === 'zero' ? '=0' : cat\n options.push(`${key} {${props[cat]}}`)\n }\n }\n\n if (options.length === 0) return ''\n const offsetPrefix = offset ? `offset:${offset} ` : ''\n return `{${countVar}, plural, ${offsetPrefix}${options.join(' ')}}`\n}\n\nfunction walkTemplate(\n node: TemplateNode,\n filename: string,\n messages: ExtractedMessage[],\n): void {\n if (node.type === ELEMENT_NODE) {\n const vtDirective = node.props?.find(\n (p) => p.type === DIRECTIVE_PROP && getPropName(p) === 't',\n )\n\n if (vtDirective) {\n const RESERVED_MODIFIERS = new Set(['plural'])\n const modifiers = (vtDirective.modifiers ?? []).map(\n (m: string | { content: string }) => (typeof m === 'string' ? m : m.content),\n )\n const isPlural = modifiers.includes('plural')\n // Reconstruct dotted ID: v-t:checkout.title → arg=\"checkout\", modifier=\"title\" → \"checkout.title\"\n // Non-reserved modifiers are treated as ID path segments\n const idSegments = modifiers.filter((m: string) => !RESERVED_MODIFIERS.has(m))\n const argContent = vtDirective.arg?.content\n const explicitId = argContent\n ? [argContent, ...idSegments].join('.')\n : undefined\n const textContent = getTextContent(node.children ?? [])\n\n if (isPlural) {\n const countVar = vtDirective.exp?.content ?? 'count'\n const message = buildPluralICUFromPipe(textContent, countVar)\n const id = explicitId ?? createMessageId(message)\n messages.push({\n id,\n message,\n origin: {\n file: filename,\n line: vtDirective.loc.start.line,\n column: vtDirective.loc.start.column,\n },\n })\n } else if (textContent) {\n const id = explicitId ?? createMessageId(textContent)\n messages.push({\n id,\n message: textContent,\n origin: {\n file: filename,\n line: vtDirective.loc.start.line,\n column: vtDirective.loc.start.column,\n },\n })\n }\n }\n\n if (node.tag === 'Trans') {\n const messageProp = node.props?.find(\n (p) => p.type === ATTRIBUTE_PROP && getPropName(p) === 'message',\n )\n const idProp = node.props?.find(\n (p) => p.type === ATTRIBUTE_PROP && getPropName(p) === 'id',\n )\n const contextProp = node.props?.find(\n (p) => p.type === ATTRIBUTE_PROP && getPropName(p) === 'context',\n )\n const commentProp = node.props?.find(\n (p) => p.type === ATTRIBUTE_PROP && getPropName(p) === 'comment',\n )\n const context = contextProp?.value?.content\n const comment = commentProp?.value?.content\n\n if (messageProp?.value) {\n // Old API: <Trans message=\"...\" />\n const message = messageProp.value.content\n const id = idProp?.value?.content ?? createMessageId(message, context)\n messages.push({\n id,\n message,\n ...(context !== undefined ? { context } : {}),\n ...(comment !== undefined ? { comment } : {}),\n origin: {\n file: filename,\n line: node.loc.start.line,\n column: node.loc.start.column,\n },\n })\n } else if (node.children && node.children.length > 0) {\n // New API: <Trans>content with <a>rich text</a></Trans>\n const richText = extractRichTextFromTemplateChildren(node.children)\n if (richText.message) {\n const id = idProp?.value?.content ?? createMessageId(richText.message, context)\n messages.push({\n id,\n message: richText.message,\n ...(context !== undefined ? { context } : {}),\n ...(comment !== undefined ? { comment } : {}),\n origin: {\n file: filename,\n line: node.loc.start.line,\n column: node.loc.start.column,\n },\n })\n }\n }\n }\n\n if (node.tag === 'Plural') {\n const propsMap: Record<string, string> = {}\n let valueExpr: string | undefined\n let offsetExpr: string | undefined\n for (const prop of node.props ?? []) {\n if (prop.type === ATTRIBUTE_PROP && prop.value) {\n propsMap[getPropName(prop)] = prop.value.content\n }\n // Handle :value=\"expr\" binding (directive prop)\n if (prop.type === DIRECTIVE_PROP && getPropName(prop) === 'bind' && prop.arg?.content === 'value' && prop.exp) {\n valueExpr = prop.exp.content\n }\n if (prop.type === DIRECTIVE_PROP && getPropName(prop) === 'bind' && prop.arg?.content === 'offset' && prop.exp) {\n offsetExpr = prop.exp.content\n }\n }\n\n // Use :value binding expression as count variable, fall back to 'count' static prop\n const countVar = valueExpr ?? propsMap['count'] ?? 'count'\n const offset = offsetExpr ?? propsMap['offset']\n const pluralMessage = buildPluralICUFromProps({\n ...propsMap,\n count: countVar,\n ...(offset !== undefined ? { offset } : {}),\n })\n if (pluralMessage) {\n const id = propsMap['id'] ?? createMessageId(pluralMessage)\n messages.push({\n id,\n message: pluralMessage,\n origin: {\n file: filename,\n line: node.loc.start.line,\n column: node.loc.start.column,\n },\n })\n }\n }\n }\n\n if (node.children) {\n for (const child of node.children) {\n walkTemplate(child, filename, messages)\n }\n }\n}\n\nfunction extractRichTextFromTemplateChildren(\n children: TemplateNode[],\n): { message: string; hasElements: boolean } {\n let elementIndex = 0\n let hasElements = false\n\n const parts = children.map((child) => {\n if (child.type === TEXT_NODE) {\n return (child.content ?? '').trim() ? child.content ?? '' : ''\n }\n if (child.type === ELEMENT_NODE && child.tag) {\n hasElements = true\n const idx = elementIndex++\n const innerText = extractRichTextFromTemplateChildren(child.children ?? []).message\n return `<${idx}>${innerText}</${idx}>`\n }\n return ''\n })\n\n return {\n message: parts.join('').trim(),\n hasElements,\n }\n}\n\nfunction getPropName(prop: TemplateProp): string {\n if (typeof prop.name === 'string') return prop.name\n return prop.name.content\n}\n\nfunction extractTemplateInterpolations(\n content: string,\n filename: string,\n): ExtractedMessage[] {\n const messages: ExtractedMessage[] = []\n const interpolationRegex = /\\{\\{([\\s\\S]*?)\\}\\}/g\n let match: RegExpExecArray | null\n\n while ((match = interpolationRegex.exec(content)) !== null) {\n const expression = match[1]?.trim()\n if (!expression) continue\n\n const extracted = extractFromTsx(expression, filename)\n if (extracted.length === 0) continue\n\n const lineOffset = content.slice(0, match.index).split('\\n').length - 1\n for (const msg of extracted) {\n messages.push({\n ...msg,\n origin: {\n ...msg.origin,\n line: msg.origin.line + lineOffset,\n },\n })\n }\n }\n\n return messages\n}\n\n/** Extract messages from Vue SFC files */\nexport function extractFromVue(code: string, filename: string): ExtractedMessage[] {\n const messages: ExtractedMessage[] = []\n\n const { descriptor } = parseSFC(code, { filename })\n\n if (descriptor.template?.ast) {\n walkTemplate(descriptor.template.ast as unknown as TemplateNode, filename, messages)\n }\n\n // Also extract t() function calls from raw template source\n // (picks up t('source text') in template expressions like {{ t('...') }})\n if (descriptor.template?.content) {\n const templateMessages = extractFromTsx(descriptor.template.content, filename)\n const templateLoc = descriptor.template.loc\n const lineOffset = templateLoc.start.line - 1\n const existingIds = new Set(messages.map((m) => m.id))\n for (const msg of templateMessages) {\n if (!existingIds.has(msg.id)) {\n messages.push({\n ...msg,\n origin: {\n ...msg.origin,\n line: msg.origin.line + lineOffset,\n },\n })\n }\n }\n\n const interpolationMessages = extractTemplateInterpolations(descriptor.template.content, filename)\n for (const msg of interpolationMessages) {\n if (!existingIds.has(msg.id)) {\n messages.push({\n ...msg,\n origin: {\n ...msg.origin,\n line: msg.origin.line + lineOffset,\n },\n })\n }\n }\n }\n\n if (descriptor.scriptSetup?.content) {\n const scriptMessages = extractFromTsx(descriptor.scriptSetup.content, filename)\n const scriptLoc = descriptor.scriptSetup.loc\n const lineOffset = scriptLoc.start.line - 1\n for (const msg of scriptMessages) {\n messages.push({\n ...msg,\n origin: {\n ...msg.origin,\n line: msg.origin.line + lineOffset,\n },\n })\n }\n }\n\n if (descriptor.script?.content) {\n const scriptMessages = extractFromTsx(descriptor.script.content, filename)\n const scriptLoc = descriptor.script.loc\n const lineOffset = scriptLoc.start.line - 1\n for (const msg of scriptMessages) {\n messages.push({\n ...msg,\n origin: {\n ...msg.origin,\n line: msg.origin.line + lineOffset,\n },\n })\n }\n }\n\n return messages\n}\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"vue-extractor.cjs","names":[],"sources":["../src/vue-extractor.ts"],"sourcesContent":["import type { ExtractedMessage } from '@fluenti/core/internal'\nimport { parse as parseSFC } from '@vue/compiler-sfc'\nimport { createMessageId } from '@fluenti/core/transform'\nimport { extractFromTsx } from './tsx-extractor'\n\n// Vue template AST node types\nconst ELEMENT_NODE = 1\nconst TEXT_NODE = 2\nconst DIRECTIVE_PROP = 7\nconst ATTRIBUTE_PROP = 6\n\ninterface LocInfo {\n line: number\n column: number\n offset: number\n}\n\ninterface SourceLoc {\n start: LocInfo\n end: LocInfo\n source: string\n}\n\ninterface TemplateNode {\n type: number\n tag?: string\n tagType?: number\n props?: TemplateProp[]\n children?: TemplateNode[]\n content?: string\n loc: SourceLoc\n}\n\ninterface TemplateProp {\n type: number\n name: string | { content: string }\n rawName?: string\n arg?: { content: string; isStatic: boolean }\n exp?: { content: string }\n modifiers?: Array<{ content: string } | string>\n value?: { content: string }\n nameLoc?: SourceLoc\n loc: SourceLoc\n}\n\nfunction getTextContent(children: TemplateNode[]): string {\n return children\n .filter((c) => c.type === TEXT_NODE)\n .map((c) => (c.content ?? '').trim())\n .join('')\n}\n\nfunction buildPluralICUFromPipe(text: string, countVar: string): string {\n const forms = text.split('|').map((s) => s.trim())\n const categories = ['one', 'other', 'zero', 'few', 'many']\n const options: string[] = []\n\n if (forms.length === 2) {\n options.push(`one {${forms[0]}}`)\n options.push(`other {${forms[1]}}`)\n } else {\n for (let i = 0; i < forms.length && i < categories.length; i++) {\n options.push(`${categories[i]} {${forms[i]}}`)\n }\n }\n\n return `{${countVar}, plural, ${options.join(' ')}}`\n}\n\nfunction buildPluralICUFromProps(props: Record<string, string>): string {\n const countVar = props['count'] ?? 'count'\n const categories = ['zero', 'one', 'two', 'few', 'many', 'other']\n const options: string[] = []\n const offset = props['offset']\n\n for (const cat of categories) {\n if (props[cat] !== undefined) {\n const key = cat === 'zero' ? '=0' : cat\n options.push(`${key} {${props[cat]}}`)\n }\n }\n\n if (options.length === 0) return ''\n const offsetPrefix = offset ? `offset:${offset} ` : ''\n return `{${countVar}, plural, ${offsetPrefix}${options.join(' ')}}`\n}\n\nfunction walkTemplate(\n node: TemplateNode,\n filename: string,\n messages: ExtractedMessage[],\n idGenerator?: (message: string, context?: string) => string,\n): void {\n if (node.type === ELEMENT_NODE) {\n const vtDirective = node.props?.find(\n (p) => p.type === DIRECTIVE_PROP && getPropName(p) === 't',\n )\n\n if (vtDirective) {\n const RESERVED_MODIFIERS = new Set(['plural'])\n const modifiers = (vtDirective.modifiers ?? []).map(\n (m: string | { content: string }) => (typeof m === 'string' ? m : m.content),\n )\n const isPlural = modifiers.includes('plural')\n // Reconstruct dotted ID: v-t:checkout.title → arg=\"checkout\", modifier=\"title\" → \"checkout.title\"\n // Non-reserved modifiers are treated as ID path segments\n const idSegments = modifiers.filter((m: string) => !RESERVED_MODIFIERS.has(m))\n const argContent = vtDirective.arg?.content\n const explicitId = argContent\n ? [argContent, ...idSegments].join('.')\n : undefined\n const textContent = getTextContent(node.children ?? [])\n\n if (isPlural) {\n const countVar = vtDirective.exp?.content ?? 'count'\n const message = buildPluralICUFromPipe(textContent, countVar)\n const id = explicitId ?? (idGenerator ?? createMessageId)(message)\n messages.push({\n id,\n message,\n origin: {\n file: filename,\n line: vtDirective.loc.start.line,\n column: vtDirective.loc.start.column,\n },\n })\n } else if (textContent) {\n const id = explicitId ?? (idGenerator ?? createMessageId)(textContent)\n messages.push({\n id,\n message: textContent,\n origin: {\n file: filename,\n line: vtDirective.loc.start.line,\n column: vtDirective.loc.start.column,\n },\n })\n }\n }\n\n if (node.tag === 'Trans') {\n const messageProp = node.props?.find(\n (p) => p.type === ATTRIBUTE_PROP && getPropName(p) === 'message',\n )\n const idProp = node.props?.find(\n (p) => p.type === ATTRIBUTE_PROP && getPropName(p) === 'id',\n )\n const contextProp = node.props?.find(\n (p) => p.type === ATTRIBUTE_PROP && getPropName(p) === 'context',\n )\n const commentProp = node.props?.find(\n (p) => p.type === ATTRIBUTE_PROP && getPropName(p) === 'comment',\n )\n const context = contextProp?.value?.content\n const comment = commentProp?.value?.content\n\n if (messageProp?.value) {\n // Old API: <Trans message=\"...\" />\n const message = messageProp.value.content\n const generateId = idGenerator ?? createMessageId\n const id = idProp?.value?.content ?? generateId(message, context)\n messages.push({\n id,\n message,\n ...(context !== undefined ? { context } : {}),\n ...(comment !== undefined ? { comment } : {}),\n origin: {\n file: filename,\n line: node.loc.start.line,\n column: node.loc.start.column,\n },\n })\n } else if (node.children && node.children.length > 0) {\n // New API: <Trans>content with <a>rich text</a></Trans>\n const richText = extractRichTextFromTemplateChildren(node.children)\n if (richText.message) {\n const generateId = idGenerator ?? createMessageId\n const id = idProp?.value?.content ?? generateId(richText.message, context)\n messages.push({\n id,\n message: richText.message,\n ...(context !== undefined ? { context } : {}),\n ...(comment !== undefined ? { comment } : {}),\n origin: {\n file: filename,\n line: node.loc.start.line,\n column: node.loc.start.column,\n },\n })\n }\n }\n }\n\n if (node.tag === 'Plural') {\n const propsMap: Record<string, string> = {}\n let valueExpr: string | undefined\n let offsetExpr: string | undefined\n for (const prop of node.props ?? []) {\n if (prop.type === ATTRIBUTE_PROP && prop.value) {\n propsMap[getPropName(prop)] = prop.value.content\n }\n // Handle :value=\"expr\" binding (directive prop)\n if (prop.type === DIRECTIVE_PROP && getPropName(prop) === 'bind' && prop.arg?.content === 'value' && prop.exp) {\n valueExpr = prop.exp.content\n }\n if (prop.type === DIRECTIVE_PROP && getPropName(prop) === 'bind' && prop.arg?.content === 'offset' && prop.exp) {\n offsetExpr = prop.exp.content\n }\n }\n\n // Use :value binding expression as count variable, fall back to 'count' static prop\n const countVar = valueExpr ?? propsMap['count'] ?? 'count'\n const offset = offsetExpr ?? propsMap['offset']\n const pluralMessage = buildPluralICUFromProps({\n ...propsMap,\n count: countVar,\n ...(offset !== undefined ? { offset } : {}),\n })\n if (pluralMessage) {\n const id = propsMap['id'] ?? (idGenerator ?? createMessageId)(pluralMessage)\n messages.push({\n id,\n message: pluralMessage,\n origin: {\n file: filename,\n line: node.loc.start.line,\n column: node.loc.start.column,\n },\n })\n }\n }\n }\n\n if (node.children) {\n for (const child of node.children) {\n walkTemplate(child, filename, messages, idGenerator)\n }\n }\n}\n\nfunction extractRichTextFromTemplateChildren(\n children: TemplateNode[],\n): { message: string; hasElements: boolean } {\n let elementIndex = 0\n let hasElements = false\n\n const parts = children.map((child) => {\n if (child.type === TEXT_NODE) {\n return (child.content ?? '').trim() ? child.content ?? '' : ''\n }\n if (child.type === ELEMENT_NODE && child.tag) {\n hasElements = true\n const idx = elementIndex++\n const innerText = extractRichTextFromTemplateChildren(child.children ?? []).message\n return `<${idx}>${innerText}</${idx}>`\n }\n return ''\n })\n\n return {\n message: parts.join('').trim(),\n hasElements,\n }\n}\n\nfunction getPropName(prop: TemplateProp): string {\n if (typeof prop.name === 'string') return prop.name\n return prop.name.content\n}\n\nfunction extractTemplateInterpolations(\n content: string,\n filename: string,\n idGenerator?: (message: string, context?: string) => string,\n): ExtractedMessage[] {\n const messages: ExtractedMessage[] = []\n const interpolationRegex = /\\{\\{([\\s\\S]*?)\\}\\}/g\n let match: RegExpExecArray | null\n\n while ((match = interpolationRegex.exec(content)) !== null) {\n const expression = match[1]?.trim()\n if (!expression) continue\n\n const extracted = extractFromTsx(expression, filename, idGenerator)\n if (extracted.length === 0) continue\n\n const lineOffset = content.slice(0, match.index).split('\\n').length - 1\n for (const msg of extracted) {\n messages.push({\n ...msg,\n origin: {\n ...msg.origin,\n line: msg.origin.line + lineOffset,\n },\n })\n }\n }\n\n return messages\n}\n\n/** Extract messages from Vue SFC files */\nexport function extractFromVue(\n code: string,\n filename: string,\n idGenerator?: (message: string, context?: string) => string,\n): ExtractedMessage[] {\n const messages: ExtractedMessage[] = []\n\n const { descriptor } = parseSFC(code, { filename })\n\n if (descriptor.template?.ast) {\n walkTemplate(descriptor.template.ast as unknown as TemplateNode, filename, messages, idGenerator)\n }\n\n // Also extract t() function calls from raw template source\n // (picks up t('source text') in template expressions like {{ t('...') }})\n if (descriptor.template?.content) {\n const templateMessages = extractFromTsx(descriptor.template.content, filename, idGenerator)\n const templateLoc = descriptor.template.loc\n const lineOffset = templateLoc.start.line - 1\n const existingIds = new Set(messages.map((m) => m.id))\n for (const msg of templateMessages) {\n if (!existingIds.has(msg.id)) {\n messages.push({\n ...msg,\n origin: {\n ...msg.origin,\n line: msg.origin.line + lineOffset,\n },\n })\n }\n }\n\n const interpolationMessages = extractTemplateInterpolations(descriptor.template.content, filename, idGenerator)\n for (const msg of interpolationMessages) {\n if (!existingIds.has(msg.id)) {\n messages.push({\n ...msg,\n origin: {\n ...msg.origin,\n line: msg.origin.line + lineOffset,\n },\n })\n }\n }\n }\n\n if (descriptor.scriptSetup?.content) {\n const scriptMessages = extractFromTsx(descriptor.scriptSetup.content, filename, idGenerator)\n const scriptLoc = descriptor.scriptSetup.loc\n const lineOffset = scriptLoc.start.line - 1\n for (const msg of scriptMessages) {\n messages.push({\n ...msg,\n origin: {\n ...msg.origin,\n line: msg.origin.line + lineOffset,\n },\n })\n }\n }\n\n if (descriptor.script?.content) {\n const scriptMessages = extractFromTsx(descriptor.script.content, filename, idGenerator)\n const scriptLoc = descriptor.script.loc\n const lineOffset = scriptLoc.start.line - 1\n for (const msg of scriptMessages) {\n messages.push({\n ...msg,\n origin: {\n ...msg.origin,\n line: msg.origin.line + lineOffset,\n },\n })\n }\n }\n\n return messages\n}\n"],"mappings":"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"}
|
package/dist/vue-extractor.d.ts
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { ExtractedMessage } from '@fluenti/core';
|
|
1
|
+
import { ExtractedMessage } from '@fluenti/core/internal';
|
|
2
2
|
/** Extract messages from Vue SFC files */
|
|
3
|
-
export declare function extractFromVue(code: string, filename: string): ExtractedMessage[];
|
|
3
|
+
export declare function extractFromVue(code: string, filename: string, idGenerator?: (message: string, context?: string) => string): ExtractedMessage[];
|
|
4
4
|
//# sourceMappingURL=vue-extractor.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"vue-extractor.d.ts","sourceRoot":"","sources":["../src/vue-extractor.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,
|
|
1
|
+
{"version":3,"file":"vue-extractor.d.ts","sourceRoot":"","sources":["../src/vue-extractor.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,wBAAwB,CAAA;AA6S9D,0CAA0C;AAC1C,wBAAgB,cAAc,CAC5B,IAAI,EAAE,MAAM,EACZ,QAAQ,EAAE,MAAM,EAChB,WAAW,CAAC,EAAE,CAAC,OAAO,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE,MAAM,KAAK,MAAM,GAC1D,gBAAgB,EAAE,CAyEpB"}
|