@upstart.gg/vite-plugins 0.0.48 → 0.0.50
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.
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"vite-plugin-upstart-attrs.d.ts","names":[],"sources":["../src/vite-plugin-upstart-attrs.ts"],"sourcesContent":[],"mappings":";;;;
|
|
1
|
+
{"version":3,"file":"vite-plugin-upstart-attrs.d.ts","names":[],"sources":["../src/vite-plugin-upstart-attrs.ts"],"sourcesContent":[],"mappings":";;;;UAeU,OAAA;;;;AAAA,UA0CO,aAAA,CA1CA;EA0CA,IAAA,EAAA,MAAA;EAoBD,IAAA,EAAA,MAAA,GAAW,WAAmB;EAK9B,WAAA,EAAA,MAAa;EAehB,SAAA,EAAA,MAkEX;EAEc,eAAA,EAAA,MAAgB;EAgP/B,OAAA,EAAA;;;;iBAxUe,WAAA,CAAA,GAAe,eAAe;iBAK9B,aAAA,CAAA;cAeH,eAAa,SAAA,CAAA,iBAAA;iBAoEV,gBAAA;;OAAgB,aAAA,CAAA;;cAgP/B"}
|
|
@@ -168,7 +168,7 @@ function transformWithOxc(code, filePath) {
|
|
|
168
168
|
if (state$1.loopStack.length > 0) {
|
|
169
169
|
const loop = state$1.loopStack[state$1.loopStack.length - 1];
|
|
170
170
|
attributes.push(`data-upstart-loop-item="${escapeProp(loop.itemName)}"`);
|
|
171
|
-
if (loop.indexName) attributes.push(`data-upstart-loop-index={${loop.indexName
|
|
171
|
+
if (loop.indexName) attributes.push(`data-upstart-loop-index={${loop.indexName}}`);
|
|
172
172
|
attributes.push(`data-upstart-loop-array="${escapeProp(loop.arrayExpr)}"`);
|
|
173
173
|
}
|
|
174
174
|
for (const attr of opening.attributes) {
|
|
@@ -177,8 +177,6 @@ function transformWithOxc(code, filePath) {
|
|
|
177
177
|
const binding = analyzeBinding(attr.value, state$1.code);
|
|
178
178
|
if (binding) {
|
|
179
179
|
attributes.push(`data-upstart-prop-${propName.toLowerCase()}="${escapeProp(binding.path)}"`);
|
|
180
|
-
if (binding.datasource) attributes.push(`data-upstart-datasource-${propName.toLowerCase()}="${escapeProp(binding.datasource)}"`);
|
|
181
|
-
if (binding.recordId) attributes.push(`data-upstart-record-id-${propName.toLowerCase()}={${binding.recordId}}`);
|
|
182
180
|
if (binding.isConditional) attributes.push(`data-upstart-conditional-${propName.toLowerCase()}="true"`);
|
|
183
181
|
}
|
|
184
182
|
}
|
|
@@ -228,11 +226,7 @@ function analyzeBinding(value, code) {
|
|
|
228
226
|
return null;
|
|
229
227
|
}
|
|
230
228
|
function analyzeExpression(expr, code) {
|
|
231
|
-
if (expr.type === "MemberExpression") return {
|
|
232
|
-
path: exprToString(expr, code),
|
|
233
|
-
datasource: extractDatasource(expr),
|
|
234
|
-
recordId: extractRecordId(expr)
|
|
235
|
-
};
|
|
229
|
+
if (expr.type === "MemberExpression") return { path: exprToString(expr, code) };
|
|
236
230
|
if (expr.type === "Identifier") return { path: expr.name };
|
|
237
231
|
if (expr.type === "ConditionalExpression") {
|
|
238
232
|
const consequent = analyzeExpression(expr.consequent, code);
|
|
@@ -271,19 +265,6 @@ function exprToString(expr, code) {
|
|
|
271
265
|
}
|
|
272
266
|
return "";
|
|
273
267
|
}
|
|
274
|
-
function extractDatasource(expr) {
|
|
275
|
-
let current = expr.object;
|
|
276
|
-
while (current.type === "MemberExpression") current = current.object;
|
|
277
|
-
if (current.type === "Identifier") return current.name;
|
|
278
|
-
}
|
|
279
|
-
function extractRecordId(expr) {
|
|
280
|
-
const obj = expr.object;
|
|
281
|
-
if (obj.type === "Identifier") return `${obj.name}.$id`;
|
|
282
|
-
if (obj.type === "MemberExpression") {
|
|
283
|
-
const datasource = extractDatasource(obj);
|
|
284
|
-
if (datasource) return `${datasource}.$id`;
|
|
285
|
-
}
|
|
286
|
-
}
|
|
287
268
|
function detectMapCall(node, code) {
|
|
288
269
|
if (node.callee.type !== "MemberExpression" || node.callee.property.type !== "Identifier" || node.callee.property.name !== "map") return null;
|
|
289
270
|
const callback = node.arguments[0];
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"vite-plugin-upstart-attrs.js","names":["state: TransformState","state","attributes: string[]","current: Expression","keyResult: { value: string; expr: string | null } | null","nsResult: { value: string; expr: string | null } | null","results: I18nKeyInfo[]"],"sources":["../src/vite-plugin-upstart-attrs.ts"],"sourcesContent":["import { createUnplugin } from \"unplugin\";\nimport { parseSync } from \"oxc-parser\";\nimport { walk } from \"zimmerframe\";\nimport MagicString from \"magic-string\";\nimport path from \"path\";\nimport type {\n Program,\n Node,\n JSXElement,\n JSXOpeningElement,\n JSXAttribute,\n Expression,\n MemberExpression,\n CallExpression,\n} from \"estree-jsx\";\n\ninterface Options {\n enabled: boolean;\n emitRegistry?: boolean;\n}\n\ntype NodeWithRange = Node & { start: number; end: number };\n\nfunction hasRange(node: any): node is NodeWithRange {\n return node && typeof node.start === \"number\" && typeof node.end === \"number\";\n}\n\n// Fast, stable hash function (djb2 variant)\n// Produces a short hex string that's stable across rebuilds\nfunction hashContent(content: string): string {\n let hash = 5381;\n for (let i = 0; i < content.length; i++) {\n hash = ((hash << 5) + hash) ^ content.charCodeAt(i);\n }\n // Convert to unsigned 32-bit and then to hex\n return (hash >>> 0).toString(16);\n}\n\ninterface LoopContext {\n itemName: string;\n indexName: string | null;\n arrayExpr: string;\n}\n\ninterface I18nKeyInfo {\n /** The resolved key with namespace, e.g. \"dashboard:features.title\" */\n fullKey: string;\n /** Just the translation key, e.g. \"features.title\" */\n key: string;\n /** The namespace, e.g. \"dashboard\" */\n namespace: string;\n /** Source expression for i18nKey when dynamic (e.g. \"stat.labelKey\"), null when static */\n keyExpr: string | null;\n /** Source expression for namespace when dynamic, null when static */\n nsExpr: string | null;\n}\n\n// Registry entry for editable elements (text and className)\nexport interface EditableEntry {\n file: string;\n type: \"text\" | \"className\";\n startOffset: number;\n endOffset: number;\n originalContent: string;\n context: { parentTag: string };\n}\n\n// Module-level registry (collected across all files during build)\nconst editableRegistry = new Map<string, EditableEntry>();\n\n// Generate stable ID for editable elements\n// Replace the counter-based ID generation with position-based\nfunction generateId(filePath: string, node: NodeWithRange): string {\n // Use file path + start position for a stable, deterministic ID\n return `${filePath}:${node.start}`;\n}\n\n// Export for testing - get a copy of the current registry\nexport function getRegistry(): Record<string, EditableEntry> {\n return Object.fromEntries(editableRegistry);\n}\n\n// Export for testing - clear registry and counters\nexport function clearRegistry(): void {\n editableRegistry.clear();\n}\n\ninterface TransformState {\n filePath: string;\n code: string;\n s: MagicString;\n loopStack: LoopContext[];\n /** Tracks const varName = \"literal\" declarations for resolving dynamic i18nKey expressions */\n constants: Map<string, string>;\n /** Maps t-function variable names to their default namespace, e.g. { \"t\": \"dashboard\" } */\n tFunctions: Map<string, string>;\n}\n\nexport const upstartEditor = createUnplugin<Options>((options) => {\n if (!options.enabled) {\n return { name: \"upstart-editor-disabled\" };\n }\n\n const emitRegistry = options.emitRegistry ?? true;\n let root = process.cwd();\n\n return {\n name: \"upstart-editor\",\n enforce: \"pre\",\n\n vite: {\n configResolved(config) {\n root = config.root;\n },\n generateBundle() {\n if (!emitRegistry || editableRegistry.size === 0) {\n return;\n }\n\n const registry = {\n version: 1,\n generatedAt: new Date().toISOString(),\n elements: Object.fromEntries(editableRegistry),\n };\n\n this.emitFile({\n type: \"asset\",\n fileName: \"upstart-registry.json\",\n source: JSON.stringify(registry, null, 2),\n });\n\n // Clear for next build\n editableRegistry.clear();\n },\n },\n\n transformInclude(id) {\n return /\\.(tsx|jsx)$/.test(id) && !id.includes(\"node_modules\");\n },\n\n transform(code, id) {\n // Fast path: skip files without JSX\n if (!code.includes(\"<\")) {\n return null;\n }\n\n try {\n const relativePath = path.relative(root, id);\n const result = transformWithOxc(code, relativePath);\n\n if (!result) {\n return null;\n }\n\n return {\n code: result.code,\n map: result.map,\n };\n } catch (error) {\n console.error(`Error transforming ${id}:`, error);\n return null;\n }\n },\n };\n});\n\nexport function transformWithOxc(code: string, filePath: string) {\n // Parse with oxc (super fast!)\n const ast = parseSync(filePath, code, {\n sourceType: \"module\",\n });\n\n if (!ast.program) {\n return null;\n }\n\n const s = new MagicString(code);\n const state: TransformState = {\n filePath,\n code,\n s,\n loopStack: [],\n constants: new Map(),\n tFunctions: new Map(),\n };\n\n let modified = false;\n\n // Use zimmerframe to walk and transform the AST\n walk(ast.program as Program, state, {\n _(node: Node, { state, next }: { state: TransformState; next: () => void }) {\n // Track useTranslation() calls to resolve t() function namespaces\n if (node.type === \"VariableDeclaration\") {\n trackUseTranslation(node, state);\n\n // Track const varName = \"literal\" for resolving dynamic i18nKey expressions\n for (const decl of (node as any).declarations) {\n if (\n decl.type === \"VariableDeclarator\" &&\n decl.id?.type === \"Identifier\" &&\n decl.init?.type === \"Literal\" &&\n typeof decl.init.value === \"string\"\n ) {\n state.constants.set(decl.id.name, decl.init.value);\n }\n }\n }\n\n // Handle .map() calls to track loops (must be before JSXElement)\n if (node.type === \"CallExpression\") {\n const loopInfo = detectMapCall(node as CallExpression, state.code);\n\n if (loopInfo) {\n // Push loop context\n state.loopStack.push(loopInfo);\n\n // Continue walking into the callback\n next();\n\n // Pop loop context after traversing\n state.loopStack.pop();\n return;\n }\n }\n\n // Handle JSX elements\n if (node.type === \"JSXElement\") {\n const jsxNode = node as JSXElement;\n const opening = jsxNode.openingElement;\n const tagName = getJSXElementName(opening);\n\n // Skip ALL attribute injection for <Trans> elements.\n // <Trans> renders to a text node at runtime, so DOM attributes are lost.\n // The i18n attributes are promoted to the parent element instead.\n if (tagName === \"Trans\") {\n next();\n return;\n }\n\n const insertPos = getAttributeInsertPosition(opening, state.code);\n\n // Track attributes to inject\n const attributes: string[] = [];\n\n // Compute stable hash from the element's source code (includes all children)\n // This hash changes if ANY part of the element or its children change\n if (hasRange(jsxNode)) {\n const elementSource = state.code.slice(jsxNode.start, jsxNode.end);\n const hash = hashContent(elementSource);\n\n // Collect index variables from all enclosing loops so elements\n // inside .map() get a unique hash per iteration at runtime.\n const loopIndices = state.loopStack.map((l) => l.indexName).filter((n): n is string => n !== null);\n\n if (loopIndices.length > 0) {\n const suffix = loopIndices.map((n) => `\\${${n}}`).join(\"-\");\n attributes.push(`data-upstart-hash={\\`${hash}-${suffix}\\`}`);\n } else {\n attributes.push(`data-upstart-hash=\"${hash}\"`);\n }\n }\n\n // --- Editable text detection ---\n // Priority: i18n (true) > non-i18n text (false) > no text (no attribute)\n\n // Step 1: Check for Trans children AND t() calls (editable via i18n, gets \"true\")\n const transChildren = findTransInChildren(jsxNode, state.code, state.constants);\n const tCallChildren = findTCallsInChildren(jsxNode, state.code, state);\n const allI18nKeys = [...transChildren, ...tCallChildren];\n const hasI18n = allI18nKeys.length > 0;\n\n if (hasI18n) {\n attributes.push('data-upstart-editable-text=\"true\"');\n\n const hasDynamic = allI18nKeys.some((t) => t.keyExpr || t.nsExpr);\n if (hasDynamic) {\n // Build a runtime JSX template expression for dynamic i18n keys\n const parts = allI18nKeys.map((t) => {\n const nsPart = t.nsExpr ? `\\${${t.nsExpr}}` : t.namespace;\n const keyPart = t.keyExpr ? `\\${${t.keyExpr}}` : t.key;\n return `${nsPart}:${keyPart}`;\n });\n attributes.push(`data-upstart-i18n={\\`${parts.join(\",\")}\\`}`);\n } else {\n const keys = allI18nKeys.map((t: I18nKeyInfo) => escapeProp(t.fullKey)).join(\",\");\n attributes.push(`data-upstart-i18n=\"${keys}\"`);\n }\n }\n\n // Step 2: Text leaf elements — registry tracking + non-editable flag\n if (isTextLeafElement(jsxNode)) {\n if (!hasI18n) {\n attributes.push('data-upstart-editable-text=\"false\"');\n }\n\n // Find the actual JSXText node to track its location\n const textChild = jsxNode.children.find((c) => c.type === \"JSXText\" && (c as any).value?.trim());\n\n if (textChild && hasRange(textChild)) {\n const id = generateId(state.filePath, textChild);\n const textValue = (textChild as any).value as string;\n\n // Calculate trimmed offsets (exclude leading/trailing whitespace)\n const trimmedStart = textChild.start + (textValue.length - textValue.trimStart().length);\n const trimmedEnd = textChild.end - (textValue.length - textValue.trimEnd().length);\n\n editableRegistry.set(id, {\n file: state.filePath,\n type: \"text\",\n startOffset: trimmedStart,\n endOffset: trimmedEnd,\n originalContent: textValue.trim(),\n context: { parentTag: tagName || \"unknown\" },\n });\n\n attributes.push(`data-upstart-id=\"${id}\"`);\n }\n }\n // Step 3: Non-leaf elements with visible text content (expressions, mixed content)\n else if (!hasI18n && hasVisibleTextContent(jsxNode)) {\n attributes.push('data-upstart-editable-text=\"false\"');\n }\n\n // Track className attribute if it's a string literal\n const classNameAttr = opening.attributes.find(\n (attr): attr is JSXAttribute =>\n attr.type === \"JSXAttribute\" &&\n attr.name.type === \"JSXIdentifier\" &&\n attr.name.name === \"className\" &&\n attr.value?.type === \"Literal\" &&\n typeof (attr.value as any).value === \"string\",\n );\n\n if (classNameAttr && classNameAttr.value && hasRange(classNameAttr.value)) {\n const id = generateId(state.filePath, classNameAttr.value);\n const classValue = (classNameAttr.value as any).value as string;\n\n // +1 and -1 to exclude the quotes\n editableRegistry.set(id, {\n file: state.filePath,\n type: \"className\",\n startOffset: classNameAttr.value.start + 1,\n endOffset: classNameAttr.value.end - 1,\n originalContent: classValue,\n context: { parentTag: tagName || \"unknown\" },\n });\n\n attributes.push(`data-upstart-classname-id=\"${id}\"`);\n }\n\n // Process PascalCase components for additional tracking\n if (tagName && /^[A-Z]/.test(tagName)) {\n // File and component tracking\n attributes.push(`data-upstart-file=\"${escapeProp(state.filePath)}\"`);\n attributes.push(`data-upstart-component=\"${escapeProp(tagName)}\"`);\n\n // Loop context tracking\n if (state.loopStack.length > 0) {\n const loop = state.loopStack[state.loopStack.length - 1];\n attributes.push(`data-upstart-loop-item=\"${escapeProp(loop.itemName)}\"`);\n if (loop.indexName) {\n attributes.push(`data-upstart-loop-index={${loop.indexName.toLowerCase()}}`);\n }\n attributes.push(`data-upstart-loop-array=\"${escapeProp(loop.arrayExpr)}\"`);\n }\n\n // Analyze each prop for data bindings\n for (const attr of opening.attributes) {\n if (attr.type !== \"JSXAttribute\" || attr.name.type !== \"JSXIdentifier\") {\n continue;\n }\n\n const propName = attr.name.name;\n const binding = analyzeBinding(attr.value, state.code);\n\n if (binding) {\n attributes.push(`data-upstart-prop-${propName.toLowerCase()}=\"${escapeProp(binding.path)}\"`);\n\n if (binding.datasource) {\n attributes.push(\n `data-upstart-datasource-${propName.toLowerCase()}=\"${escapeProp(binding.datasource)}\"`,\n );\n }\n\n if (binding.recordId) {\n attributes.push(`data-upstart-record-id-${propName.toLowerCase()}={${binding.recordId}}`);\n }\n\n // Track conditional expressions\n if (binding.isConditional) {\n attributes.push(`data-upstart-conditional-${propName.toLowerCase()}=\"true\"`);\n }\n }\n }\n }\n\n // Inject attributes if any\n if (insertPos !== -1 && attributes.length > 0) {\n const attrString = \" \" + attributes.join(\" \");\n state.s.appendLeft(insertPos, attrString);\n modified = true;\n }\n }\n\n next();\n },\n });\n\n if (!modified) {\n return null;\n }\n\n return {\n code: s.toString(),\n map: s.generateMap({ hires: true }),\n };\n}\n\n// Helper: Get JSX element name\nfunction getJSXElementName(opening: JSXOpeningElement): string | null {\n if (opening.name.type === \"JSXIdentifier\") {\n return opening.name.name;\n }\n\n // Handle JSXMemberExpression like <Foo.Bar>\n if (opening.name.type === \"JSXMemberExpression\") {\n let current = opening.name;\n while (current.property) {\n if (current.property.type === \"JSXIdentifier\") {\n return current.property.name;\n }\n if (current.type === \"JSXMemberExpression\") {\n current = current.property as any;\n } else {\n break;\n }\n }\n }\n\n return null;\n}\n\n// Helper: Find where to insert attributes in JSX opening tag\nfunction getAttributeInsertPosition(opening: JSXOpeningElement, code: string): number {\n // If there are existing attributes, insert before the first one\n if (opening.attributes.length > 0) {\n const firstAttr = opening.attributes[0];\n if (hasRange(firstAttr)) {\n return firstAttr.start;\n }\n }\n\n // Otherwise, insert after the tag name\n if (opening.name.type === \"JSXIdentifier\" && hasRange(opening.name)) {\n return opening.name.end;\n }\n\n if (opening.name.type === \"JSXMemberExpression\" && hasRange(opening.name)) {\n return opening.name.end;\n }\n\n return -1;\n}\n\n// Helper: Analyze a prop value to extract data binding\nfunction analyzeBinding(\n value: JSXAttribute[\"value\"],\n code: string,\n): {\n path: string;\n datasource?: string;\n recordId?: string;\n isConditional?: boolean;\n} | null {\n if (!value) {\n return null;\n }\n\n if (value.type === \"JSXExpressionContainer\") {\n const expr = value.expression;\n\n if (expr.type === \"JSXEmptyExpression\") {\n return null;\n }\n\n return analyzeExpression(expr, code);\n }\n\n return null;\n}\n\n// Helper: Analyze any expression to extract binding info\nfunction analyzeExpression(\n expr: Expression,\n code: string,\n): {\n path: string;\n datasource?: string;\n recordId?: string;\n isConditional?: boolean;\n} | null {\n // Handle member expressions: user.name, product.price, etc.\n if (expr.type === \"MemberExpression\") {\n const path = exprToString(expr, code);\n const datasource = extractDatasource(expr);\n const recordId = extractRecordId(expr);\n\n return { path, datasource, recordId };\n }\n\n // Handle identifiers: userName, price, etc.\n if (expr.type === \"Identifier\") {\n return { path: expr.name };\n }\n\n // Handle conditional expressions: condition ? value1 : value2\n if (expr.type === \"ConditionalExpression\") {\n // Try to extract from the consequent\n const consequent = analyzeExpression(expr.consequent, code);\n if (consequent) {\n return {\n ...consequent,\n isConditional: true,\n path: exprToString(expr, code),\n };\n }\n }\n\n // Handle logical expressions: value1 && value2, value1 || value2\n if (expr.type === \"LogicalExpression\") {\n const right = analyzeExpression(expr.right, code);\n if (right) {\n return {\n ...right,\n isConditional: true,\n path: exprToString(expr, code),\n };\n }\n }\n\n // For other complex expressions, just return the path\n const exprWithRange = expr as any;\n if (exprWithRange.start !== undefined && exprWithRange.end !== undefined) {\n return {\n path: code.slice(exprWithRange.start, exprWithRange.end),\n };\n }\n\n return null;\n}\n\n// Helper: Convert expression AST to string\nfunction exprToString(expr: Expression, code: string): string {\n // Use the source range to get the actual code\n const exprWithRange = expr as any;\n if (exprWithRange.start !== undefined && exprWithRange.end !== undefined) {\n return code.slice(exprWithRange.start, exprWithRange.end);\n }\n\n // Fallback: reconstruct from AST\n if (expr.type === \"Identifier\") {\n return expr.name;\n }\n\n if (expr.type === \"MemberExpression\") {\n const obj = exprToString(expr.object as Expression, code);\n const prop =\n expr.property.type === \"Identifier\" && !expr.computed\n ? expr.property.name\n : exprToString(expr.property as Expression, code);\n return expr.computed ? `${obj}[${prop}]` : `${obj}.${prop}`;\n }\n\n if (expr.type === \"ConditionalExpression\") {\n const test = exprToString(expr.test, code);\n const consequent = exprToString(expr.consequent, code);\n const alternate = exprToString(expr.alternate, code);\n return `${test} ? ${consequent} : ${alternate}`;\n }\n\n if (expr.type === \"LogicalExpression\") {\n const left = exprToString(expr.left, code);\n const right = exprToString(expr.right, code);\n return `${left} ${expr.operator} ${right}`;\n }\n\n return \"\";\n}\n\n// Helper: Extract datasource name from member expression\nfunction extractDatasource(expr: MemberExpression): string | undefined {\n let current: Expression = expr.object as Expression;\n\n // Traverse to the root object\n while (current.type === \"MemberExpression\") {\n current = current.object as Expression;\n }\n\n if (current.type === \"Identifier\") {\n return current.name;\n }\n\n return undefined;\n}\n\n// Helper: Extract record ID reference\nfunction extractRecordId(expr: MemberExpression): string | undefined {\n const obj = expr.object;\n\n if (obj.type === \"Identifier\") {\n // Assume the ID field is `${object}.$id`\n return `${obj.name}.$id`;\n }\n\n if (obj.type === \"MemberExpression\") {\n // For nested objects like user.profile.id\n const datasource = extractDatasource(obj);\n if (datasource) {\n return `${datasource}.$id`;\n }\n }\n\n return undefined;\n}\n\n// Helper: Detect .map() calls and extract loop context\nfunction detectMapCall(node: CallExpression, code: string): LoopContext | null {\n // Check if this is a .map() call\n if (\n node.callee.type !== \"MemberExpression\" ||\n node.callee.property.type !== \"Identifier\" ||\n node.callee.property.name !== \"map\"\n ) {\n return null;\n }\n\n const callback = node.arguments[0];\n\n if (!callback) {\n return null;\n }\n\n // Check if callback is an arrow function or function expression\n if (callback.type !== \"ArrowFunctionExpression\" && callback.type !== \"FunctionExpression\") {\n return null;\n }\n\n const params = callback.params;\n const itemParam = params[0];\n const indexParam = params[1];\n\n if (!itemParam) {\n return null;\n }\n\n // Extract parameter names\n const itemName = itemParam.type === \"Identifier\" ? itemParam.name : \"item\";\n const indexName = indexParam?.type === \"Identifier\" ? indexParam.name : null;\n\n // Get the array expression\n const arrayExpr = exprToString(node.callee.object as Expression, code);\n\n return {\n itemName,\n indexName,\n arrayExpr,\n };\n}\n\n// Helper: Escape prop values for JSX attributes\nfunction escapeProp(value: string): string {\n return value\n .replace(/&/g, \"&\")\n .replace(/\"/g, \""\")\n .replace(/'/g, \"'\")\n .replace(/</g, \"<\")\n .replace(/>/g, \">\");\n}\n\n// Helper: Track useTranslation() calls to map t-function variable names to their namespace\nfunction trackUseTranslation(node: Node, state: TransformState): void {\n if (node.type !== \"VariableDeclaration\") return;\n\n const decl = node as any;\n for (const declarator of decl.declarations) {\n if (\n declarator.id?.type !== \"ObjectPattern\" ||\n !declarator.init ||\n declarator.init.type !== \"CallExpression\"\n ) {\n continue;\n }\n\n const callExpr = declarator.init;\n if (callExpr.callee?.type !== \"Identifier\" || callExpr.callee.name !== \"useTranslation\") {\n continue;\n }\n\n const namespace = extractUseTranslationNamespace(callExpr);\n\n for (const prop of declarator.id.properties) {\n if (prop.type !== \"Property\") continue;\n if (prop.key?.type !== \"Identifier\" || prop.key.name !== \"t\") continue;\n\n // Handle aliased destructuring: { t: translate } → maps \"translate\" to namespace\n const localName = prop.value?.type === \"Identifier\" ? prop.value.name : \"t\";\n state.tFunctions.set(localName, namespace);\n }\n }\n}\n\n// Helper: Extract namespace from useTranslation() call arguments\nfunction extractUseTranslationNamespace(callExpr: any): string {\n const firstArg = callExpr.arguments?.[0];\n if (!firstArg) return \"translation\";\n\n // String literal: useTranslation(\"dashboard\")\n if (firstArg.type === \"Literal\" && typeof firstArg.value === \"string\") {\n return firstArg.value;\n }\n\n // Array expression: useTranslation([\"dashboard\", \"common\"])\n if (firstArg.type === \"ArrayExpression\" && firstArg.elements?.length > 0) {\n const first = firstArg.elements[0];\n if (first?.type === \"Literal\" && typeof first.value === \"string\") {\n return first.value;\n }\n }\n\n return \"translation\";\n}\n\n// Helper: Resolve a JSX attribute value to a string.\n// Handles string literals directly, and JSXExpressionContainer by resolving\n// Identifier references via the constants map, or returning the expression source for dynamic values.\nfunction resolveJSXAttrValue(\n value: any,\n code: string,\n constants: Map<string, string>,\n): { value: string; expr: string | null } | null {\n if (!value) return null;\n\n if (value.type === \"Literal\" && typeof value.value === \"string\") {\n return { value: value.value, expr: null };\n }\n\n if (value.type === \"JSXExpressionContainer\") {\n const expression = value.expression;\n if (!expression) return null;\n\n // Resolve simple identifiers via constants map (e.g. i18nKey={labelKey} where const labelKey = \"...\")\n if (expression.type === \"Identifier\" && constants.has(expression.name)) {\n return { value: constants.get(expression.name)!, expr: null };\n }\n\n // Dynamic expression — return source text so caller can emit a runtime JSX expression\n if (hasRange(expression)) {\n const src = code.slice(expression.start, expression.end);\n return { value: src, expr: src };\n }\n }\n\n return null;\n}\n\n// Helper: Detect <Trans i18nKey=\"...\" /> component and extract i18n key info\nfunction detectTransComponent(\n jsxElement: JSXElement,\n code: string,\n constants: Map<string, string>,\n): I18nKeyInfo | null {\n const opening = jsxElement.openingElement;\n const tagName = getJSXElementName(opening);\n\n if (tagName !== \"Trans\") return null;\n\n let keyResult: { value: string; expr: string | null } | null = null;\n let nsResult: { value: string; expr: string | null } | null = null;\n\n for (const attr of opening.attributes) {\n if (attr.type !== \"JSXAttribute\" || attr.name.type !== \"JSXIdentifier\") continue;\n\n const attrName = attr.name.name;\n\n if (attrName === \"i18nKey\") {\n keyResult = resolveJSXAttrValue(attr.value, code, constants);\n }\n\n if (attrName === \"ns\") {\n nsResult = resolveJSXAttrValue(attr.value, code, constants);\n }\n }\n\n if (!keyResult) {\n return null;\n }\n\n const key = keyResult.value;\n const namespace = nsResult?.value ?? \"translation\";\n const keyExpr = keyResult.expr;\n const nsExpr = nsResult?.expr ?? null;\n\n return {\n fullKey: `${namespace}:${key}`,\n key,\n namespace,\n keyExpr,\n nsExpr,\n };\n}\n\n// Helper: Scan an element's direct children for <Trans> components and extract i18n key info.\n// Handles direct <Trans> children and <Trans> inside JSXExpressionContainer (e.g. {show && <Trans>}).\nfunction findTransInChildren(\n jsxElement: JSXElement,\n code: string,\n constants: Map<string, string>,\n): I18nKeyInfo[] {\n const results: I18nKeyInfo[] = [];\n\n for (const child of jsxElement.children) {\n if (child.type === \"JSXElement\") {\n const info = detectTransComponent(child as JSXElement, code, constants);\n if (info) results.push(info);\n }\n\n if (child.type === \"JSXExpressionContainer\") {\n const expr = (child as any).expression;\n if (expr && expr.type !== \"JSXEmptyExpression\") {\n findTransInExpression(expr, code, results, constants);\n }\n }\n }\n\n return results;\n}\n\n// Helper: Recursively search an expression tree for <Trans> JSXElements.\n// Handles LogicalExpression (&&, ||), ConditionalExpression (?:), CallExpression (.map()),\n// ArrowFunctionExpression/FunctionExpression (callbacks), BlockStatement, and direct JSXElement.\nfunction findTransInExpression(\n expr: any,\n code: string,\n results: I18nKeyInfo[],\n constants: Map<string, string>,\n): void {\n if (!expr || !expr.type) return;\n\n if (expr.type === \"JSXElement\") {\n const info = detectTransComponent(expr as JSXElement, code, constants);\n if (info) results.push(info);\n return;\n }\n\n if (expr.type === \"LogicalExpression\") {\n findTransInExpression(expr.left, code, results, constants);\n findTransInExpression(expr.right, code, results, constants);\n return;\n }\n\n if (expr.type === \"ConditionalExpression\") {\n findTransInExpression(expr.consequent, code, results, constants);\n findTransInExpression(expr.alternate, code, results, constants);\n return;\n }\n\n // Handle .map() and other call expressions — recurse into arguments\n if (expr.type === \"CallExpression\") {\n for (const arg of expr.arguments) {\n findTransInExpression(arg, code, results, constants);\n }\n return;\n }\n\n // Handle arrow/function expressions — recurse into body\n if (expr.type === \"ArrowFunctionExpression\" || expr.type === \"FunctionExpression\") {\n findTransInExpression(expr.body, code, results, constants);\n return;\n }\n\n // Handle block bodies (arrow functions with braces)\n if (expr.type === \"BlockStatement\") {\n for (const stmt of expr.body) {\n if (stmt.type === \"ReturnStatement\" && stmt.argument) {\n findTransInExpression(stmt.argument, code, results, constants);\n }\n }\n return;\n }\n}\n\n// Helper: Scan an element's direct children for t() function calls and extract i18n key info.\nfunction findTCallsInChildren(jsxElement: JSXElement, code: string, state: TransformState): I18nKeyInfo[] {\n if (state.tFunctions.size === 0) return [];\n\n const results: I18nKeyInfo[] = [];\n\n for (const child of jsxElement.children) {\n if (child.type === \"JSXExpressionContainer\") {\n const expr = (child as any).expression;\n if (expr && expr.type !== \"JSXEmptyExpression\") {\n findTCallInExpression(expr, code, results, state);\n }\n }\n }\n\n return results;\n}\n\n// Helper: Recursively search an expression tree for t() calls.\n// Handles LogicalExpression (&&, ||), ConditionalExpression (?:), CallExpression (.map()),\n// ArrowFunctionExpression/FunctionExpression (callbacks), BlockStatement, and direct CallExpression.\nfunction findTCallInExpression(expr: any, code: string, results: I18nKeyInfo[], state: TransformState): void {\n if (!expr || !expr.type) return;\n\n if (expr.type === \"CallExpression\") {\n const info = detectTCall(expr, code, state);\n if (info) {\n results.push(info);\n return;\n }\n // Not a t() call — recurse into arguments (for .map() callbacks etc.)\n for (const arg of expr.arguments) {\n findTCallInExpression(arg, code, results, state);\n }\n return;\n }\n\n if (expr.type === \"LogicalExpression\") {\n findTCallInExpression(expr.left, code, results, state);\n findTCallInExpression(expr.right, code, results, state);\n return;\n }\n\n if (expr.type === \"ConditionalExpression\") {\n findTCallInExpression(expr.consequent, code, results, state);\n findTCallInExpression(expr.alternate, code, results, state);\n return;\n }\n\n if (expr.type === \"ArrowFunctionExpression\" || expr.type === \"FunctionExpression\") {\n findTCallInExpression(expr.body, code, results, state);\n return;\n }\n\n if (expr.type === \"BlockStatement\") {\n for (const stmt of expr.body) {\n if (stmt.type === \"ReturnStatement\" && stmt.argument) {\n findTCallInExpression(stmt.argument, code, results, state);\n }\n }\n return;\n }\n}\n\n// Helper: Detect a t(\"key\") call and extract i18n key info.\nfunction detectTCall(callExpr: any, code: string, state: TransformState): I18nKeyInfo | null {\n if (callExpr.callee?.type !== \"Identifier\") return null;\n\n const calleeName = callExpr.callee.name;\n const namespace = state.tFunctions.get(calleeName);\n if (namespace === undefined) return null;\n\n const firstArg = callExpr.arguments?.[0];\n if (!firstArg) return null;\n\n // Static string key: t(\"features.title\")\n if (firstArg.type === \"Literal\" && typeof firstArg.value === \"string\") {\n const key = firstArg.value;\n return { fullKey: `${namespace}:${key}`, key, namespace, keyExpr: null, nsExpr: null };\n }\n\n // Dynamic key — try constant resolution first, then fall back to expression\n if (hasRange(firstArg)) {\n const exprSrc = code.slice(firstArg.start, firstArg.end);\n\n if (firstArg.type === \"Identifier\" && state.constants.has(firstArg.name)) {\n const key = state.constants.get(firstArg.name)!;\n return { fullKey: `${namespace}:${key}`, key, namespace, keyExpr: null, nsExpr: null };\n }\n\n return { fullKey: `${namespace}:${exprSrc}`, key: exprSrc, namespace, keyExpr: exprSrc, nsExpr: null };\n }\n\n return null;\n}\n\n// Helper: Check if element is a \"leaf\" with only static text (no nested elements or expressions)\nfunction isTextLeafElement(jsxElement: JSXElement): boolean {\n let hasText = false;\n\n for (const child of jsxElement.children) {\n if (child.type === \"JSXText\") {\n const textValue = (child as any).value;\n if (textValue?.trim()) {\n hasText = true;\n }\n } else if (\n child.type === \"JSXElement\" ||\n child.type === \"JSXFragment\" ||\n child.type === \"JSXExpressionContainer\" ||\n child.type === \"JSXSpreadChild\"\n ) {\n // Has non-text children, not a leaf element\n return false;\n }\n }\n\n return hasText;\n}\n\n// Helper: Check if element has any visible text content (static or dynamic).\n// Broader than isTextLeafElement — returns true even if the element has other child types.\n// Detects JSXText with non-whitespace content and expression containers with\n// text-producing expressions (Identifier, MemberExpression, Literal, TemplateLiteral).\n// Skips CallExpression, ConditionalExpression, LogicalExpression (these typically produce elements).\nfunction hasVisibleTextContent(jsxElement: JSXElement): boolean {\n for (const child of jsxElement.children) {\n if (child.type === \"JSXText\") {\n const textValue = (child as any).value;\n if (textValue?.trim()) {\n return true;\n }\n continue;\n }\n\n if (child.type === \"JSXExpressionContainer\") {\n const expr = (child as any).expression;\n if (!expr || expr.type === \"JSXEmptyExpression\") {\n continue;\n }\n if (\n expr.type === \"Identifier\" ||\n expr.type === \"MemberExpression\" ||\n expr.type === \"Literal\" ||\n expr.type === \"TemplateLiteral\"\n ) {\n return true;\n }\n }\n }\n\n return false;\n}\n\nexport default upstartEditor.vite;\n"],"mappings":";;;;;;;AAuBA,SAAS,SAAS,MAAkC;AAClD,QAAO,QAAQ,OAAO,KAAK,UAAU,YAAY,OAAO,KAAK,QAAQ;;AAKvE,SAAS,YAAY,SAAyB;CAC5C,IAAI,OAAO;AACX,MAAK,IAAI,IAAI,GAAG,IAAI,QAAQ,QAAQ,IAClC,SAAS,QAAQ,KAAK,OAAQ,QAAQ,WAAW,EAAE;AAGrD,SAAQ,SAAS,GAAG,SAAS,GAAG;;AAiClC,MAAM,mCAAmB,IAAI,KAA4B;AAIzD,SAAS,WAAW,UAAkB,MAA6B;AAEjE,QAAO,GAAG,SAAS,GAAG,KAAK;;AAI7B,SAAgB,cAA6C;AAC3D,QAAO,OAAO,YAAY,iBAAiB;;AAI7C,SAAgB,gBAAsB;AACpC,kBAAiB,OAAO;;AAc1B,MAAa,gBAAgB,gBAAyB,YAAY;AAChE,KAAI,CAAC,QAAQ,QACX,QAAO,EAAE,MAAM,2BAA2B;CAG5C,MAAM,eAAe,QAAQ,gBAAgB;CAC7C,IAAI,OAAO,QAAQ,KAAK;AAExB,QAAO;EACL,MAAM;EACN,SAAS;EAET,MAAM;GACJ,eAAe,QAAQ;AACrB,WAAO,OAAO;;GAEhB,iBAAiB;AACf,QAAI,CAAC,gBAAgB,iBAAiB,SAAS,EAC7C;IAGF,MAAM,WAAW;KACf,SAAS;KACT,8BAAa,IAAI,MAAM,EAAC,aAAa;KACrC,UAAU,OAAO,YAAY,iBAAiB;KAC/C;AAED,SAAK,SAAS;KACZ,MAAM;KACN,UAAU;KACV,QAAQ,KAAK,UAAU,UAAU,MAAM,EAAE;KAC1C,CAAC;AAGF,qBAAiB,OAAO;;GAE3B;EAED,iBAAiB,IAAI;AACnB,UAAO,eAAe,KAAK,GAAG,IAAI,CAAC,GAAG,SAAS,eAAe;;EAGhE,UAAU,MAAM,IAAI;AAElB,OAAI,CAAC,KAAK,SAAS,IAAI,CACrB,QAAO;AAGT,OAAI;IAEF,MAAM,SAAS,iBAAiB,MADX,KAAK,SAAS,MAAM,GAAG,CACO;AAEnD,QAAI,CAAC,OACH,QAAO;AAGT,WAAO;KACL,MAAM,OAAO;KACb,KAAK,OAAO;KACb;YACM,OAAO;AACd,YAAQ,MAAM,sBAAsB,GAAG,IAAI,MAAM;AACjD,WAAO;;;EAGZ;EACD;AAEF,SAAgB,iBAAiB,MAAc,UAAkB;CAE/D,MAAM,MAAM,UAAU,UAAU,MAAM,EACpC,YAAY,UACb,CAAC;AAEF,KAAI,CAAC,IAAI,QACP,QAAO;CAGT,MAAM,IAAI,IAAI,YAAY,KAAK;CAC/B,MAAMA,QAAwB;EAC5B;EACA;EACA;EACA,WAAW,EAAE;EACb,2BAAW,IAAI,KAAK;EACpB,4BAAY,IAAI,KAAK;EACtB;CAED,IAAI,WAAW;AAGf,MAAK,IAAI,SAAoB,OAAO,EAClC,EAAE,MAAY,EAAE,gBAAO,QAAqD;AAE1E,MAAI,KAAK,SAAS,uBAAuB;AACvC,uBAAoB,MAAMC,QAAM;AAGhC,QAAK,MAAM,QAAS,KAAa,aAC/B,KACE,KAAK,SAAS,wBACd,KAAK,IAAI,SAAS,gBAClB,KAAK,MAAM,SAAS,aACpB,OAAO,KAAK,KAAK,UAAU,SAE3B,SAAM,UAAU,IAAI,KAAK,GAAG,MAAM,KAAK,KAAK,MAAM;;AAMxD,MAAI,KAAK,SAAS,kBAAkB;GAClC,MAAM,WAAW,cAAc,MAAwBA,QAAM,KAAK;AAElE,OAAI,UAAU;AAEZ,YAAM,UAAU,KAAK,SAAS;AAG9B,UAAM;AAGN,YAAM,UAAU,KAAK;AACrB;;;AAKJ,MAAI,KAAK,SAAS,cAAc;GAC9B,MAAM,UAAU;GAChB,MAAM,UAAU,QAAQ;GACxB,MAAM,UAAU,kBAAkB,QAAQ;AAK1C,OAAI,YAAY,SAAS;AACvB,UAAM;AACN;;GAGF,MAAM,YAAY,2BAA2B,SAASA,QAAM,KAAK;GAGjE,MAAMC,aAAuB,EAAE;AAI/B,OAAI,SAAS,QAAQ,EAAE;IAErB,MAAM,OAAO,YADSD,QAAM,KAAK,MAAM,QAAQ,OAAO,QAAQ,IAAI,CAC3B;IAIvC,MAAM,cAAcA,QAAM,UAAU,KAAK,MAAM,EAAE,UAAU,CAAC,QAAQ,MAAmB,MAAM,KAAK;AAElG,QAAI,YAAY,SAAS,GAAG;KAC1B,MAAM,SAAS,YAAY,KAAK,MAAM,MAAM,EAAE,GAAG,CAAC,KAAK,IAAI;AAC3D,gBAAW,KAAK,wBAAwB,KAAK,GAAG,OAAO,KAAK;UAE5D,YAAW,KAAK,sBAAsB,KAAK,GAAG;;GAQlD,MAAM,gBAAgB,oBAAoB,SAASA,QAAM,MAAMA,QAAM,UAAU;GAC/E,MAAM,gBAAgB,qBAAqB,SAASA,QAAM,MAAMA,QAAM;GACtE,MAAM,cAAc,CAAC,GAAG,eAAe,GAAG,cAAc;GACxD,MAAM,UAAU,YAAY,SAAS;AAErC,OAAI,SAAS;AACX,eAAW,KAAK,sCAAoC;AAGpD,QADmB,YAAY,MAAM,MAAM,EAAE,WAAW,EAAE,OAAO,EACjD;KAEd,MAAM,QAAQ,YAAY,KAAK,MAAM;AAGnC,aAAO,GAFQ,EAAE,SAAS,MAAM,EAAE,OAAO,KAAK,EAAE,UAE/B,GADD,EAAE,UAAU,MAAM,EAAE,QAAQ,KAAK,EAAE;OAEnD;AACF,gBAAW,KAAK,wBAAwB,MAAM,KAAK,IAAI,CAAC,KAAK;WACxD;KACL,MAAM,OAAO,YAAY,KAAK,MAAmB,WAAW,EAAE,QAAQ,CAAC,CAAC,KAAK,IAAI;AACjF,gBAAW,KAAK,sBAAsB,KAAK,GAAG;;;AAKlD,OAAI,kBAAkB,QAAQ,EAAE;AAC9B,QAAI,CAAC,QACH,YAAW,KAAK,uCAAqC;IAIvD,MAAM,YAAY,QAAQ,SAAS,MAAM,MAAM,EAAE,SAAS,aAAc,EAAU,OAAO,MAAM,CAAC;AAEhG,QAAI,aAAa,SAAS,UAAU,EAAE;KACpC,MAAM,KAAK,WAAWA,QAAM,UAAU,UAAU;KAChD,MAAM,YAAa,UAAkB;KAGrC,MAAM,eAAe,UAAU,SAAS,UAAU,SAAS,UAAU,WAAW,CAAC;KACjF,MAAM,aAAa,UAAU,OAAO,UAAU,SAAS,UAAU,SAAS,CAAC;AAE3E,sBAAiB,IAAI,IAAI;MACvB,MAAMA,QAAM;MACZ,MAAM;MACN,aAAa;MACb,WAAW;MACX,iBAAiB,UAAU,MAAM;MACjC,SAAS,EAAE,WAAW,WAAW,WAAW;MAC7C,CAAC;AAEF,gBAAW,KAAK,oBAAoB,GAAG,GAAG;;cAIrC,CAAC,WAAW,sBAAsB,QAAQ,CACjD,YAAW,KAAK,uCAAqC;GAIvD,MAAM,gBAAgB,QAAQ,WAAW,MACtC,SACC,KAAK,SAAS,kBACd,KAAK,KAAK,SAAS,mBACnB,KAAK,KAAK,SAAS,eACnB,KAAK,OAAO,SAAS,aACrB,OAAQ,KAAK,MAAc,UAAU,SACxC;AAED,OAAI,iBAAiB,cAAc,SAAS,SAAS,cAAc,MAAM,EAAE;IACzE,MAAM,KAAK,WAAWA,QAAM,UAAU,cAAc,MAAM;IAC1D,MAAM,aAAc,cAAc,MAAc;AAGhD,qBAAiB,IAAI,IAAI;KACvB,MAAMA,QAAM;KACZ,MAAM;KACN,aAAa,cAAc,MAAM,QAAQ;KACzC,WAAW,cAAc,MAAM,MAAM;KACrC,iBAAiB;KACjB,SAAS,EAAE,WAAW,WAAW,WAAW;KAC7C,CAAC;AAEF,eAAW,KAAK,8BAA8B,GAAG,GAAG;;AAItD,OAAI,WAAW,SAAS,KAAK,QAAQ,EAAE;AAErC,eAAW,KAAK,sBAAsB,WAAWA,QAAM,SAAS,CAAC,GAAG;AACpE,eAAW,KAAK,2BAA2B,WAAW,QAAQ,CAAC,GAAG;AAGlE,QAAIA,QAAM,UAAU,SAAS,GAAG;KAC9B,MAAM,OAAOA,QAAM,UAAUA,QAAM,UAAU,SAAS;AACtD,gBAAW,KAAK,2BAA2B,WAAW,KAAK,SAAS,CAAC,GAAG;AACxE,SAAI,KAAK,UACP,YAAW,KAAK,4BAA4B,KAAK,UAAU,aAAa,CAAC,GAAG;AAE9E,gBAAW,KAAK,4BAA4B,WAAW,KAAK,UAAU,CAAC,GAAG;;AAI5E,SAAK,MAAM,QAAQ,QAAQ,YAAY;AACrC,SAAI,KAAK,SAAS,kBAAkB,KAAK,KAAK,SAAS,gBACrD;KAGF,MAAM,WAAW,KAAK,KAAK;KAC3B,MAAM,UAAU,eAAe,KAAK,OAAOA,QAAM,KAAK;AAEtD,SAAI,SAAS;AACX,iBAAW,KAAK,qBAAqB,SAAS,aAAa,CAAC,IAAI,WAAW,QAAQ,KAAK,CAAC,GAAG;AAE5F,UAAI,QAAQ,WACV,YAAW,KACT,2BAA2B,SAAS,aAAa,CAAC,IAAI,WAAW,QAAQ,WAAW,CAAC,GACtF;AAGH,UAAI,QAAQ,SACV,YAAW,KAAK,0BAA0B,SAAS,aAAa,CAAC,IAAI,QAAQ,SAAS,GAAG;AAI3F,UAAI,QAAQ,cACV,YAAW,KAAK,4BAA4B,SAAS,aAAa,CAAC,SAAS;;;;AAOpF,OAAI,cAAc,MAAM,WAAW,SAAS,GAAG;IAC7C,MAAM,aAAa,MAAM,WAAW,KAAK,IAAI;AAC7C,YAAM,EAAE,WAAW,WAAW,WAAW;AACzC,eAAW;;;AAIf,QAAM;IAET,CAAC;AAEF,KAAI,CAAC,SACH,QAAO;AAGT,QAAO;EACL,MAAM,EAAE,UAAU;EAClB,KAAK,EAAE,YAAY,EAAE,OAAO,MAAM,CAAC;EACpC;;AAIH,SAAS,kBAAkB,SAA2C;AACpE,KAAI,QAAQ,KAAK,SAAS,gBACxB,QAAO,QAAQ,KAAK;AAItB,KAAI,QAAQ,KAAK,SAAS,uBAAuB;EAC/C,IAAI,UAAU,QAAQ;AACtB,SAAO,QAAQ,UAAU;AACvB,OAAI,QAAQ,SAAS,SAAS,gBAC5B,QAAO,QAAQ,SAAS;AAE1B,OAAI,QAAQ,SAAS,sBACnB,WAAU,QAAQ;OAElB;;;AAKN,QAAO;;AAIT,SAAS,2BAA2B,SAA4B,MAAsB;AAEpF,KAAI,QAAQ,WAAW,SAAS,GAAG;EACjC,MAAM,YAAY,QAAQ,WAAW;AACrC,MAAI,SAAS,UAAU,CACrB,QAAO,UAAU;;AAKrB,KAAI,QAAQ,KAAK,SAAS,mBAAmB,SAAS,QAAQ,KAAK,CACjE,QAAO,QAAQ,KAAK;AAGtB,KAAI,QAAQ,KAAK,SAAS,yBAAyB,SAAS,QAAQ,KAAK,CACvE,QAAO,QAAQ,KAAK;AAGtB,QAAO;;AAIT,SAAS,eACP,OACA,MAMO;AACP,KAAI,CAAC,MACH,QAAO;AAGT,KAAI,MAAM,SAAS,0BAA0B;EAC3C,MAAM,OAAO,MAAM;AAEnB,MAAI,KAAK,SAAS,qBAChB,QAAO;AAGT,SAAO,kBAAkB,MAAM,KAAK;;AAGtC,QAAO;;AAIT,SAAS,kBACP,MACA,MAMO;AAEP,KAAI,KAAK,SAAS,mBAKhB,QAAO;EAAE,MAJI,aAAa,MAAM,KAAK;EAItB,YAHI,kBAAkB,KAAK;EAGf,UAFV,gBAAgB,KAAK;EAED;AAIvC,KAAI,KAAK,SAAS,aAChB,QAAO,EAAE,MAAM,KAAK,MAAM;AAI5B,KAAI,KAAK,SAAS,yBAAyB;EAEzC,MAAM,aAAa,kBAAkB,KAAK,YAAY,KAAK;AAC3D,MAAI,WACF,QAAO;GACL,GAAG;GACH,eAAe;GACf,MAAM,aAAa,MAAM,KAAK;GAC/B;;AAKL,KAAI,KAAK,SAAS,qBAAqB;EACrC,MAAM,QAAQ,kBAAkB,KAAK,OAAO,KAAK;AACjD,MAAI,MACF,QAAO;GACL,GAAG;GACH,eAAe;GACf,MAAM,aAAa,MAAM,KAAK;GAC/B;;CAKL,MAAM,gBAAgB;AACtB,KAAI,cAAc,UAAU,UAAa,cAAc,QAAQ,OAC7D,QAAO,EACL,MAAM,KAAK,MAAM,cAAc,OAAO,cAAc,IAAI,EACzD;AAGH,QAAO;;AAIT,SAAS,aAAa,MAAkB,MAAsB;CAE5D,MAAM,gBAAgB;AACtB,KAAI,cAAc,UAAU,UAAa,cAAc,QAAQ,OAC7D,QAAO,KAAK,MAAM,cAAc,OAAO,cAAc,IAAI;AAI3D,KAAI,KAAK,SAAS,aAChB,QAAO,KAAK;AAGd,KAAI,KAAK,SAAS,oBAAoB;EACpC,MAAM,MAAM,aAAa,KAAK,QAAsB,KAAK;EACzD,MAAM,OACJ,KAAK,SAAS,SAAS,gBAAgB,CAAC,KAAK,WACzC,KAAK,SAAS,OACd,aAAa,KAAK,UAAwB,KAAK;AACrD,SAAO,KAAK,WAAW,GAAG,IAAI,GAAG,KAAK,KAAK,GAAG,IAAI,GAAG;;AAGvD,KAAI,KAAK,SAAS,wBAIhB,QAAO,GAHM,aAAa,KAAK,MAAM,KAAK,CAG3B,KAFI,aAAa,KAAK,YAAY,KAAK,CAEvB,KADb,aAAa,KAAK,WAAW,KAAK;AAItD,KAAI,KAAK,SAAS,qBAAqB;EACrC,MAAM,OAAO,aAAa,KAAK,MAAM,KAAK;EAC1C,MAAM,QAAQ,aAAa,KAAK,OAAO,KAAK;AAC5C,SAAO,GAAG,KAAK,GAAG,KAAK,SAAS,GAAG;;AAGrC,QAAO;;AAIT,SAAS,kBAAkB,MAA4C;CACrE,IAAIE,UAAsB,KAAK;AAG/B,QAAO,QAAQ,SAAS,mBACtB,WAAU,QAAQ;AAGpB,KAAI,QAAQ,SAAS,aACnB,QAAO,QAAQ;;AAOnB,SAAS,gBAAgB,MAA4C;CACnE,MAAM,MAAM,KAAK;AAEjB,KAAI,IAAI,SAAS,aAEf,QAAO,GAAG,IAAI,KAAK;AAGrB,KAAI,IAAI,SAAS,oBAAoB;EAEnC,MAAM,aAAa,kBAAkB,IAAI;AACzC,MAAI,WACF,QAAO,GAAG,WAAW;;;AAQ3B,SAAS,cAAc,MAAsB,MAAkC;AAE7E,KACE,KAAK,OAAO,SAAS,sBACrB,KAAK,OAAO,SAAS,SAAS,gBAC9B,KAAK,OAAO,SAAS,SAAS,MAE9B,QAAO;CAGT,MAAM,WAAW,KAAK,UAAU;AAEhC,KAAI,CAAC,SACH,QAAO;AAIT,KAAI,SAAS,SAAS,6BAA6B,SAAS,SAAS,qBACnE,QAAO;CAGT,MAAM,SAAS,SAAS;CACxB,MAAM,YAAY,OAAO;CACzB,MAAM,aAAa,OAAO;AAE1B,KAAI,CAAC,UACH,QAAO;AAUT,QAAO;EACL,UAPe,UAAU,SAAS,eAAe,UAAU,OAAO;EAQlE,WAPgB,YAAY,SAAS,eAAe,WAAW,OAAO;EAQtE,WALgB,aAAa,KAAK,OAAO,QAAsB,KAAK;EAMrE;;AAIH,SAAS,WAAW,OAAuB;AACzC,QAAO,MACJ,QAAQ,MAAM,QAAQ,CACtB,QAAQ,MAAM,SAAS,CACvB,QAAQ,MAAM,SAAS,CACvB,QAAQ,MAAM,OAAO,CACrB,QAAQ,MAAM,OAAO;;AAI1B,SAAS,oBAAoB,MAAY,OAA6B;AACpE,KAAI,KAAK,SAAS,sBAAuB;CAEzC,MAAM,OAAO;AACb,MAAK,MAAM,cAAc,KAAK,cAAc;AAC1C,MACE,WAAW,IAAI,SAAS,mBACxB,CAAC,WAAW,QACZ,WAAW,KAAK,SAAS,iBAEzB;EAGF,MAAM,WAAW,WAAW;AAC5B,MAAI,SAAS,QAAQ,SAAS,gBAAgB,SAAS,OAAO,SAAS,iBACrE;EAGF,MAAM,YAAY,+BAA+B,SAAS;AAE1D,OAAK,MAAM,QAAQ,WAAW,GAAG,YAAY;AAC3C,OAAI,KAAK,SAAS,WAAY;AAC9B,OAAI,KAAK,KAAK,SAAS,gBAAgB,KAAK,IAAI,SAAS,IAAK;GAG9D,MAAM,YAAY,KAAK,OAAO,SAAS,eAAe,KAAK,MAAM,OAAO;AACxE,SAAM,WAAW,IAAI,WAAW,UAAU;;;;AAMhD,SAAS,+BAA+B,UAAuB;CAC7D,MAAM,WAAW,SAAS,YAAY;AACtC,KAAI,CAAC,SAAU,QAAO;AAGtB,KAAI,SAAS,SAAS,aAAa,OAAO,SAAS,UAAU,SAC3D,QAAO,SAAS;AAIlB,KAAI,SAAS,SAAS,qBAAqB,SAAS,UAAU,SAAS,GAAG;EACxE,MAAM,QAAQ,SAAS,SAAS;AAChC,MAAI,OAAO,SAAS,aAAa,OAAO,MAAM,UAAU,SACtD,QAAO,MAAM;;AAIjB,QAAO;;AAMT,SAAS,oBACP,OACA,MACA,WAC+C;AAC/C,KAAI,CAAC,MAAO,QAAO;AAEnB,KAAI,MAAM,SAAS,aAAa,OAAO,MAAM,UAAU,SACrD,QAAO;EAAE,OAAO,MAAM;EAAO,MAAM;EAAM;AAG3C,KAAI,MAAM,SAAS,0BAA0B;EAC3C,MAAM,aAAa,MAAM;AACzB,MAAI,CAAC,WAAY,QAAO;AAGxB,MAAI,WAAW,SAAS,gBAAgB,UAAU,IAAI,WAAW,KAAK,CACpE,QAAO;GAAE,OAAO,UAAU,IAAI,WAAW,KAAK;GAAG,MAAM;GAAM;AAI/D,MAAI,SAAS,WAAW,EAAE;GACxB,MAAM,MAAM,KAAK,MAAM,WAAW,OAAO,WAAW,IAAI;AACxD,UAAO;IAAE,OAAO;IAAK,MAAM;IAAK;;;AAIpC,QAAO;;AAIT,SAAS,qBACP,YACA,MACA,WACoB;CACpB,MAAM,UAAU,WAAW;AAG3B,KAFgB,kBAAkB,QAAQ,KAE1B,QAAS,QAAO;CAEhC,IAAIC,YAA2D;CAC/D,IAAIC,WAA0D;AAE9D,MAAK,MAAM,QAAQ,QAAQ,YAAY;AACrC,MAAI,KAAK,SAAS,kBAAkB,KAAK,KAAK,SAAS,gBAAiB;EAExE,MAAM,WAAW,KAAK,KAAK;AAE3B,MAAI,aAAa,UACf,aAAY,oBAAoB,KAAK,OAAO,MAAM,UAAU;AAG9D,MAAI,aAAa,KACf,YAAW,oBAAoB,KAAK,OAAO,MAAM,UAAU;;AAI/D,KAAI,CAAC,UACH,QAAO;CAGT,MAAM,MAAM,UAAU;CACtB,MAAM,YAAY,UAAU,SAAS;CACrC,MAAM,UAAU,UAAU;CAC1B,MAAM,SAAS,UAAU,QAAQ;AAEjC,QAAO;EACL,SAAS,GAAG,UAAU,GAAG;EACzB;EACA;EACA;EACA;EACD;;AAKH,SAAS,oBACP,YACA,MACA,WACe;CACf,MAAMC,UAAyB,EAAE;AAEjC,MAAK,MAAM,SAAS,WAAW,UAAU;AACvC,MAAI,MAAM,SAAS,cAAc;GAC/B,MAAM,OAAO,qBAAqB,OAAqB,MAAM,UAAU;AACvE,OAAI,KAAM,SAAQ,KAAK,KAAK;;AAG9B,MAAI,MAAM,SAAS,0BAA0B;GAC3C,MAAM,OAAQ,MAAc;AAC5B,OAAI,QAAQ,KAAK,SAAS,qBACxB,uBAAsB,MAAM,MAAM,SAAS,UAAU;;;AAK3D,QAAO;;AAMT,SAAS,sBACP,MACA,MACA,SACA,WACM;AACN,KAAI,CAAC,QAAQ,CAAC,KAAK,KAAM;AAEzB,KAAI,KAAK,SAAS,cAAc;EAC9B,MAAM,OAAO,qBAAqB,MAAoB,MAAM,UAAU;AACtE,MAAI,KAAM,SAAQ,KAAK,KAAK;AAC5B;;AAGF,KAAI,KAAK,SAAS,qBAAqB;AACrC,wBAAsB,KAAK,MAAM,MAAM,SAAS,UAAU;AAC1D,wBAAsB,KAAK,OAAO,MAAM,SAAS,UAAU;AAC3D;;AAGF,KAAI,KAAK,SAAS,yBAAyB;AACzC,wBAAsB,KAAK,YAAY,MAAM,SAAS,UAAU;AAChE,wBAAsB,KAAK,WAAW,MAAM,SAAS,UAAU;AAC/D;;AAIF,KAAI,KAAK,SAAS,kBAAkB;AAClC,OAAK,MAAM,OAAO,KAAK,UACrB,uBAAsB,KAAK,MAAM,SAAS,UAAU;AAEtD;;AAIF,KAAI,KAAK,SAAS,6BAA6B,KAAK,SAAS,sBAAsB;AACjF,wBAAsB,KAAK,MAAM,MAAM,SAAS,UAAU;AAC1D;;AAIF,KAAI,KAAK,SAAS,kBAAkB;AAClC,OAAK,MAAM,QAAQ,KAAK,KACtB,KAAI,KAAK,SAAS,qBAAqB,KAAK,SAC1C,uBAAsB,KAAK,UAAU,MAAM,SAAS,UAAU;AAGlE;;;AAKJ,SAAS,qBAAqB,YAAwB,MAAc,OAAsC;AACxG,KAAI,MAAM,WAAW,SAAS,EAAG,QAAO,EAAE;CAE1C,MAAMA,UAAyB,EAAE;AAEjC,MAAK,MAAM,SAAS,WAAW,SAC7B,KAAI,MAAM,SAAS,0BAA0B;EAC3C,MAAM,OAAQ,MAAc;AAC5B,MAAI,QAAQ,KAAK,SAAS,qBACxB,uBAAsB,MAAM,MAAM,SAAS,MAAM;;AAKvD,QAAO;;AAMT,SAAS,sBAAsB,MAAW,MAAc,SAAwB,OAA6B;AAC3G,KAAI,CAAC,QAAQ,CAAC,KAAK,KAAM;AAEzB,KAAI,KAAK,SAAS,kBAAkB;EAClC,MAAM,OAAO,YAAY,MAAM,MAAM,MAAM;AAC3C,MAAI,MAAM;AACR,WAAQ,KAAK,KAAK;AAClB;;AAGF,OAAK,MAAM,OAAO,KAAK,UACrB,uBAAsB,KAAK,MAAM,SAAS,MAAM;AAElD;;AAGF,KAAI,KAAK,SAAS,qBAAqB;AACrC,wBAAsB,KAAK,MAAM,MAAM,SAAS,MAAM;AACtD,wBAAsB,KAAK,OAAO,MAAM,SAAS,MAAM;AACvD;;AAGF,KAAI,KAAK,SAAS,yBAAyB;AACzC,wBAAsB,KAAK,YAAY,MAAM,SAAS,MAAM;AAC5D,wBAAsB,KAAK,WAAW,MAAM,SAAS,MAAM;AAC3D;;AAGF,KAAI,KAAK,SAAS,6BAA6B,KAAK,SAAS,sBAAsB;AACjF,wBAAsB,KAAK,MAAM,MAAM,SAAS,MAAM;AACtD;;AAGF,KAAI,KAAK,SAAS,kBAAkB;AAClC,OAAK,MAAM,QAAQ,KAAK,KACtB,KAAI,KAAK,SAAS,qBAAqB,KAAK,SAC1C,uBAAsB,KAAK,UAAU,MAAM,SAAS,MAAM;AAG9D;;;AAKJ,SAAS,YAAY,UAAe,MAAc,OAA2C;AAC3F,KAAI,SAAS,QAAQ,SAAS,aAAc,QAAO;CAEnD,MAAM,aAAa,SAAS,OAAO;CACnC,MAAM,YAAY,MAAM,WAAW,IAAI,WAAW;AAClD,KAAI,cAAc,OAAW,QAAO;CAEpC,MAAM,WAAW,SAAS,YAAY;AACtC,KAAI,CAAC,SAAU,QAAO;AAGtB,KAAI,SAAS,SAAS,aAAa,OAAO,SAAS,UAAU,UAAU;EACrE,MAAM,MAAM,SAAS;AACrB,SAAO;GAAE,SAAS,GAAG,UAAU,GAAG;GAAO;GAAK;GAAW,SAAS;GAAM,QAAQ;GAAM;;AAIxF,KAAI,SAAS,SAAS,EAAE;EACtB,MAAM,UAAU,KAAK,MAAM,SAAS,OAAO,SAAS,IAAI;AAExD,MAAI,SAAS,SAAS,gBAAgB,MAAM,UAAU,IAAI,SAAS,KAAK,EAAE;GACxE,MAAM,MAAM,MAAM,UAAU,IAAI,SAAS,KAAK;AAC9C,UAAO;IAAE,SAAS,GAAG,UAAU,GAAG;IAAO;IAAK;IAAW,SAAS;IAAM,QAAQ;IAAM;;AAGxF,SAAO;GAAE,SAAS,GAAG,UAAU,GAAG;GAAW,KAAK;GAAS;GAAW,SAAS;GAAS,QAAQ;GAAM;;AAGxG,QAAO;;AAIT,SAAS,kBAAkB,YAAiC;CAC1D,IAAI,UAAU;AAEd,MAAK,MAAM,SAAS,WAAW,SAC7B,KAAI,MAAM,SAAS,WAEjB;MADmB,MAAc,OAClB,MAAM,CACnB,WAAU;YAGZ,MAAM,SAAS,gBACf,MAAM,SAAS,iBACf,MAAM,SAAS,4BACf,MAAM,SAAS,iBAGf,QAAO;AAIX,QAAO;;AAQT,SAAS,sBAAsB,YAAiC;AAC9D,MAAK,MAAM,SAAS,WAAW,UAAU;AACvC,MAAI,MAAM,SAAS,WAAW;AAE5B,OADmB,MAAc,OAClB,MAAM,CACnB,QAAO;AAET;;AAGF,MAAI,MAAM,SAAS,0BAA0B;GAC3C,MAAM,OAAQ,MAAc;AAC5B,OAAI,CAAC,QAAQ,KAAK,SAAS,qBACzB;AAEF,OACE,KAAK,SAAS,gBACd,KAAK,SAAS,sBACd,KAAK,SAAS,aACd,KAAK,SAAS,kBAEd,QAAO;;;AAKb,QAAO;;AAGT,wCAAe,cAAc"}
|
|
1
|
+
{"version":3,"file":"vite-plugin-upstart-attrs.js","names":["state: TransformState","state","attributes: string[]","keyResult: { value: string; expr: string | null } | null","nsResult: { value: string; expr: string | null } | null","results: I18nKeyInfo[]"],"sources":["../src/vite-plugin-upstart-attrs.ts"],"sourcesContent":["import { createUnplugin } from \"unplugin\";\nimport { parseSync } from \"oxc-parser\";\nimport { walk } from \"zimmerframe\";\nimport MagicString from \"magic-string\";\nimport path from \"path\";\nimport type {\n Program,\n Node,\n JSXElement,\n JSXOpeningElement,\n JSXAttribute,\n Expression,\n CallExpression,\n} from \"estree-jsx\";\n\ninterface Options {\n enabled: boolean;\n emitRegistry?: boolean;\n}\n\ntype NodeWithRange = Node & { start: number; end: number };\n\nfunction hasRange(node: any): node is NodeWithRange {\n return node && typeof node.start === \"number\" && typeof node.end === \"number\";\n}\n\n// Fast, stable hash function (djb2 variant)\n// Produces a short hex string that's stable across rebuilds\nfunction hashContent(content: string): string {\n let hash = 5381;\n for (let i = 0; i < content.length; i++) {\n hash = ((hash << 5) + hash) ^ content.charCodeAt(i);\n }\n // Convert to unsigned 32-bit and then to hex\n return (hash >>> 0).toString(16);\n}\n\ninterface LoopContext {\n itemName: string;\n indexName: string | null;\n arrayExpr: string;\n}\n\ninterface I18nKeyInfo {\n /** The resolved key with namespace, e.g. \"dashboard:features.title\" */\n fullKey: string;\n /** Just the translation key, e.g. \"features.title\" */\n key: string;\n /** The namespace, e.g. \"dashboard\" */\n namespace: string;\n /** Source expression for i18nKey when dynamic (e.g. \"stat.labelKey\"), null when static */\n keyExpr: string | null;\n /** Source expression for namespace when dynamic, null when static */\n nsExpr: string | null;\n}\n\n// Registry entry for editable elements (text and className)\nexport interface EditableEntry {\n file: string;\n type: \"text\" | \"className\";\n startOffset: number;\n endOffset: number;\n originalContent: string;\n context: { parentTag: string };\n}\n\n// Module-level registry (collected across all files during build)\nconst editableRegistry = new Map<string, EditableEntry>();\n\n// Generate stable ID for editable elements\n// Replace the counter-based ID generation with position-based\nfunction generateId(filePath: string, node: NodeWithRange): string {\n // Use file path + start position for a stable, deterministic ID\n return `${filePath}:${node.start}`;\n}\n\n// Export for testing - get a copy of the current registry\nexport function getRegistry(): Record<string, EditableEntry> {\n return Object.fromEntries(editableRegistry);\n}\n\n// Export for testing - clear registry and counters\nexport function clearRegistry(): void {\n editableRegistry.clear();\n}\n\ninterface TransformState {\n filePath: string;\n code: string;\n s: MagicString;\n loopStack: LoopContext[];\n /** Tracks const varName = \"literal\" declarations for resolving dynamic i18nKey expressions */\n constants: Map<string, string>;\n /** Maps t-function variable names to their default namespace, e.g. { \"t\": \"dashboard\" } */\n tFunctions: Map<string, string>;\n}\n\nexport const upstartEditor = createUnplugin<Options>((options) => {\n if (!options.enabled) {\n return { name: \"upstart-editor-disabled\" };\n }\n\n const emitRegistry = options.emitRegistry ?? true;\n let root = process.cwd();\n\n return {\n name: \"upstart-editor\",\n enforce: \"pre\",\n\n vite: {\n configResolved(config) {\n root = config.root;\n },\n generateBundle() {\n if (!emitRegistry || editableRegistry.size === 0) {\n return;\n }\n\n const registry = {\n version: 1,\n generatedAt: new Date().toISOString(),\n elements: Object.fromEntries(editableRegistry),\n };\n\n this.emitFile({\n type: \"asset\",\n fileName: \"upstart-registry.json\",\n source: JSON.stringify(registry, null, 2),\n });\n\n // Clear for next build\n editableRegistry.clear();\n },\n },\n\n transformInclude(id) {\n return /\\.(tsx|jsx)$/.test(id) && !id.includes(\"node_modules\");\n },\n\n transform(code, id) {\n // Fast path: skip files without JSX\n if (!code.includes(\"<\")) {\n return null;\n }\n\n try {\n const relativePath = path.relative(root, id);\n const result = transformWithOxc(code, relativePath);\n\n if (!result) {\n return null;\n }\n\n return {\n code: result.code,\n map: result.map,\n };\n } catch (error) {\n console.error(`Error transforming ${id}:`, error);\n return null;\n }\n },\n };\n});\n\nexport function transformWithOxc(code: string, filePath: string) {\n // Parse with oxc (super fast!)\n const ast = parseSync(filePath, code, {\n sourceType: \"module\",\n });\n\n if (!ast.program) {\n return null;\n }\n\n const s = new MagicString(code);\n const state: TransformState = {\n filePath,\n code,\n s,\n loopStack: [],\n constants: new Map(),\n tFunctions: new Map(),\n };\n\n let modified = false;\n\n // Use zimmerframe to walk and transform the AST\n walk(ast.program as Program, state, {\n _(node: Node, { state, next }: { state: TransformState; next: () => void }) {\n // Track useTranslation() calls to resolve t() function namespaces\n if (node.type === \"VariableDeclaration\") {\n trackUseTranslation(node, state);\n\n // Track const varName = \"literal\" for resolving dynamic i18nKey expressions\n for (const decl of (node as any).declarations) {\n if (\n decl.type === \"VariableDeclarator\" &&\n decl.id?.type === \"Identifier\" &&\n decl.init?.type === \"Literal\" &&\n typeof decl.init.value === \"string\"\n ) {\n state.constants.set(decl.id.name, decl.init.value);\n }\n }\n }\n\n // Handle .map() calls to track loops (must be before JSXElement)\n if (node.type === \"CallExpression\") {\n const loopInfo = detectMapCall(node as CallExpression, state.code);\n\n if (loopInfo) {\n // Push loop context\n state.loopStack.push(loopInfo);\n\n // Continue walking into the callback\n next();\n\n // Pop loop context after traversing\n state.loopStack.pop();\n return;\n }\n }\n\n // Handle JSX elements\n if (node.type === \"JSXElement\") {\n const jsxNode = node as JSXElement;\n const opening = jsxNode.openingElement;\n const tagName = getJSXElementName(opening);\n\n // Skip ALL attribute injection for <Trans> elements.\n // <Trans> renders to a text node at runtime, so DOM attributes are lost.\n // The i18n attributes are promoted to the parent element instead.\n if (tagName === \"Trans\") {\n next();\n return;\n }\n\n const insertPos = getAttributeInsertPosition(opening, state.code);\n\n // Track attributes to inject\n const attributes: string[] = [];\n\n // Compute stable hash from the element's source code (includes all children)\n // This hash changes if ANY part of the element or its children change\n if (hasRange(jsxNode)) {\n const elementSource = state.code.slice(jsxNode.start, jsxNode.end);\n const hash = hashContent(elementSource);\n\n // Collect index variables from all enclosing loops so elements\n // inside .map() get a unique hash per iteration at runtime.\n const loopIndices = state.loopStack.map((l) => l.indexName).filter((n): n is string => n !== null);\n\n if (loopIndices.length > 0) {\n const suffix = loopIndices.map((n) => `\\${${n}}`).join(\"-\");\n attributes.push(`data-upstart-hash={\\`${hash}-${suffix}\\`}`);\n } else {\n attributes.push(`data-upstart-hash=\"${hash}\"`);\n }\n }\n\n // --- Editable text detection ---\n // Priority: i18n (true) > non-i18n text (false) > no text (no attribute)\n\n // Step 1: Check for Trans children AND t() calls (editable via i18n, gets \"true\")\n const transChildren = findTransInChildren(jsxNode, state.code, state.constants);\n const tCallChildren = findTCallsInChildren(jsxNode, state.code, state);\n const allI18nKeys = [...transChildren, ...tCallChildren];\n const hasI18n = allI18nKeys.length > 0;\n\n if (hasI18n) {\n attributes.push('data-upstart-editable-text=\"true\"');\n\n const hasDynamic = allI18nKeys.some((t) => t.keyExpr || t.nsExpr);\n if (hasDynamic) {\n // Build a runtime JSX template expression for dynamic i18n keys\n const parts = allI18nKeys.map((t) => {\n const nsPart = t.nsExpr ? `\\${${t.nsExpr}}` : t.namespace;\n const keyPart = t.keyExpr ? `\\${${t.keyExpr}}` : t.key;\n return `${nsPart}:${keyPart}`;\n });\n attributes.push(`data-upstart-i18n={\\`${parts.join(\",\")}\\`}`);\n } else {\n const keys = allI18nKeys.map((t: I18nKeyInfo) => escapeProp(t.fullKey)).join(\",\");\n attributes.push(`data-upstart-i18n=\"${keys}\"`);\n }\n }\n\n // Step 2: Text leaf elements — registry tracking + non-editable flag\n if (isTextLeafElement(jsxNode)) {\n if (!hasI18n) {\n attributes.push('data-upstart-editable-text=\"false\"');\n }\n\n // Find the actual JSXText node to track its location\n const textChild = jsxNode.children.find((c) => c.type === \"JSXText\" && (c as any).value?.trim());\n\n if (textChild && hasRange(textChild)) {\n const id = generateId(state.filePath, textChild);\n const textValue = (textChild as any).value as string;\n\n // Calculate trimmed offsets (exclude leading/trailing whitespace)\n const trimmedStart = textChild.start + (textValue.length - textValue.trimStart().length);\n const trimmedEnd = textChild.end - (textValue.length - textValue.trimEnd().length);\n\n editableRegistry.set(id, {\n file: state.filePath,\n type: \"text\",\n startOffset: trimmedStart,\n endOffset: trimmedEnd,\n originalContent: textValue.trim(),\n context: { parentTag: tagName || \"unknown\" },\n });\n\n attributes.push(`data-upstart-id=\"${id}\"`);\n }\n }\n // Step 3: Non-leaf elements with visible text content (expressions, mixed content)\n else if (!hasI18n && hasVisibleTextContent(jsxNode)) {\n attributes.push('data-upstart-editable-text=\"false\"');\n }\n\n // Track className attribute if it's a string literal\n const classNameAttr = opening.attributes.find(\n (attr): attr is JSXAttribute =>\n attr.type === \"JSXAttribute\" &&\n attr.name.type === \"JSXIdentifier\" &&\n attr.name.name === \"className\" &&\n attr.value?.type === \"Literal\" &&\n typeof (attr.value as any).value === \"string\",\n );\n\n if (classNameAttr && classNameAttr.value && hasRange(classNameAttr.value)) {\n const id = generateId(state.filePath, classNameAttr.value);\n const classValue = (classNameAttr.value as any).value as string;\n\n // +1 and -1 to exclude the quotes\n editableRegistry.set(id, {\n file: state.filePath,\n type: \"className\",\n startOffset: classNameAttr.value.start + 1,\n endOffset: classNameAttr.value.end - 1,\n originalContent: classValue,\n context: { parentTag: tagName || \"unknown\" },\n });\n\n attributes.push(`data-upstart-classname-id=\"${id}\"`);\n }\n\n // Process PascalCase components for additional tracking\n if (tagName && /^[A-Z]/.test(tagName)) {\n // File and component tracking\n attributes.push(`data-upstart-file=\"${escapeProp(state.filePath)}\"`);\n attributes.push(`data-upstart-component=\"${escapeProp(tagName)}\"`);\n\n // Loop context tracking\n if (state.loopStack.length > 0) {\n const loop = state.loopStack[state.loopStack.length - 1];\n attributes.push(`data-upstart-loop-item=\"${escapeProp(loop.itemName)}\"`);\n if (loop.indexName) {\n attributes.push(`data-upstart-loop-index={${loop.indexName}}`);\n }\n attributes.push(`data-upstart-loop-array=\"${escapeProp(loop.arrayExpr)}\"`);\n }\n\n // Analyze each prop for data bindings\n for (const attr of opening.attributes) {\n if (attr.type !== \"JSXAttribute\" || attr.name.type !== \"JSXIdentifier\") {\n continue;\n }\n\n const propName = attr.name.name;\n const binding = analyzeBinding(attr.value, state.code);\n\n if (binding) {\n attributes.push(`data-upstart-prop-${propName.toLowerCase()}=\"${escapeProp(binding.path)}\"`);\n\n // Track conditional expressions\n if (binding.isConditional) {\n attributes.push(`data-upstart-conditional-${propName.toLowerCase()}=\"true\"`);\n }\n }\n }\n }\n\n // Inject attributes if any\n if (insertPos !== -1 && attributes.length > 0) {\n const attrString = \" \" + attributes.join(\" \");\n state.s.appendLeft(insertPos, attrString);\n modified = true;\n }\n }\n\n next();\n },\n });\n\n if (!modified) {\n return null;\n }\n\n return {\n code: s.toString(),\n map: s.generateMap({ hires: true }),\n };\n}\n\n// Helper: Get JSX element name\nfunction getJSXElementName(opening: JSXOpeningElement): string | null {\n if (opening.name.type === \"JSXIdentifier\") {\n return opening.name.name;\n }\n\n // Handle JSXMemberExpression like <Foo.Bar>\n if (opening.name.type === \"JSXMemberExpression\") {\n let current = opening.name;\n while (current.property) {\n if (current.property.type === \"JSXIdentifier\") {\n return current.property.name;\n }\n if (current.type === \"JSXMemberExpression\") {\n current = current.property as any;\n } else {\n break;\n }\n }\n }\n\n return null;\n}\n\n// Helper: Find where to insert attributes in JSX opening tag\nfunction getAttributeInsertPosition(opening: JSXOpeningElement, code: string): number {\n // If there are existing attributes, insert before the first one\n if (opening.attributes.length > 0) {\n const firstAttr = opening.attributes[0];\n if (hasRange(firstAttr)) {\n return firstAttr.start;\n }\n }\n\n // Otherwise, insert after the tag name\n if (opening.name.type === \"JSXIdentifier\" && hasRange(opening.name)) {\n return opening.name.end;\n }\n\n if (opening.name.type === \"JSXMemberExpression\" && hasRange(opening.name)) {\n return opening.name.end;\n }\n\n return -1;\n}\n\n// Helper: Analyze a prop value to extract data binding\nfunction analyzeBinding(\n value: JSXAttribute[\"value\"],\n code: string,\n): {\n path: string;\n isConditional?: boolean;\n} | null {\n if (!value) {\n return null;\n }\n\n if (value.type === \"JSXExpressionContainer\") {\n const expr = value.expression;\n\n if (expr.type === \"JSXEmptyExpression\") {\n return null;\n }\n\n return analyzeExpression(expr, code);\n }\n\n return null;\n}\n\n// Helper: Analyze any expression to extract binding info\nfunction analyzeExpression(\n expr: Expression,\n code: string,\n): {\n path: string;\n isConditional?: boolean;\n} | null {\n // Handle member expressions: user.name, product.price, etc.\n if (expr.type === \"MemberExpression\") {\n const path = exprToString(expr, code);\n return { path };\n }\n\n // Handle identifiers: userName, price, etc.\n if (expr.type === \"Identifier\") {\n return { path: expr.name };\n }\n\n // Handle conditional expressions: condition ? value1 : value2\n if (expr.type === \"ConditionalExpression\") {\n // Try to extract from the consequent\n const consequent = analyzeExpression(expr.consequent, code);\n if (consequent) {\n return {\n ...consequent,\n isConditional: true,\n path: exprToString(expr, code),\n };\n }\n }\n\n // Handle logical expressions: value1 && value2, value1 || value2\n if (expr.type === \"LogicalExpression\") {\n const right = analyzeExpression(expr.right, code);\n if (right) {\n return {\n ...right,\n isConditional: true,\n path: exprToString(expr, code),\n };\n }\n }\n\n // For other complex expressions, just return the path\n const exprWithRange = expr as any;\n if (exprWithRange.start !== undefined && exprWithRange.end !== undefined) {\n return {\n path: code.slice(exprWithRange.start, exprWithRange.end),\n };\n }\n\n return null;\n}\n\n// Helper: Convert expression AST to string\nfunction exprToString(expr: Expression, code: string): string {\n // Use the source range to get the actual code\n const exprWithRange = expr as any;\n if (exprWithRange.start !== undefined && exprWithRange.end !== undefined) {\n return code.slice(exprWithRange.start, exprWithRange.end);\n }\n\n // Fallback: reconstruct from AST\n if (expr.type === \"Identifier\") {\n return expr.name;\n }\n\n if (expr.type === \"MemberExpression\") {\n const obj = exprToString(expr.object as Expression, code);\n const prop =\n expr.property.type === \"Identifier\" && !expr.computed\n ? expr.property.name\n : exprToString(expr.property as Expression, code);\n return expr.computed ? `${obj}[${prop}]` : `${obj}.${prop}`;\n }\n\n if (expr.type === \"ConditionalExpression\") {\n const test = exprToString(expr.test, code);\n const consequent = exprToString(expr.consequent, code);\n const alternate = exprToString(expr.alternate, code);\n return `${test} ? ${consequent} : ${alternate}`;\n }\n\n if (expr.type === \"LogicalExpression\") {\n const left = exprToString(expr.left, code);\n const right = exprToString(expr.right, code);\n return `${left} ${expr.operator} ${right}`;\n }\n\n return \"\";\n}\n\n// Helper: Detect .map() calls and extract loop context\nfunction detectMapCall(node: CallExpression, code: string): LoopContext | null {\n // Check if this is a .map() call\n if (\n node.callee.type !== \"MemberExpression\" ||\n node.callee.property.type !== \"Identifier\" ||\n node.callee.property.name !== \"map\"\n ) {\n return null;\n }\n\n const callback = node.arguments[0];\n\n if (!callback) {\n return null;\n }\n\n // Check if callback is an arrow function or function expression\n if (callback.type !== \"ArrowFunctionExpression\" && callback.type !== \"FunctionExpression\") {\n return null;\n }\n\n const params = callback.params;\n const itemParam = params[0];\n const indexParam = params[1];\n\n if (!itemParam) {\n return null;\n }\n\n // Extract parameter names\n const itemName = itemParam.type === \"Identifier\" ? itemParam.name : \"item\";\n const indexName = indexParam?.type === \"Identifier\" ? indexParam.name : null;\n\n // Get the array expression\n const arrayExpr = exprToString(node.callee.object as Expression, code);\n\n return {\n itemName,\n indexName,\n arrayExpr,\n };\n}\n\n// Helper: Escape prop values for JSX attributes\nfunction escapeProp(value: string): string {\n return value\n .replace(/&/g, \"&\")\n .replace(/\"/g, \""\")\n .replace(/'/g, \"'\")\n .replace(/</g, \"<\")\n .replace(/>/g, \">\");\n}\n\n// Helper: Track useTranslation() calls to map t-function variable names to their namespace\nfunction trackUseTranslation(node: Node, state: TransformState): void {\n if (node.type !== \"VariableDeclaration\") return;\n\n const decl = node as any;\n for (const declarator of decl.declarations) {\n if (\n declarator.id?.type !== \"ObjectPattern\" ||\n !declarator.init ||\n declarator.init.type !== \"CallExpression\"\n ) {\n continue;\n }\n\n const callExpr = declarator.init;\n if (callExpr.callee?.type !== \"Identifier\" || callExpr.callee.name !== \"useTranslation\") {\n continue;\n }\n\n const namespace = extractUseTranslationNamespace(callExpr);\n\n for (const prop of declarator.id.properties) {\n if (prop.type !== \"Property\") continue;\n if (prop.key?.type !== \"Identifier\" || prop.key.name !== \"t\") continue;\n\n // Handle aliased destructuring: { t: translate } → maps \"translate\" to namespace\n const localName = prop.value?.type === \"Identifier\" ? prop.value.name : \"t\";\n state.tFunctions.set(localName, namespace);\n }\n }\n}\n\n// Helper: Extract namespace from useTranslation() call arguments\nfunction extractUseTranslationNamespace(callExpr: any): string {\n const firstArg = callExpr.arguments?.[0];\n if (!firstArg) return \"translation\";\n\n // String literal: useTranslation(\"dashboard\")\n if (firstArg.type === \"Literal\" && typeof firstArg.value === \"string\") {\n return firstArg.value;\n }\n\n // Array expression: useTranslation([\"dashboard\", \"common\"])\n if (firstArg.type === \"ArrayExpression\" && firstArg.elements?.length > 0) {\n const first = firstArg.elements[0];\n if (first?.type === \"Literal\" && typeof first.value === \"string\") {\n return first.value;\n }\n }\n\n return \"translation\";\n}\n\n// Helper: Resolve a JSX attribute value to a string.\n// Handles string literals directly, and JSXExpressionContainer by resolving\n// Identifier references via the constants map, or returning the expression source for dynamic values.\nfunction resolveJSXAttrValue(\n value: any,\n code: string,\n constants: Map<string, string>,\n): { value: string; expr: string | null } | null {\n if (!value) return null;\n\n if (value.type === \"Literal\" && typeof value.value === \"string\") {\n return { value: value.value, expr: null };\n }\n\n if (value.type === \"JSXExpressionContainer\") {\n const expression = value.expression;\n if (!expression) return null;\n\n // Resolve simple identifiers via constants map (e.g. i18nKey={labelKey} where const labelKey = \"...\")\n if (expression.type === \"Identifier\" && constants.has(expression.name)) {\n return { value: constants.get(expression.name)!, expr: null };\n }\n\n // Dynamic expression — return source text so caller can emit a runtime JSX expression\n if (hasRange(expression)) {\n const src = code.slice(expression.start, expression.end);\n return { value: src, expr: src };\n }\n }\n\n return null;\n}\n\n// Helper: Detect <Trans i18nKey=\"...\" /> component and extract i18n key info\nfunction detectTransComponent(\n jsxElement: JSXElement,\n code: string,\n constants: Map<string, string>,\n): I18nKeyInfo | null {\n const opening = jsxElement.openingElement;\n const tagName = getJSXElementName(opening);\n\n if (tagName !== \"Trans\") return null;\n\n let keyResult: { value: string; expr: string | null } | null = null;\n let nsResult: { value: string; expr: string | null } | null = null;\n\n for (const attr of opening.attributes) {\n if (attr.type !== \"JSXAttribute\" || attr.name.type !== \"JSXIdentifier\") continue;\n\n const attrName = attr.name.name;\n\n if (attrName === \"i18nKey\") {\n keyResult = resolveJSXAttrValue(attr.value, code, constants);\n }\n\n if (attrName === \"ns\") {\n nsResult = resolveJSXAttrValue(attr.value, code, constants);\n }\n }\n\n if (!keyResult) {\n return null;\n }\n\n const key = keyResult.value;\n const namespace = nsResult?.value ?? \"translation\";\n const keyExpr = keyResult.expr;\n const nsExpr = nsResult?.expr ?? null;\n\n return {\n fullKey: `${namespace}:${key}`,\n key,\n namespace,\n keyExpr,\n nsExpr,\n };\n}\n\n// Helper: Scan an element's direct children for <Trans> components and extract i18n key info.\n// Handles direct <Trans> children and <Trans> inside JSXExpressionContainer (e.g. {show && <Trans>}).\nfunction findTransInChildren(\n jsxElement: JSXElement,\n code: string,\n constants: Map<string, string>,\n): I18nKeyInfo[] {\n const results: I18nKeyInfo[] = [];\n\n for (const child of jsxElement.children) {\n if (child.type === \"JSXElement\") {\n const info = detectTransComponent(child as JSXElement, code, constants);\n if (info) results.push(info);\n }\n\n if (child.type === \"JSXExpressionContainer\") {\n const expr = (child as any).expression;\n if (expr && expr.type !== \"JSXEmptyExpression\") {\n findTransInExpression(expr, code, results, constants);\n }\n }\n }\n\n return results;\n}\n\n// Helper: Recursively search an expression tree for <Trans> JSXElements.\n// Handles LogicalExpression (&&, ||), ConditionalExpression (?:), CallExpression (.map()),\n// ArrowFunctionExpression/FunctionExpression (callbacks), BlockStatement, and direct JSXElement.\nfunction findTransInExpression(\n expr: any,\n code: string,\n results: I18nKeyInfo[],\n constants: Map<string, string>,\n): void {\n if (!expr || !expr.type) return;\n\n if (expr.type === \"JSXElement\") {\n const info = detectTransComponent(expr as JSXElement, code, constants);\n if (info) results.push(info);\n return;\n }\n\n if (expr.type === \"LogicalExpression\") {\n findTransInExpression(expr.left, code, results, constants);\n findTransInExpression(expr.right, code, results, constants);\n return;\n }\n\n if (expr.type === \"ConditionalExpression\") {\n findTransInExpression(expr.consequent, code, results, constants);\n findTransInExpression(expr.alternate, code, results, constants);\n return;\n }\n\n // Handle .map() and other call expressions — recurse into arguments\n if (expr.type === \"CallExpression\") {\n for (const arg of expr.arguments) {\n findTransInExpression(arg, code, results, constants);\n }\n return;\n }\n\n // Handle arrow/function expressions — recurse into body\n if (expr.type === \"ArrowFunctionExpression\" || expr.type === \"FunctionExpression\") {\n findTransInExpression(expr.body, code, results, constants);\n return;\n }\n\n // Handle block bodies (arrow functions with braces)\n if (expr.type === \"BlockStatement\") {\n for (const stmt of expr.body) {\n if (stmt.type === \"ReturnStatement\" && stmt.argument) {\n findTransInExpression(stmt.argument, code, results, constants);\n }\n }\n return;\n }\n}\n\n// Helper: Scan an element's direct children for t() function calls and extract i18n key info.\nfunction findTCallsInChildren(jsxElement: JSXElement, code: string, state: TransformState): I18nKeyInfo[] {\n if (state.tFunctions.size === 0) return [];\n\n const results: I18nKeyInfo[] = [];\n\n for (const child of jsxElement.children) {\n if (child.type === \"JSXExpressionContainer\") {\n const expr = (child as any).expression;\n if (expr && expr.type !== \"JSXEmptyExpression\") {\n findTCallInExpression(expr, code, results, state);\n }\n }\n }\n\n return results;\n}\n\n// Helper: Recursively search an expression tree for t() calls.\n// Handles LogicalExpression (&&, ||), ConditionalExpression (?:), CallExpression (.map()),\n// ArrowFunctionExpression/FunctionExpression (callbacks), BlockStatement, and direct CallExpression.\nfunction findTCallInExpression(expr: any, code: string, results: I18nKeyInfo[], state: TransformState): void {\n if (!expr || !expr.type) return;\n\n if (expr.type === \"CallExpression\") {\n const info = detectTCall(expr, code, state);\n if (info) {\n results.push(info);\n return;\n }\n // Not a t() call — recurse into arguments (for .map() callbacks etc.)\n for (const arg of expr.arguments) {\n findTCallInExpression(arg, code, results, state);\n }\n return;\n }\n\n if (expr.type === \"LogicalExpression\") {\n findTCallInExpression(expr.left, code, results, state);\n findTCallInExpression(expr.right, code, results, state);\n return;\n }\n\n if (expr.type === \"ConditionalExpression\") {\n findTCallInExpression(expr.consequent, code, results, state);\n findTCallInExpression(expr.alternate, code, results, state);\n return;\n }\n\n if (expr.type === \"ArrowFunctionExpression\" || expr.type === \"FunctionExpression\") {\n findTCallInExpression(expr.body, code, results, state);\n return;\n }\n\n if (expr.type === \"BlockStatement\") {\n for (const stmt of expr.body) {\n if (stmt.type === \"ReturnStatement\" && stmt.argument) {\n findTCallInExpression(stmt.argument, code, results, state);\n }\n }\n return;\n }\n}\n\n// Helper: Detect a t(\"key\") call and extract i18n key info.\nfunction detectTCall(callExpr: any, code: string, state: TransformState): I18nKeyInfo | null {\n if (callExpr.callee?.type !== \"Identifier\") return null;\n\n const calleeName = callExpr.callee.name;\n const namespace = state.tFunctions.get(calleeName);\n if (namespace === undefined) return null;\n\n const firstArg = callExpr.arguments?.[0];\n if (!firstArg) return null;\n\n // Static string key: t(\"features.title\")\n if (firstArg.type === \"Literal\" && typeof firstArg.value === \"string\") {\n const key = firstArg.value;\n return { fullKey: `${namespace}:${key}`, key, namespace, keyExpr: null, nsExpr: null };\n }\n\n // Dynamic key — try constant resolution first, then fall back to expression\n if (hasRange(firstArg)) {\n const exprSrc = code.slice(firstArg.start, firstArg.end);\n\n if (firstArg.type === \"Identifier\" && state.constants.has(firstArg.name)) {\n const key = state.constants.get(firstArg.name)!;\n return { fullKey: `${namespace}:${key}`, key, namespace, keyExpr: null, nsExpr: null };\n }\n\n return { fullKey: `${namespace}:${exprSrc}`, key: exprSrc, namespace, keyExpr: exprSrc, nsExpr: null };\n }\n\n return null;\n}\n\n// Helper: Check if element is a \"leaf\" with only static text (no nested elements or expressions)\nfunction isTextLeafElement(jsxElement: JSXElement): boolean {\n let hasText = false;\n\n for (const child of jsxElement.children) {\n if (child.type === \"JSXText\") {\n const textValue = (child as any).value;\n if (textValue?.trim()) {\n hasText = true;\n }\n } else if (\n child.type === \"JSXElement\" ||\n child.type === \"JSXFragment\" ||\n child.type === \"JSXExpressionContainer\" ||\n child.type === \"JSXSpreadChild\"\n ) {\n // Has non-text children, not a leaf element\n return false;\n }\n }\n\n return hasText;\n}\n\n// Helper: Check if element has any visible text content (static or dynamic).\n// Broader than isTextLeafElement — returns true even if the element has other child types.\n// Detects JSXText with non-whitespace content and expression containers with\n// text-producing expressions (Identifier, MemberExpression, Literal, TemplateLiteral).\n// Skips CallExpression, ConditionalExpression, LogicalExpression (these typically produce elements).\nfunction hasVisibleTextContent(jsxElement: JSXElement): boolean {\n for (const child of jsxElement.children) {\n if (child.type === \"JSXText\") {\n const textValue = (child as any).value;\n if (textValue?.trim()) {\n return true;\n }\n continue;\n }\n\n if (child.type === \"JSXExpressionContainer\") {\n const expr = (child as any).expression;\n if (!expr || expr.type === \"JSXEmptyExpression\") {\n continue;\n }\n if (\n expr.type === \"Identifier\" ||\n expr.type === \"MemberExpression\" ||\n expr.type === \"Literal\" ||\n expr.type === \"TemplateLiteral\"\n ) {\n return true;\n }\n }\n }\n\n return false;\n}\n\nexport default upstartEditor.vite;\n"],"mappings":"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"}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@upstart.gg/vite-plugins",
|
|
3
|
-
"version": "0.0.
|
|
3
|
+
"version": "0.0.50",
|
|
4
4
|
"type": "module",
|
|
5
5
|
"files": [
|
|
6
6
|
"dist",
|
|
@@ -23,7 +23,7 @@
|
|
|
23
23
|
"oxc-parser": "^0.101.0",
|
|
24
24
|
"unplugin": "^2.3.11",
|
|
25
25
|
"zimmerframe": "^1.1.4",
|
|
26
|
-
"@upstart.gg/sdk": "^0.0.
|
|
26
|
+
"@upstart.gg/sdk": "^0.0.50"
|
|
27
27
|
},
|
|
28
28
|
"devDependencies": {
|
|
29
29
|
"@types/bun": "1.3.5",
|
|
@@ -37,32 +37,88 @@
|
|
|
37
37
|
"@rolldown/binding-linux-x64-gnu": "1.0.0-beta.53"
|
|
38
38
|
},
|
|
39
39
|
"exports": {
|
|
40
|
-
"./
|
|
41
|
-
"./
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
"./vite-plugin-upstart-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
"./vite-plugin-upstart-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
40
|
+
"./package.json": "./package.json",
|
|
41
|
+
"./upstart-editor-api": {
|
|
42
|
+
"import": "./dist/upstart-editor-api.js",
|
|
43
|
+
"types": "./dist/upstart-editor-api.d.ts",
|
|
44
|
+
"bun": "./src/upstart-editor-api.ts"
|
|
45
|
+
},
|
|
46
|
+
"./vite-plugin-upstart-attrs": {
|
|
47
|
+
"import": "./dist/vite-plugin-upstart-attrs.js",
|
|
48
|
+
"types": "./dist/vite-plugin-upstart-attrs.d.ts",
|
|
49
|
+
"bun": "./src/vite-plugin-upstart-attrs.ts"
|
|
50
|
+
},
|
|
51
|
+
"./vite-plugin-upstart-branding/plugin": {
|
|
52
|
+
"import": "./dist/vite-plugin-upstart-branding/plugin.js",
|
|
53
|
+
"types": "./dist/vite-plugin-upstart-branding/plugin.d.ts",
|
|
54
|
+
"bun": "./src/vite-plugin-upstart-branding/plugin.ts"
|
|
55
|
+
},
|
|
56
|
+
"./vite-plugin-upstart-branding/runtime": {
|
|
57
|
+
"import": "./dist/vite-plugin-upstart-branding/runtime.js",
|
|
58
|
+
"types": "./dist/vite-plugin-upstart-branding/runtime.d.ts",
|
|
59
|
+
"bun": "./src/vite-plugin-upstart-branding/runtime.ts"
|
|
60
|
+
},
|
|
61
|
+
"./vite-plugin-upstart-branding/types": {
|
|
62
|
+
"import": "./dist/vite-plugin-upstart-branding/types.js",
|
|
63
|
+
"types": "./dist/vite-plugin-upstart-branding/types.d.ts",
|
|
64
|
+
"bun": "./src/vite-plugin-upstart-branding/types.ts"
|
|
65
|
+
},
|
|
66
|
+
"./vite-plugin-upstart-editor/plugin": {
|
|
67
|
+
"import": "./dist/vite-plugin-upstart-editor/plugin.js",
|
|
68
|
+
"types": "./dist/vite-plugin-upstart-editor/plugin.d.ts",
|
|
69
|
+
"bun": "./src/vite-plugin-upstart-editor/plugin.ts"
|
|
70
|
+
},
|
|
71
|
+
"./vite-plugin-upstart-editor/runtime": {
|
|
72
|
+
"import": "./dist/vite-plugin-upstart-editor/runtime/index.js",
|
|
73
|
+
"types": "./dist/vite-plugin-upstart-editor/runtime/index.d.ts",
|
|
74
|
+
"bun": "./src/vite-plugin-upstart-editor/runtime/index.ts"
|
|
75
|
+
},
|
|
76
|
+
"./vite-plugin-upstart-editor/runtime/click-handler": {
|
|
77
|
+
"import": "./dist/vite-plugin-upstart-editor/runtime/click-handler.js",
|
|
78
|
+
"types": "./dist/vite-plugin-upstart-editor/runtime/click-handler.d.ts",
|
|
79
|
+
"bun": "./src/vite-plugin-upstart-editor/runtime/click-handler.ts"
|
|
80
|
+
},
|
|
81
|
+
"./vite-plugin-upstart-editor/runtime/error-handler": {
|
|
82
|
+
"import": "./dist/vite-plugin-upstart-editor/runtime/error-handler.js",
|
|
83
|
+
"types": "./dist/vite-plugin-upstart-editor/runtime/error-handler.d.ts",
|
|
84
|
+
"bun": "./src/vite-plugin-upstart-editor/runtime/error-handler.ts"
|
|
85
|
+
},
|
|
86
|
+
"./vite-plugin-upstart-editor/runtime/hover-overlay": {
|
|
87
|
+
"import": "./dist/vite-plugin-upstart-editor/runtime/hover-overlay.js",
|
|
88
|
+
"types": "./dist/vite-plugin-upstart-editor/runtime/hover-overlay.d.ts",
|
|
89
|
+
"bun": "./src/vite-plugin-upstart-editor/runtime/hover-overlay.ts"
|
|
90
|
+
},
|
|
91
|
+
"./vite-plugin-upstart-editor/runtime/text-editor": {
|
|
92
|
+
"import": "./dist/vite-plugin-upstart-editor/runtime/text-editor.js",
|
|
93
|
+
"types": "./dist/vite-plugin-upstart-editor/runtime/text-editor.d.ts",
|
|
94
|
+
"bun": "./src/vite-plugin-upstart-editor/runtime/text-editor.ts"
|
|
95
|
+
},
|
|
96
|
+
"./vite-plugin-upstart-editor/runtime/types": {
|
|
97
|
+
"import": "./dist/vite-plugin-upstart-editor/runtime/types.js",
|
|
98
|
+
"types": "./dist/vite-plugin-upstart-editor/runtime/types.d.ts",
|
|
99
|
+
"bun": "./src/vite-plugin-upstart-editor/runtime/types.ts"
|
|
100
|
+
},
|
|
101
|
+
"./vite-plugin-upstart-editor/runtime/utils": {
|
|
102
|
+
"import": "./dist/vite-plugin-upstart-editor/runtime/utils.js",
|
|
103
|
+
"types": "./dist/vite-plugin-upstart-editor/runtime/utils.d.ts",
|
|
104
|
+
"bun": "./src/vite-plugin-upstart-editor/runtime/utils.ts"
|
|
105
|
+
},
|
|
106
|
+
"./vite-plugin-upstart-theme": {
|
|
107
|
+
"import": "./dist/vite-plugin-upstart-theme.js",
|
|
108
|
+
"types": "./dist/vite-plugin-upstart-theme.d.ts",
|
|
109
|
+
"bun": "./src/vite-plugin-upstart-theme.ts"
|
|
110
|
+
}
|
|
55
111
|
},
|
|
56
112
|
"peerDependencies": {
|
|
57
113
|
"zod": "4.2.1",
|
|
58
|
-
"@upstart.gg/sdk": "^0.0.
|
|
114
|
+
"@upstart.gg/sdk": "^0.0.50"
|
|
59
115
|
},
|
|
60
116
|
"author": "Upstart",
|
|
61
117
|
"publishConfig": {
|
|
62
118
|
"access": "public"
|
|
63
119
|
},
|
|
64
120
|
"scripts": {
|
|
65
|
-
"test": "vitest --run",
|
|
121
|
+
"test": "bun run --bun vitest --run",
|
|
66
122
|
"test:bun": "bun test",
|
|
67
123
|
"build": "tsdown",
|
|
68
124
|
"lint": "biome check . --write && tsc --noEmit",
|
|
@@ -36,20 +36,6 @@ describe("upstart-editor-vite-plugin", () => {
|
|
|
36
36
|
expect(result).toContain('data-upstart-component="MyComponent"');
|
|
37
37
|
});
|
|
38
38
|
|
|
39
|
-
test("should track prop bindings with member expressions", () => {
|
|
40
|
-
const code = `
|
|
41
|
-
export default function App() {
|
|
42
|
-
const user = { name: "Alice" };
|
|
43
|
-
return <MyComponent name={user.name} />;
|
|
44
|
-
}
|
|
45
|
-
`;
|
|
46
|
-
|
|
47
|
-
const result = transform(code);
|
|
48
|
-
|
|
49
|
-
expect(result).toContain('data-upstart-prop-name="user.name"');
|
|
50
|
-
expect(result).toContain('data-upstart-datasource-name="user"');
|
|
51
|
-
});
|
|
52
|
-
|
|
53
39
|
test("should track prop bindings with identifiers", () => {
|
|
54
40
|
const code = `
|
|
55
41
|
export default function App() {
|
|
@@ -63,19 +49,6 @@ describe("upstart-editor-vite-plugin", () => {
|
|
|
63
49
|
expect(result).toContain('data-upstart-prop-name="userName"');
|
|
64
50
|
});
|
|
65
51
|
|
|
66
|
-
test("should track record IDs for datasource bindings", () => {
|
|
67
|
-
const code = `
|
|
68
|
-
export default function App() {
|
|
69
|
-
const user = { $id: 1, name: "Alice" };
|
|
70
|
-
return <MyComponent name={user.name} />;
|
|
71
|
-
}
|
|
72
|
-
`;
|
|
73
|
-
|
|
74
|
-
const result = transform(code);
|
|
75
|
-
|
|
76
|
-
expect(result).toContain("data-upstart-record-id-name={user.$id}");
|
|
77
|
-
});
|
|
78
|
-
|
|
79
52
|
test("should handle conditional prop expressions", () => {
|
|
80
53
|
const code = `
|
|
81
54
|
export default function App() {
|
|
@@ -10,7 +10,6 @@ import type {
|
|
|
10
10
|
JSXOpeningElement,
|
|
11
11
|
JSXAttribute,
|
|
12
12
|
Expression,
|
|
13
|
-
MemberExpression,
|
|
14
13
|
CallExpression,
|
|
15
14
|
} from "estree-jsx";
|
|
16
15
|
|
|
@@ -359,7 +358,7 @@ export function transformWithOxc(code: string, filePath: string) {
|
|
|
359
358
|
const loop = state.loopStack[state.loopStack.length - 1];
|
|
360
359
|
attributes.push(`data-upstart-loop-item="${escapeProp(loop.itemName)}"`);
|
|
361
360
|
if (loop.indexName) {
|
|
362
|
-
attributes.push(`data-upstart-loop-index={${loop.indexName
|
|
361
|
+
attributes.push(`data-upstart-loop-index={${loop.indexName}}`);
|
|
363
362
|
}
|
|
364
363
|
attributes.push(`data-upstart-loop-array="${escapeProp(loop.arrayExpr)}"`);
|
|
365
364
|
}
|
|
@@ -376,16 +375,6 @@ export function transformWithOxc(code: string, filePath: string) {
|
|
|
376
375
|
if (binding) {
|
|
377
376
|
attributes.push(`data-upstart-prop-${propName.toLowerCase()}="${escapeProp(binding.path)}"`);
|
|
378
377
|
|
|
379
|
-
if (binding.datasource) {
|
|
380
|
-
attributes.push(
|
|
381
|
-
`data-upstart-datasource-${propName.toLowerCase()}="${escapeProp(binding.datasource)}"`,
|
|
382
|
-
);
|
|
383
|
-
}
|
|
384
|
-
|
|
385
|
-
if (binding.recordId) {
|
|
386
|
-
attributes.push(`data-upstart-record-id-${propName.toLowerCase()}={${binding.recordId}}`);
|
|
387
|
-
}
|
|
388
|
-
|
|
389
378
|
// Track conditional expressions
|
|
390
379
|
if (binding.isConditional) {
|
|
391
380
|
attributes.push(`data-upstart-conditional-${propName.toLowerCase()}="true"`);
|
|
@@ -468,8 +457,6 @@ function analyzeBinding(
|
|
|
468
457
|
code: string,
|
|
469
458
|
): {
|
|
470
459
|
path: string;
|
|
471
|
-
datasource?: string;
|
|
472
|
-
recordId?: string;
|
|
473
460
|
isConditional?: boolean;
|
|
474
461
|
} | null {
|
|
475
462
|
if (!value) {
|
|
@@ -495,17 +482,12 @@ function analyzeExpression(
|
|
|
495
482
|
code: string,
|
|
496
483
|
): {
|
|
497
484
|
path: string;
|
|
498
|
-
datasource?: string;
|
|
499
|
-
recordId?: string;
|
|
500
485
|
isConditional?: boolean;
|
|
501
486
|
} | null {
|
|
502
487
|
// Handle member expressions: user.name, product.price, etc.
|
|
503
488
|
if (expr.type === "MemberExpression") {
|
|
504
489
|
const path = exprToString(expr, code);
|
|
505
|
-
|
|
506
|
-
const recordId = extractRecordId(expr);
|
|
507
|
-
|
|
508
|
-
return { path, datasource, recordId };
|
|
490
|
+
return { path };
|
|
509
491
|
}
|
|
510
492
|
|
|
511
493
|
// Handle identifiers: userName, price, etc.
|
|
@@ -587,42 +569,6 @@ function exprToString(expr: Expression, code: string): string {
|
|
|
587
569
|
return "";
|
|
588
570
|
}
|
|
589
571
|
|
|
590
|
-
// Helper: Extract datasource name from member expression
|
|
591
|
-
function extractDatasource(expr: MemberExpression): string | undefined {
|
|
592
|
-
let current: Expression = expr.object as Expression;
|
|
593
|
-
|
|
594
|
-
// Traverse to the root object
|
|
595
|
-
while (current.type === "MemberExpression") {
|
|
596
|
-
current = current.object as Expression;
|
|
597
|
-
}
|
|
598
|
-
|
|
599
|
-
if (current.type === "Identifier") {
|
|
600
|
-
return current.name;
|
|
601
|
-
}
|
|
602
|
-
|
|
603
|
-
return undefined;
|
|
604
|
-
}
|
|
605
|
-
|
|
606
|
-
// Helper: Extract record ID reference
|
|
607
|
-
function extractRecordId(expr: MemberExpression): string | undefined {
|
|
608
|
-
const obj = expr.object;
|
|
609
|
-
|
|
610
|
-
if (obj.type === "Identifier") {
|
|
611
|
-
// Assume the ID field is `${object}.$id`
|
|
612
|
-
return `${obj.name}.$id`;
|
|
613
|
-
}
|
|
614
|
-
|
|
615
|
-
if (obj.type === "MemberExpression") {
|
|
616
|
-
// For nested objects like user.profile.id
|
|
617
|
-
const datasource = extractDatasource(obj);
|
|
618
|
-
if (datasource) {
|
|
619
|
-
return `${datasource}.$id`;
|
|
620
|
-
}
|
|
621
|
-
}
|
|
622
|
-
|
|
623
|
-
return undefined;
|
|
624
|
-
}
|
|
625
|
-
|
|
626
572
|
// Helper: Detect .map() calls and extract loop context
|
|
627
573
|
function detectMapCall(node: CallExpression, code: string): LoopContext | null {
|
|
628
574
|
// Check if this is a .map() call
|