@sun-asterisk/sungen 1.0.24 → 2.0.1

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.
Files changed (378) hide show
  1. package/README.md +198 -74
  2. package/dist/cli/commands/add.d.ts.map +1 -1
  3. package/dist/cli/commands/add.js +5 -3
  4. package/dist/cli/commands/add.js.map +1 -1
  5. package/dist/cli/commands/generate.d.ts.map +1 -1
  6. package/dist/cli/commands/generate.js +110 -35
  7. package/dist/cli/commands/generate.js.map +1 -1
  8. package/dist/cli/index.d.ts +2 -2
  9. package/dist/cli/index.js +5 -16
  10. package/dist/cli/index.js.map +1 -1
  11. package/dist/generators/test-generator/adapters/playwright/templates/steps/actions/frame-enter-action.hbs +1 -0
  12. package/dist/generators/test-generator/adapters/playwright/templates/steps/actions/frame-exit-action.hbs +1 -0
  13. package/dist/generators/test-generator/adapters/playwright/templates/steps/actions/keyboard-global-action.hbs +1 -0
  14. package/dist/generators/test-generator/adapters/playwright/templates/steps/actions/scroll-action.hbs +1 -0
  15. package/dist/generators/test-generator/adapters/playwright/templates/steps/actions/table-action-in-row.hbs +2 -0
  16. package/dist/generators/test-generator/adapters/playwright/templates/steps/assertions/table-cell-by-filter.hbs +3 -0
  17. package/dist/generators/test-generator/adapters/playwright/templates/steps/assertions/table-cell-by-index.hbs +3 -0
  18. package/dist/generators/test-generator/adapters/playwright/templates/steps/assertions/table-column-exists.hbs +2 -0
  19. package/dist/generators/test-generator/adapters/playwright/templates/steps/assertions/table-empty.hbs +2 -0
  20. package/dist/generators/test-generator/adapters/playwright/templates/steps/assertions/table-row-count.hbs +2 -0
  21. package/dist/generators/test-generator/adapters/playwright/templates/steps/assertions/table-row-exists.hbs +2 -0
  22. package/dist/generators/test-generator/adapters/playwright/templates/steps/assertions/table-row-not-exists.hbs +2 -0
  23. package/dist/generators/test-generator/adapters/playwright/templates/steps/assertions/visible-with-value-assertion.hbs +1 -1
  24. package/dist/generators/test-generator/adapters/playwright/templates/steps/partials/locator-base.hbs +11 -2
  25. package/dist/generators/test-generator/adapters/playwright/templates/steps/partials/locator.hbs +11 -2
  26. package/dist/generators/test-generator/code-generator.d.ts +0 -1
  27. package/dist/generators/test-generator/code-generator.d.ts.map +1 -1
  28. package/dist/generators/test-generator/code-generator.js +10 -47
  29. package/dist/generators/test-generator/code-generator.js.map +1 -1
  30. package/dist/generators/test-generator/patterns/assertion-patterns.d.ts.map +1 -1
  31. package/dist/generators/test-generator/patterns/assertion-patterns.js +2 -0
  32. package/dist/generators/test-generator/patterns/assertion-patterns.js.map +1 -1
  33. package/dist/generators/test-generator/patterns/index.d.ts +4 -1
  34. package/dist/generators/test-generator/patterns/index.d.ts.map +1 -1
  35. package/dist/generators/test-generator/patterns/index.js +17 -5
  36. package/dist/generators/test-generator/patterns/index.js.map +1 -1
  37. package/dist/generators/test-generator/patterns/interaction-patterns.js +1 -1
  38. package/dist/generators/test-generator/patterns/interaction-patterns.js.map +1 -1
  39. package/dist/generators/test-generator/patterns/keyboard-patterns.d.ts +7 -0
  40. package/dist/generators/test-generator/patterns/keyboard-patterns.d.ts.map +1 -0
  41. package/dist/generators/test-generator/patterns/keyboard-patterns.js +47 -0
  42. package/dist/generators/test-generator/patterns/keyboard-patterns.js.map +1 -0
  43. package/dist/generators/test-generator/patterns/scope-patterns.d.ts +7 -0
  44. package/dist/generators/test-generator/patterns/scope-patterns.d.ts.map +1 -0
  45. package/dist/generators/test-generator/patterns/scope-patterns.js +36 -0
  46. package/dist/generators/test-generator/patterns/scope-patterns.js.map +1 -0
  47. package/dist/generators/test-generator/patterns/scroll-patterns.d.ts +7 -0
  48. package/dist/generators/test-generator/patterns/scroll-patterns.d.ts.map +1 -0
  49. package/dist/generators/test-generator/patterns/scroll-patterns.js +25 -0
  50. package/dist/generators/test-generator/patterns/scroll-patterns.js.map +1 -0
  51. package/dist/generators/test-generator/patterns/table-patterns.d.ts +7 -0
  52. package/dist/generators/test-generator/patterns/table-patterns.d.ts.map +1 -0
  53. package/dist/generators/test-generator/patterns/table-patterns.js +192 -0
  54. package/dist/generators/test-generator/patterns/table-patterns.js.map +1 -0
  55. package/dist/generators/test-generator/step-mapper.d.ts +5 -3
  56. package/dist/generators/test-generator/step-mapper.d.ts.map +1 -1
  57. package/dist/generators/test-generator/step-mapper.js +38 -27
  58. package/dist/generators/test-generator/step-mapper.js.map +1 -1
  59. package/dist/generators/test-generator/template-engine.d.ts.map +1 -1
  60. package/dist/generators/test-generator/template-engine.js +4 -1
  61. package/dist/generators/test-generator/template-engine.js.map +1 -1
  62. package/dist/generators/test-generator/types.d.ts +7 -24
  63. package/dist/generators/test-generator/types.d.ts.map +1 -1
  64. package/dist/generators/test-generator/types.js +2 -101
  65. package/dist/generators/test-generator/types.js.map +1 -1
  66. package/dist/generators/test-generator/utils/selector-resolver.d.ts +14 -0
  67. package/dist/generators/test-generator/utils/selector-resolver.d.ts.map +1 -1
  68. package/dist/generators/test-generator/utils/selector-resolver.js +37 -11
  69. package/dist/generators/test-generator/utils/selector-resolver.js.map +1 -1
  70. package/dist/orchestrator/project-initializer.d.ts +12 -0
  71. package/dist/orchestrator/project-initializer.d.ts.map +1 -1
  72. package/dist/orchestrator/project-initializer.js +72 -160
  73. package/dist/orchestrator/project-initializer.js.map +1 -1
  74. package/dist/orchestrator/screen-manager.d.ts +1 -32
  75. package/dist/orchestrator/screen-manager.d.ts.map +1 -1
  76. package/dist/orchestrator/screen-manager.js +55 -216
  77. package/dist/orchestrator/screen-manager.js.map +1 -1
  78. package/dist/orchestrator/templates/ai-rules.md +189 -0
  79. package/dist/orchestrator/templates/gitignore +16 -0
  80. package/dist/orchestrator/templates/playwright.config.d.ts +10 -0
  81. package/dist/orchestrator/templates/playwright.config.d.ts.map +1 -0
  82. package/dist/orchestrator/templates/playwright.config.js +77 -0
  83. package/dist/orchestrator/templates/playwright.config.js.map +1 -0
  84. package/dist/orchestrator/templates/playwright.config.ts +80 -0
  85. package/dist/orchestrator/templates/readme.md +197 -0
  86. package/dist/utils/selector-types.d.ts +1 -1
  87. package/dist/utils/selector-types.d.ts.map +1 -1
  88. package/dist/utils/selector-types.js +3 -0
  89. package/dist/utils/selector-types.js.map +1 -1
  90. package/docs/gherkin standards/gherkin-core-standard.md +377 -0
  91. package/docs/gherkin standards/gherkin-core-standard.vi.md +303 -0
  92. package/docs/gherkin-dictionary.md +1071 -0
  93. package/docs/makeauth.md +225 -0
  94. package/package.json +4 -3
  95. package/src/cli/commands/add.ts +5 -3
  96. package/src/cli/commands/generate.ts +90 -38
  97. package/src/cli/index.ts +5 -16
  98. package/src/generators/test-generator/adapters/playwright/templates/steps/actions/frame-enter-action.hbs +1 -0
  99. package/src/generators/test-generator/adapters/playwright/templates/steps/actions/frame-exit-action.hbs +1 -0
  100. package/src/generators/test-generator/adapters/playwright/templates/steps/actions/keyboard-global-action.hbs +1 -0
  101. package/src/generators/test-generator/adapters/playwright/templates/steps/actions/scroll-action.hbs +1 -0
  102. package/src/generators/test-generator/adapters/playwright/templates/steps/actions/table-action-in-row.hbs +2 -0
  103. package/src/generators/test-generator/adapters/playwright/templates/steps/assertions/table-cell-by-filter.hbs +3 -0
  104. package/src/generators/test-generator/adapters/playwright/templates/steps/assertions/table-cell-by-index.hbs +3 -0
  105. package/src/generators/test-generator/adapters/playwright/templates/steps/assertions/table-column-exists.hbs +2 -0
  106. package/src/generators/test-generator/adapters/playwright/templates/steps/assertions/table-empty.hbs +2 -0
  107. package/src/generators/test-generator/adapters/playwright/templates/steps/assertions/table-row-count.hbs +2 -0
  108. package/src/generators/test-generator/adapters/playwright/templates/steps/assertions/table-row-exists.hbs +2 -0
  109. package/src/generators/test-generator/adapters/playwright/templates/steps/assertions/table-row-not-exists.hbs +2 -0
  110. package/src/generators/test-generator/adapters/playwright/templates/steps/assertions/visible-with-value-assertion.hbs +1 -1
  111. package/src/generators/test-generator/adapters/playwright/templates/steps/partials/locator-base.hbs +11 -2
  112. package/src/generators/test-generator/adapters/playwright/templates/steps/partials/locator.hbs +11 -2
  113. package/src/generators/test-generator/code-generator.ts +11 -59
  114. package/src/generators/test-generator/patterns/assertion-patterns.ts +2 -0
  115. package/src/generators/test-generator/patterns/index.ts +12 -3
  116. package/src/generators/test-generator/patterns/interaction-patterns.ts +1 -1
  117. package/src/generators/test-generator/patterns/keyboard-patterns.ts +51 -0
  118. package/src/generators/test-generator/patterns/scope-patterns.ts +40 -0
  119. package/src/generators/test-generator/patterns/scroll-patterns.ts +27 -0
  120. package/src/generators/test-generator/patterns/table-patterns.ts +232 -0
  121. package/src/generators/test-generator/step-mapper.ts +42 -27
  122. package/src/generators/test-generator/template-engine.ts +3 -1
  123. package/src/generators/test-generator/types.ts +7 -112
  124. package/src/generators/test-generator/utils/selector-resolver.ts +70 -25
  125. package/src/orchestrator/project-initializer.ts +75 -160
  126. package/src/orchestrator/screen-manager.ts +61 -233
  127. package/src/orchestrator/templates/ai-rules.md +189 -0
  128. package/src/orchestrator/templates/gitignore +16 -0
  129. package/src/orchestrator/templates/playwright.config.ts +80 -0
  130. package/src/orchestrator/templates/readme.md +197 -0
  131. package/src/utils/selector-types.ts +3 -0
  132. package/dist/cli/commands/cache-clear.d.ts +0 -3
  133. package/dist/cli/commands/cache-clear.d.ts.map +0 -1
  134. package/dist/cli/commands/cache-clear.js +0 -24
  135. package/dist/cli/commands/cache-clear.js.map +0 -1
  136. package/dist/cli/commands/full.d.ts +0 -3
  137. package/dist/cli/commands/full.d.ts.map +0 -1
  138. package/dist/cli/commands/full.js +0 -37
  139. package/dist/cli/commands/full.js.map +0 -1
  140. package/dist/cli/commands/live-scan.d.ts +0 -3
  141. package/dist/cli/commands/live-scan.d.ts.map +0 -1
  142. package/dist/cli/commands/live-scan.js +0 -78
  143. package/dist/cli/commands/live-scan.js.map +0 -1
  144. package/dist/cli/commands/map.d.ts +0 -3
  145. package/dist/cli/commands/map.d.ts.map +0 -1
  146. package/dist/cli/commands/map.js +0 -93
  147. package/dist/cli/commands/map.js.map +0 -1
  148. package/dist/cli/commands/validate.d.ts +0 -3
  149. package/dist/cli/commands/validate.d.ts.map +0 -1
  150. package/dist/cli/commands/validate.js +0 -43
  151. package/dist/cli/commands/validate.js.map +0 -1
  152. package/dist/cli/utils.d.ts +0 -6
  153. package/dist/cli/utils.d.ts.map +0 -1
  154. package/dist/cli/utils.js +0 -101
  155. package/dist/cli/utils.js.map +0 -1
  156. package/dist/config/config-loader.d.ts +0 -51
  157. package/dist/config/config-loader.d.ts.map +0 -1
  158. package/dist/config/config-loader.js +0 -216
  159. package/dist/config/config-loader.js.map +0 -1
  160. package/dist/config/config-schema.d.ts +0 -121
  161. package/dist/config/config-schema.d.ts.map +0 -1
  162. package/dist/config/config-schema.js +0 -7
  163. package/dist/config/config-schema.js.map +0 -1
  164. package/dist/core/live-scanner/config-reader.d.ts +0 -10
  165. package/dist/core/live-scanner/config-reader.d.ts.map +0 -1
  166. package/dist/core/live-scanner/config-reader.js +0 -87
  167. package/dist/core/live-scanner/config-reader.js.map +0 -1
  168. package/dist/core/live-scanner/element-finder.d.ts +0 -20
  169. package/dist/core/live-scanner/element-finder.d.ts.map +0 -1
  170. package/dist/core/live-scanner/element-finder.js +0 -481
  171. package/dist/core/live-scanner/element-finder.js.map +0 -1
  172. package/dist/core/live-scanner/index.d.ts +0 -8
  173. package/dist/core/live-scanner/index.d.ts.map +0 -1
  174. package/dist/core/live-scanner/index.js +0 -33
  175. package/dist/core/live-scanner/index.js.map +0 -1
  176. package/dist/core/live-scanner/matrix-reader.d.ts +0 -17
  177. package/dist/core/live-scanner/matrix-reader.d.ts.map +0 -1
  178. package/dist/core/live-scanner/matrix-reader.js +0 -60
  179. package/dist/core/live-scanner/matrix-reader.js.map +0 -1
  180. package/dist/core/live-scanner/matrix-writer.d.ts +0 -7
  181. package/dist/core/live-scanner/matrix-writer.d.ts.map +0 -1
  182. package/dist/core/live-scanner/matrix-writer.js +0 -103
  183. package/dist/core/live-scanner/matrix-writer.js.map +0 -1
  184. package/dist/core/live-scanner/role-fallback.d.ts +0 -15
  185. package/dist/core/live-scanner/role-fallback.d.ts.map +0 -1
  186. package/dist/core/live-scanner/role-fallback.js +0 -46
  187. package/dist/core/live-scanner/role-fallback.js.map +0 -1
  188. package/dist/core/live-scanner/scanner.d.ts +0 -22
  189. package/dist/core/live-scanner/scanner.d.ts.map +0 -1
  190. package/dist/core/live-scanner/scanner.js +0 -303
  191. package/dist/core/live-scanner/scanner.js.map +0 -1
  192. package/dist/core/live-scanner/step-replayer.d.ts +0 -26
  193. package/dist/core/live-scanner/step-replayer.d.ts.map +0 -1
  194. package/dist/core/live-scanner/step-replayer.js +0 -473
  195. package/dist/core/live-scanner/step-replayer.js.map +0 -1
  196. package/dist/core/live-scanner/types.d.ts +0 -52
  197. package/dist/core/live-scanner/types.d.ts.map +0 -1
  198. package/dist/core/live-scanner/types.js +0 -14
  199. package/dist/core/live-scanner/types.js.map +0 -1
  200. package/dist/core/selector-base/annotation-handler.d.ts +0 -45
  201. package/dist/core/selector-base/annotation-handler.d.ts.map +0 -1
  202. package/dist/core/selector-base/annotation-handler.js +0 -102
  203. package/dist/core/selector-base/annotation-handler.js.map +0 -1
  204. package/dist/core/selector-base/base-generator.d.ts +0 -49
  205. package/dist/core/selector-base/base-generator.d.ts.map +0 -1
  206. package/dist/core/selector-base/base-generator.js +0 -214
  207. package/dist/core/selector-base/base-generator.js.map +0 -1
  208. package/dist/core/selector-base/gherkin-parser.d.ts +0 -24
  209. package/dist/core/selector-base/gherkin-parser.d.ts.map +0 -1
  210. package/dist/core/selector-base/gherkin-parser.js +0 -42
  211. package/dist/core/selector-base/gherkin-parser.js.map +0 -1
  212. package/dist/core/selector-mapper/priority-mapper.d.ts +0 -74
  213. package/dist/core/selector-mapper/priority-mapper.d.ts.map +0 -1
  214. package/dist/core/selector-mapper/priority-mapper.js +0 -477
  215. package/dist/core/selector-mapper/priority-mapper.js.map +0 -1
  216. package/dist/core/ui-scanner/heuristics/base-heuristic.d.ts +0 -91
  217. package/dist/core/ui-scanner/heuristics/base-heuristic.d.ts.map +0 -1
  218. package/dist/core/ui-scanner/heuristics/base-heuristic.js +0 -175
  219. package/dist/core/ui-scanner/heuristics/base-heuristic.js.map +0 -1
  220. package/dist/core/ui-scanner/react-scanner.d.ts +0 -32
  221. package/dist/core/ui-scanner/react-scanner.d.ts.map +0 -1
  222. package/dist/core/ui-scanner/react-scanner.js +0 -163
  223. package/dist/core/ui-scanner/react-scanner.js.map +0 -1
  224. package/dist/core/ui-scanner/scanner-interface.d.ts +0 -94
  225. package/dist/core/ui-scanner/scanner-interface.d.ts.map +0 -1
  226. package/dist/core/ui-scanner/scanner-interface.js +0 -33
  227. package/dist/core/ui-scanner/scanner-interface.js.map +0 -1
  228. package/dist/core/ui-scanner/strict-scanner.d.ts +0 -81
  229. package/dist/core/ui-scanner/strict-scanner.d.ts.map +0 -1
  230. package/dist/core/ui-scanner/strict-scanner.js +0 -511
  231. package/dist/core/ui-scanner/strict-scanner.js.map +0 -1
  232. package/dist/core/validator/data-validator.d.ts +0 -38
  233. package/dist/core/validator/data-validator.d.ts.map +0 -1
  234. package/dist/core/validator/data-validator.js +0 -212
  235. package/dist/core/validator/data-validator.js.map +0 -1
  236. package/dist/core/validator/feature-validator.d.ts +0 -27
  237. package/dist/core/validator/feature-validator.d.ts.map +0 -1
  238. package/dist/core/validator/feature-validator.js +0 -182
  239. package/dist/core/validator/feature-validator.js.map +0 -1
  240. package/dist/core/validator/index.d.ts +0 -46
  241. package/dist/core/validator/index.d.ts.map +0 -1
  242. package/dist/core/validator/index.js +0 -17
  243. package/dist/core/validator/index.js.map +0 -1
  244. package/dist/core/validator/screen-validator.d.ts +0 -35
  245. package/dist/core/validator/screen-validator.d.ts.map +0 -1
  246. package/dist/core/validator/screen-validator.js +0 -195
  247. package/dist/core/validator/screen-validator.js.map +0 -1
  248. package/dist/core/validator/selector-validator.d.ts +0 -36
  249. package/dist/core/validator/selector-validator.d.ts.map +0 -1
  250. package/dist/core/validator/selector-validator.js +0 -210
  251. package/dist/core/validator/selector-validator.js.map +0 -1
  252. package/dist/external/ai-provider.d.ts +0 -60
  253. package/dist/external/ai-provider.d.ts.map +0 -1
  254. package/dist/external/ai-provider.js +0 -30
  255. package/dist/external/ai-provider.js.map +0 -1
  256. package/dist/external/anthropic-provider.d.ts +0 -29
  257. package/dist/external/anthropic-provider.d.ts.map +0 -1
  258. package/dist/external/anthropic-provider.js +0 -85
  259. package/dist/external/anthropic-provider.js.map +0 -1
  260. package/dist/generators/cache/cache-manager.d.ts +0 -66
  261. package/dist/generators/cache/cache-manager.d.ts.map +0 -1
  262. package/dist/generators/cache/cache-manager.js +0 -286
  263. package/dist/generators/cache/cache-manager.js.map +0 -1
  264. package/dist/generators/dsl-writer/index.d.ts +0 -33
  265. package/dist/generators/dsl-writer/index.d.ts.map +0 -1
  266. package/dist/generators/dsl-writer/index.js +0 -226
  267. package/dist/generators/dsl-writer/index.js.map +0 -1
  268. package/dist/generators/scaffold-generator/index.d.ts +0 -162
  269. package/dist/generators/scaffold-generator/index.d.ts.map +0 -1
  270. package/dist/generators/scaffold-generator/index.js +0 -877
  271. package/dist/generators/scaffold-generator/index.js.map +0 -1
  272. package/dist/generators/selector-mapper/ai-mapper.d.ts +0 -56
  273. package/dist/generators/selector-mapper/ai-mapper.d.ts.map +0 -1
  274. package/dist/generators/selector-mapper/ai-mapper.js +0 -457
  275. package/dist/generators/selector-mapper/ai-mapper.js.map +0 -1
  276. package/dist/generators/selector-mapper/hybrid-mapper.d.ts +0 -67
  277. package/dist/generators/selector-mapper/hybrid-mapper.d.ts.map +0 -1
  278. package/dist/generators/selector-mapper/hybrid-mapper.js +0 -349
  279. package/dist/generators/selector-mapper/hybrid-mapper.js.map +0 -1
  280. package/dist/generators/selector-mapper/index.d.ts +0 -8
  281. package/dist/generators/selector-mapper/index.d.ts.map +0 -1
  282. package/dist/generators/selector-mapper/index.js +0 -12
  283. package/dist/generators/selector-mapper/index.js.map +0 -1
  284. package/dist/generators/selector-mapper/intelligent-mapper.d.ts +0 -125
  285. package/dist/generators/selector-mapper/intelligent-mapper.d.ts.map +0 -1
  286. package/dist/generators/selector-mapper/intelligent-mapper.js +0 -391
  287. package/dist/generators/selector-mapper/intelligent-mapper.js.map +0 -1
  288. package/dist/generators/test-generator/ai-step-mapper.d.ts +0 -27
  289. package/dist/generators/test-generator/ai-step-mapper.d.ts.map +0 -1
  290. package/dist/generators/test-generator/ai-step-mapper.js +0 -204
  291. package/dist/generators/test-generator/ai-step-mapper.js.map +0 -1
  292. package/dist/generators/test-generator/auth-setup-generator.d.ts +0 -18
  293. package/dist/generators/test-generator/auth-setup-generator.d.ts.map +0 -1
  294. package/dist/generators/test-generator/auth-setup-generator.js +0 -82
  295. package/dist/generators/test-generator/auth-setup-generator.js.map +0 -1
  296. package/dist/generators/test-generator/patterns/legacy-patterns.d.ts +0 -7
  297. package/dist/generators/test-generator/patterns/legacy-patterns.d.ts.map +0 -1
  298. package/dist/generators/test-generator/patterns/legacy-patterns.js +0 -98
  299. package/dist/generators/test-generator/patterns/legacy-patterns.js.map +0 -1
  300. package/dist/generators/test-generator/templates/auth-setup.ts.hbs +0 -36
  301. package/dist/generators/ui-model-builder/deep-scanner.d.ts +0 -121
  302. package/dist/generators/ui-model-builder/deep-scanner.d.ts.map +0 -1
  303. package/dist/generators/ui-model-builder/deep-scanner.js +0 -1113
  304. package/dist/generators/ui-model-builder/deep-scanner.js.map +0 -1
  305. package/dist/generators/ui-model-builder/enhanced-deep-scanner.d.ts +0 -110
  306. package/dist/generators/ui-model-builder/enhanced-deep-scanner.d.ts.map +0 -1
  307. package/dist/generators/ui-model-builder/enhanced-deep-scanner.js +0 -608
  308. package/dist/generators/ui-model-builder/enhanced-deep-scanner.js.map +0 -1
  309. package/dist/generators/ui-model-builder/react-scanner.d.ts +0 -107
  310. package/dist/generators/ui-model-builder/react-scanner.d.ts.map +0 -1
  311. package/dist/generators/ui-model-builder/react-scanner.js +0 -797
  312. package/dist/generators/ui-model-builder/react-scanner.js.map +0 -1
  313. package/dist/orchestrator/cache-manager.d.ts +0 -15
  314. package/dist/orchestrator/cache-manager.d.ts.map +0 -1
  315. package/dist/orchestrator/cache-manager.js +0 -62
  316. package/dist/orchestrator/cache-manager.js.map +0 -1
  317. package/dist/orchestrator/pipeline.d.ts +0 -56
  318. package/dist/orchestrator/pipeline.d.ts.map +0 -1
  319. package/dist/orchestrator/pipeline.js +0 -298
  320. package/dist/orchestrator/pipeline.js.map +0 -1
  321. package/dist/orchestrator/reporter.d.ts +0 -15
  322. package/dist/orchestrator/reporter.d.ts.map +0 -1
  323. package/dist/orchestrator/reporter.js +0 -30
  324. package/dist/orchestrator/reporter.js.map +0 -1
  325. package/src/cli/commands/cache-clear.ts +0 -22
  326. package/src/cli/commands/full.ts +0 -35
  327. package/src/cli/commands/live-scan.ts +0 -82
  328. package/src/cli/commands/map.ts +0 -97
  329. package/src/cli/commands/validate.ts +0 -43
  330. package/src/cli/utils.ts +0 -106
  331. package/src/config/ai-providers.yaml +0 -56
  332. package/src/config/config-loader.ts +0 -248
  333. package/src/config/config-schema.ts +0 -148
  334. package/src/config/default.config.yaml +0 -107
  335. package/src/config/framework.config.yaml +0 -52
  336. package/src/config/routes.yaml +0 -31
  337. package/src/core/live-scanner/config-reader.ts +0 -57
  338. package/src/core/live-scanner/element-finder.ts +0 -534
  339. package/src/core/live-scanner/index.ts +0 -7
  340. package/src/core/live-scanner/matrix-reader.ts +0 -65
  341. package/src/core/live-scanner/matrix-writer.ts +0 -77
  342. package/src/core/live-scanner/role-fallback.ts +0 -44
  343. package/src/core/live-scanner/scanner.ts +0 -321
  344. package/src/core/live-scanner/step-replayer.ts +0 -503
  345. package/src/core/live-scanner/types.ts +0 -58
  346. package/src/core/selector-base/annotation-handler.ts +0 -127
  347. package/src/core/selector-base/base-generator.ts +0 -234
  348. package/src/core/selector-base/gherkin-parser.ts +0 -57
  349. package/src/core/selector-mapper/priority-mapper.ts +0 -607
  350. package/src/core/ui-scanner/heuristics/base-heuristic.ts +0 -216
  351. package/src/core/ui-scanner/react-scanner.ts +0 -156
  352. package/src/core/ui-scanner/scanner-interface.ts +0 -133
  353. package/src/core/ui-scanner/strict-scanner.ts +0 -629
  354. package/src/core/validator/data-validator.ts +0 -202
  355. package/src/core/validator/feature-validator.ts +0 -176
  356. package/src/core/validator/index.ts +0 -57
  357. package/src/core/validator/screen-validator.ts +0 -209
  358. package/src/core/validator/selector-validator.ts +0 -209
  359. package/src/external/ai-provider.ts +0 -90
  360. package/src/external/anthropic-provider.ts +0 -114
  361. package/src/generators/README.md +0 -410
  362. package/src/generators/cache/cache-manager.ts +0 -322
  363. package/src/generators/dsl-writer/index.ts +0 -253
  364. package/src/generators/scaffold-generator/index.ts +0 -1029
  365. package/src/generators/selector-mapper/ai-mapper.ts +0 -528
  366. package/src/generators/selector-mapper/hybrid-mapper.ts +0 -427
  367. package/src/generators/selector-mapper/index.ts +0 -10
  368. package/src/generators/selector-mapper/intelligent-mapper.ts +0 -530
  369. package/src/generators/test-generator/ai-step-mapper.ts +0 -224
  370. package/src/generators/test-generator/auth-setup-generator.ts +0 -59
  371. package/src/generators/test-generator/patterns/legacy-patterns.ts +0 -104
  372. package/src/generators/test-generator/templates/auth-setup.ts.hbs +0 -36
  373. package/src/generators/ui-model-builder/deep-scanner.ts +0 -1244
  374. package/src/generators/ui-model-builder/enhanced-deep-scanner.ts +0 -731
  375. package/src/generators/ui-model-builder/react-scanner.ts +0 -959
  376. package/src/orchestrator/cache-manager.ts +0 -32
  377. package/src/orchestrator/pipeline.ts +0 -354
  378. package/src/orchestrator/reporter.ts +0 -36
@@ -1,959 +0,0 @@
1
- /**
2
- * React/Next.js UI Model Builder
3
- * Scans React/Next.js source code to extract UI elements using heuristic analysis
4
- */
5
-
6
- import * as parser from '@babel/parser';
7
- import traverse, { NodePath } from '@babel/traverse';
8
- import * as t from '@babel/types';
9
- import * as fs from 'fs';
10
- import * as path from 'path';
11
- import { glob } from 'glob';
12
- import { UIModel, UIElement, FrameworkType } from '../types';
13
-
14
- // ============================================================================
15
- // Configuration
16
- // ============================================================================
17
-
18
- interface ScanConfig {
19
- sourceRoot: string;
20
- framework: FrameworkType;
21
- verbose?: boolean;
22
- maxDepth?: number; // Maximum depth for component scanning (default: 999 = unlimited)
23
- }
24
-
25
- // ============================================================================
26
- // React Scanner Class
27
- // ============================================================================
28
-
29
- interface ComponentInfo {
30
- name: string; // Component name (e.g., "ChatInput")
31
- importPath: string; // Import path (e.g., "@/components/ChatInput")
32
- sourceFile: string; // File where component was used
33
- }
34
-
35
- export class ReactScanner {
36
- private config: ScanConfig;
37
- private elements: UIElement[] = [];
38
- private elementCounter = 0;
39
- private detectedComponents: ComponentInfo[] = [];
40
- private scannedComponentPaths = new Set<string>(); // Avoid duplicate scans
41
-
42
- constructor(config: ScanConfig) {
43
- this.config = config;
44
- }
45
-
46
- /**
47
- * Discover all routes/screens in Next.js app directory
48
- */
49
- async discoverScreens(): Promise<string[]> {
50
- const appDir = this.config.sourceRoot;
51
-
52
- // Find all page.tsx/page.jsx files (Next.js 13+ app directory)
53
- const pageFiles = await glob('**/page.{tsx,jsx}', {
54
- cwd: appDir,
55
- absolute: false,
56
- ignore: ['**/node_modules/**', '**/api/**']
57
- });
58
-
59
- // Extract screen IDs from paths
60
- // Example: "login/page.tsx" -> "login"
61
- // Example: "auth/login/page.tsx" -> "auth-login"
62
- // Example: "page.tsx" -> "home"
63
- const screens = pageFiles.map(file => {
64
- const dir = path.dirname(file);
65
- return dir === '.' ? 'home' : dir.replace(/\//g, '-');
66
- });
67
-
68
- return screens;
69
- }
70
-
71
- /**
72
- * Build UI Model for a specific screen
73
- */
74
- async buildUIModel(screenId: string): Promise<UIModel> {
75
- // Reset state for new screen
76
- this.elements = [];
77
- this.elementCounter = 0;
78
- this.detectedComponents = [];
79
- this.scannedComponentPaths.clear();
80
-
81
- // Determine route path from screen ID
82
- // "home" -> "" (root)
83
- // "auth-login" -> "auth/login"
84
- const routePath = screenId === 'home' ? '' : screenId.replace(/-/g, '/');
85
- const routeDir = path.join(this.config.sourceRoot, routePath);
86
-
87
- // Collect source files to scan
88
- const sourceFiles: string[] = [];
89
-
90
- // 1. App root layout
91
- const rootLayout = path.join(this.config.sourceRoot, 'layout.tsx');
92
- if (fs.existsSync(rootLayout)) {
93
- sourceFiles.push(rootLayout);
94
- }
95
-
96
- // 2. Route-specific layout (if exists)
97
- const routeLayout = path.join(routeDir, 'layout.tsx');
98
- if (fs.existsSync(routeLayout)) {
99
- sourceFiles.push(routeLayout);
100
- }
101
-
102
- // 3. Page component
103
- const pageTsx = path.join(routeDir, 'page.tsx');
104
- const pageJsx = path.join(routeDir, 'page.jsx');
105
- if (fs.existsSync(pageTsx)) {
106
- sourceFiles.push(pageTsx);
107
- } else if (fs.existsSync(pageJsx)) {
108
- sourceFiles.push(pageJsx);
109
- } else {
110
- throw new Error(`Page component not found for screen: ${screenId} at ${routeDir}`);
111
- }
112
-
113
- // Phase 1: Scan all source files (page components)
114
- for (const filePath of sourceFiles) {
115
- await this.scanFile(filePath);
116
- }
117
-
118
- // Phase 2: Scan detected components (unlimited depth by default)
119
- const maxDepth = this.config.maxDepth !== undefined ? this.config.maxDepth : 999; // Default unlimited
120
- const depthLabel = maxDepth === 999 ? 'unlimited' : maxDepth.toString();
121
- if (this.config.verbose && this.detectedComponents.length > 0) {
122
- console.log(` Detected ${this.detectedComponents.length} component(s), scanning with depth=${depthLabel}...`);
123
- }
124
-
125
- await this.scanComponentsRecursive(this.detectedComponents, 1, maxDepth);
126
-
127
- // Build and return UI Model
128
- const uiModel: UIModel = {
129
- screenId,
130
- routeId: routePath || 'home',
131
- framework: this.config.framework,
132
- sourceFiles: sourceFiles.map(f => path.relative(this.config.sourceRoot, f)),
133
- elements: this.elements,
134
- metadata: {
135
- scannedAt: new Date().toISOString(),
136
- version: '1.0.0',
137
- hash: this.generateHash(this.elements)
138
- }
139
- };
140
-
141
- return uiModel;
142
- }
143
-
144
- /**
145
- * Scan a single file and extract UI elements
146
- */
147
- private async scanFile(filePath: string): Promise<void> {
148
- const sourceFileName = path.basename(filePath);
149
- const importMap = new Map<string, string>(); // componentName → importPath
150
-
151
- if (this.config.verbose) {
152
- console.log(` Scanning: ${filePath}`);
153
- }
154
-
155
- const code = fs.readFileSync(filePath, 'utf-8');
156
-
157
- // Parse with Babel
158
- const ast = parser.parse(code, {
159
- sourceType: 'module',
160
- plugins: ['typescript', 'jsx']
161
- });
162
-
163
- // Traverse AST to extract imports and JSX elements
164
- traverse(ast, {
165
- // Detect imports
166
- ImportDeclaration: (path: NodePath<t.ImportDeclaration>) => {
167
- const importPath = path.node.source.value;
168
-
169
- // Extract default import: import ChatInput from '...'
170
- const defaultSpecifier = path.node.specifiers.find(s => t.isImportDefaultSpecifier(s));
171
- if (defaultSpecifier && t.isImportDefaultSpecifier(defaultSpecifier)) {
172
- const componentName = defaultSpecifier.local.name;
173
- importMap.set(componentName, importPath);
174
- }
175
-
176
- // Extract named imports: import { Button } from '...'
177
- path.node.specifiers
178
- .filter(s => t.isImportSpecifier(s))
179
- .forEach(specifier => {
180
- if (t.isImportSpecifier(specifier) && t.isIdentifier(specifier.imported)) {
181
- const componentName = specifier.imported.name;
182
- importMap.set(componentName, importPath);
183
- }
184
- });
185
- },
186
-
187
- // Extract JSX elements
188
- JSXElement: (path: NodePath<t.JSXElement>) => {
189
- this.extractElement(path, sourceFileName, filePath, importMap);
190
- },
191
-
192
- // PHASE 1: Detect conditional rendering patterns
193
- JSXExpressionContainer: (path: NodePath<t.JSXExpressionContainer>) => {
194
- this.detectConditionalElements(path, sourceFileName, filePath, importMap);
195
- }
196
- });
197
- }
198
-
199
- /**
200
- * Smart element filter - determines if element should be scanned
201
- */
202
- private shouldScanElement(
203
- tagName: string,
204
- attributes: Record<string, string>,
205
- textContent: string | null,
206
- path?: NodePath<t.JSXElement>
207
- ): boolean {
208
- const lowercaseTag = tagName.toLowerCase();
209
-
210
- // 1. Always scan form elements
211
- const formElements = ['input', 'button', 'textarea', 'select', 'form'];
212
- if (formElements.includes(lowercaseTag)) return true;
213
-
214
- // 2. Always scan links (a, Link)
215
- if (lowercaseTag === 'a' || tagName === 'Link') return true;
216
-
217
- // 3. Scan headings with text content
218
- if (/^h[1-6]$/.test(lowercaseTag) && textContent) return true;
219
-
220
- // 4. Scan images with alt text
221
- if (lowercaseTag === 'img' && attributes['alt']) return true;
222
-
223
- // 5. Scan elements with identifying attributes
224
- const hasIdentifier =
225
- attributes['id'] ||
226
- attributes['name'] ||
227
- attributes['data-testid'] ||
228
- attributes['data-test-id'] ||
229
- attributes['aria-label'] ||
230
- attributes['ariaLabel'];
231
- if (hasIdentifier) return true;
232
-
233
- // 6. Scan semantic components (ChatInput, SubmitButton, UserProfile, etc.)
234
- // Pattern: *Input, *Field, *TextArea, *Select, *Button, *Link, *Form
235
- const semanticPatterns = [
236
- /Input$/, // ChatInput, MessageInput
237
- /Field$/, // TextField, EmailField
238
- /TextArea$/, // MessageTextArea
239
- /Select$/, // CountrySelect
240
- /Button$/, // SubmitButton, CancelButton
241
- /Link$/, // NavLink
242
- /Form$/, // LoginForm
243
- /Avatar$/, // UserAvatar
244
- /Icon$/, // SearchIcon
245
- /Image$/, // ProfileImage
246
- /Card$/, // UserCard
247
- /List$/, // MessageList
248
- /Item$/ // ListItem
249
- ];
250
-
251
- if (semanticPatterns.some(pattern => pattern.test(tagName))) {
252
- return true;
253
- }
254
-
255
- // 7. Scan text containers with content
256
- const textContainers = ['p', 'span', 'div', 'section', 'article'];
257
- if (textContainers.includes(lowercaseTag) && textContent && textContent.length > 0) {
258
- return true;
259
- }
260
-
261
- // 7.5. PHASE 1.5: Scan error/success message elements by className pattern (even without text)
262
- const className = attributes['className'] || '';
263
- const isErrorOrSuccessMessage =
264
- className.includes('text-red') ||
265
- className.includes('text-green') ||
266
- className.includes('bg-red') ||
267
- className.includes('bg-green') ||
268
- className.includes('error') ||
269
- className.includes('success') ||
270
- className.includes('alert');
271
-
272
- if (textContainers.includes(lowercaseTag) && isErrorOrSuccessMessage) {
273
- return true;
274
- }
275
-
276
- // 7.6. PHASE 1.6: Scan timestamp/metadata elements by className pattern
277
- // Fix for elements with dynamic expressions like {timestamp.toLocaleTimeString()}
278
- const isTimestampOrMetadata =
279
- className.includes('timestamp') ||
280
- className.includes('time') ||
281
- (className.includes('text-xs') && className.includes('text-gray')) ||
282
- (className.includes('text-sm') && className.includes('text-gray') && className.includes('mt-'));
283
-
284
- if (textContainers.includes(lowercaseTag) && isTimestampOrMetadata) {
285
- return true;
286
- }
287
-
288
- // 8. NEW: Smart container detection for structural elements
289
- if (['div', 'section', 'article', 'main', 'aside', 'nav'].includes(lowercaseTag)) {
290
- if (path && this.shouldScanContainer(tagName, attributes, path)) {
291
- return true;
292
- }
293
- }
294
-
295
- return false; // Skip everything else
296
- }
297
-
298
- /**
299
- * Smart container detection - checks if container is structurally important
300
- * NEW: Phase 1 - Task 1
301
- */
302
- private shouldScanContainer(
303
- tagName: string,
304
- attributes: Record<string, string>,
305
- path: NodePath<t.JSXElement>
306
- ): boolean {
307
-
308
- // 1. Has important children (form elements, buttons, custom components)?
309
- const children = path.node.children.filter(child => t.isJSXElement(child)) as t.JSXElement[];
310
-
311
- for (const child of children) {
312
- const childTag = this.getTagName(child.openingElement.name);
313
- const childTagLower = childTag.toLowerCase();
314
-
315
- // Has form elements
316
- if (['input', 'button', 'textarea', 'select', 'form'].includes(childTagLower)) {
317
- return true;
318
- }
319
-
320
- // Has custom components (PascalCase)
321
- if (/^[A-Z]/.test(childTag)) {
322
- return true;
323
- }
324
- }
325
-
326
- // 2. Semantic className keywords
327
- const className = attributes['className'] || '';
328
- const semanticKeywords = [
329
- 'message', 'messages', 'chat',
330
- 'list', 'items',
331
- 'container', 'wrapper', 'content',
332
- 'scroll', 'overflow',
333
- 'grid', 'flex'
334
- ];
335
-
336
- const hasSemanticClass = semanticKeywords.some(keyword =>
337
- className.toLowerCase().includes(keyword)
338
- );
339
- if (hasSemanticClass) {
340
- return true;
341
- }
342
-
343
- // 3. Has scroll behavior (overflow-auto, overflow-y-auto, etc.)
344
- if (className.includes('overflow')) {
345
- return true;
346
- }
347
-
348
- // 4. Has ARIA role
349
- if (attributes['role']) {
350
- return true;
351
- }
352
-
353
- // 5. Has ref (often used for important containers)
354
- if (attributes['ref']) {
355
- return true;
356
- }
357
-
358
- return false;
359
- }
360
-
361
- /**
362
- * Extract UI element from JSX node
363
- */
364
- private extractElement(
365
- path: NodePath<t.JSXElement>,
366
- sourceFile: string,
367
- fullFilePath: string,
368
- importMap: Map<string, string>
369
- ): void {
370
- const openingElement = path.node.openingElement;
371
- const tagName = this.getTagName(openingElement.name);
372
-
373
- // Extract attributes and text first (needed for filtering)
374
- const attributes = this.extractAttributes(openingElement.attributes);
375
- const textContent = this.extractTextContent(path.node);
376
-
377
- // Check if this is a custom component (starts with uppercase)
378
- const isCustomComponent = /^[A-Z]/.test(tagName);
379
-
380
- // If it's a custom component and we have an import for it, track it for scanning
381
- // UPDATED: Phase 1 - Task 2 - Track ALL PascalCase components, not just semantic patterns
382
- if (isCustomComponent && importMap.has(tagName)) {
383
- const importPath = importMap.get(tagName)!;
384
-
385
- // Check if not already tracked
386
- const alreadyTracked = this.detectedComponents.some(c => c.name === tagName && c.importPath === importPath);
387
- if (!alreadyTracked) {
388
- this.detectedComponents.push({
389
- name: tagName,
390
- importPath,
391
- sourceFile: fullFilePath
392
- });
393
-
394
- if (this.config.verbose) {
395
- console.log(` Found component: ${tagName} from ${importPath}`);
396
- }
397
- }
398
- }
399
-
400
- // Use smart filter instead of static whitelist (pass path for container detection)
401
- if (!this.shouldScanElement(tagName, attributes, textContent, path)) {
402
- return; // Skip non-scannable elements
403
- }
404
-
405
- // Build UI Element
406
- const element: UIElement = {
407
- key: `e${++this.elementCounter}`,
408
- tag: tagName.toLowerCase(),
409
- role: attributes['role'] || this.inferRole(tagName),
410
- name: attributes['name'],
411
- id: attributes['id'],
412
- placeholder: attributes['placeholder'],
413
- ariaLabel: attributes['aria-label'] || attributes['ariaLabel'],
414
- testId: attributes['data-testid'] || attributes['data-test-id'],
415
- text: textContent || attributes['value'],
416
- source: sourceFile,
417
- props: attributes
418
- };
419
-
420
- // Add label if associated
421
- if (attributes['aria-labelledby']) {
422
- element.label = `[ref:${attributes['aria-labelledby']}]`;
423
- }
424
-
425
- this.elements.push(element);
426
- }
427
-
428
- /**
429
- * PHASE 1: Detect and extract conditionally rendered elements
430
- * Handles patterns: {condition && <Element />}, {condition ? <A /> : <B />}, {arr.map(...)}
431
- * PHASE 1.5: Now also traverses nested children
432
- */
433
- private detectConditionalElements(
434
- path: NodePath<t.JSXExpressionContainer>,
435
- sourceFile: string,
436
- fullFilePath: string,
437
- importMap: Map<string, string>
438
- ): void {
439
- const expr = path.node.expression;
440
-
441
- // Pattern 1: Logical AND (condition && <Element />)
442
- if (t.isLogicalExpression(expr, { operator: '&&' })) {
443
- const rightSide = expr.right;
444
-
445
- if (t.isJSXElement(rightSide)) {
446
- // Extract the direct element
447
- this.extractConditionalElement(rightSide, sourceFile, fullFilePath, importMap, 'conditional-and');
448
-
449
- // PHASE 1.5: Also traverse nested children
450
- this.traverseConditionalChildren(rightSide, sourceFile, fullFilePath, importMap, 'conditional-nested');
451
- } else if (t.isJSXFragment(rightSide)) {
452
- // Handle {condition && (<><Element1 /><Element2 /></>)}
453
- this.extractFragmentElements(rightSide, sourceFile, fullFilePath, importMap, 'conditional-and');
454
- }
455
- }
456
-
457
- // Pattern 2: Ternary operator (condition ? <A /> : <B />)
458
- if (t.isConditionalExpression(expr)) {
459
- const consequent = expr.consequent;
460
- const alternate = expr.alternate;
461
-
462
- if (t.isJSXElement(consequent)) {
463
- this.extractConditionalElement(consequent, sourceFile, fullFilePath, importMap, 'conditional-true');
464
- // PHASE 1.5: Traverse nested children
465
- this.traverseConditionalChildren(consequent, sourceFile, fullFilePath, importMap, 'conditional-nested');
466
- }
467
-
468
- if (t.isJSXElement(alternate)) {
469
- this.extractConditionalElement(alternate, sourceFile, fullFilePath, importMap, 'conditional-false');
470
- // PHASE 1.5: Traverse nested children
471
- this.traverseConditionalChildren(alternate, sourceFile, fullFilePath, importMap, 'conditional-nested');
472
- }
473
- }
474
-
475
- // Pattern 3: Array.map() rendering (arr.map(item => <Element />))
476
- if (t.isCallExpression(expr)) {
477
- const callee = expr.callee;
478
-
479
- // Check if it's a .map() call
480
- if (t.isMemberExpression(callee) && t.isIdentifier(callee.property, { name: 'map' })) {
481
- const mapCallback = expr.arguments[0];
482
-
483
- if (t.isArrowFunctionExpression(mapCallback) || t.isFunctionExpression(mapCallback)) {
484
- const body = mapCallback.body;
485
-
486
- // Case 1: Direct return - arr.map(item => <Element />)
487
- if (t.isJSXElement(body)) {
488
- this.extractConditionalElement(body, sourceFile, fullFilePath, importMap, 'loop-item');
489
- }
490
-
491
- // Case 2: Block with return - arr.map(item => { return <Element /> })
492
- if (t.isBlockStatement(body)) {
493
- for (const statement of body.body) {
494
- if (t.isReturnStatement(statement) && statement.argument && t.isJSXElement(statement.argument)) {
495
- this.extractConditionalElement(statement.argument, sourceFile, fullFilePath, importMap, 'loop-item');
496
- }
497
- }
498
- }
499
- }
500
- }
501
- }
502
- }
503
-
504
- /**
505
- * Extract elements from JSX Fragment
506
- */
507
- private extractFragmentElements(
508
- fragment: t.JSXFragment,
509
- sourceFile: string,
510
- fullFilePath: string,
511
- importMap: Map<string, string>,
512
- renderType: string
513
- ): void {
514
- for (const child of fragment.children) {
515
- if (t.isJSXElement(child)) {
516
- this.extractConditionalElement(child, sourceFile, fullFilePath, importMap, renderType);
517
- }
518
- }
519
- }
520
-
521
- /**
522
- * PHASE 1.5: Traverse nested children of conditional elements
523
- * Example: {error && <div><p>{error}</p></div>} - detect the nested <p> tag
524
- */
525
- private traverseConditionalChildren(
526
- element: t.JSXElement,
527
- sourceFile: string,
528
- fullFilePath: string,
529
- importMap: Map<string, string>,
530
- renderType: string
531
- ): void {
532
- for (const child of element.children) {
533
- // Traverse nested JSX elements
534
- if (t.isJSXElement(child)) {
535
- const openingElement = child.openingElement;
536
- const tagName = this.getTagName(openingElement.name);
537
- const attributes = this.extractAttributes(openingElement.attributes);
538
- const textContent = this.extractTextContent(child);
539
-
540
- // Check if this nested element should be scanned
541
- if (this.shouldScanElement(tagName, attributes, textContent, undefined)) {
542
- this.extractConditionalElement(child, sourceFile, fullFilePath, importMap, renderType);
543
- }
544
-
545
- // Recursively traverse deeper children
546
- this.traverseConditionalChildren(child, sourceFile, fullFilePath, importMap, renderType);
547
- }
548
-
549
- // Traverse JSX expression containers (for nested dynamic content)
550
- if (t.isJSXExpressionContainer(child)) {
551
- const expr = child.expression;
552
-
553
- // Handle nested JSX elements in expressions
554
- if (t.isJSXElement(expr)) {
555
- this.extractConditionalElement(expr, sourceFile, fullFilePath, importMap, renderType);
556
- }
557
- }
558
- }
559
- }
560
-
561
- /**
562
- * Extract a conditionally rendered JSX element
563
- */
564
- private extractConditionalElement(
565
- jsxElement: t.JSXElement,
566
- sourceFile: string,
567
- fullFilePath: string,
568
- importMap: Map<string, string>,
569
- renderType: string
570
- ): void {
571
- const openingElement = jsxElement.openingElement;
572
- const tagName = this.getTagName(openingElement.name);
573
-
574
- // Extract attributes and text
575
- const attributes = this.extractAttributes(openingElement.attributes);
576
- const textContent = this.extractTextContent(jsxElement);
577
-
578
- // Check if this is a custom component
579
- const isCustomComponent = /^[A-Z]/.test(tagName);
580
- if (isCustomComponent && importMap.has(tagName)) {
581
- const importPath = importMap.get(tagName)!;
582
- const alreadyTracked = this.detectedComponents.some(c => c.name === tagName && c.importPath === importPath);
583
- if (!alreadyTracked) {
584
- this.detectedComponents.push({
585
- name: tagName,
586
- importPath,
587
- sourceFile: fullFilePath
588
- });
589
- }
590
- }
591
-
592
- // Use smart filter to decide if we should scan this element
593
- if (!this.shouldScanElement(tagName, attributes, textContent, undefined)) {
594
- return;
595
- }
596
-
597
- // Build UI Element with conditional marker
598
- const element: UIElement = {
599
- key: `e${++this.elementCounter}`,
600
- tag: tagName.toLowerCase(),
601
- role: attributes['role'] || this.inferRole(tagName),
602
- name: attributes['name'],
603
- id: attributes['id'],
604
- placeholder: attributes['placeholder'],
605
- ariaLabel: attributes['aria-label'] || attributes['ariaLabel'],
606
- testId: attributes['data-testid'] || attributes['data-test-id'],
607
- text: textContent || attributes['value'],
608
- source: sourceFile,
609
- props: {
610
- ...attributes,
611
- isConditional: renderType // Mark as conditional with type
612
- }
613
- };
614
-
615
- if (attributes['aria-labelledby']) {
616
- element.label = `[ref:${attributes['aria-labelledby']}]`;
617
- }
618
-
619
- this.elements.push(element);
620
-
621
- if (this.config.verbose) {
622
- console.log(` Found conditional element: ${tagName} (${renderType})`);
623
- }
624
- }
625
-
626
- /**
627
- * Get tag name from JSX element
628
- */
629
- private getTagName(name: t.JSXElement['openingElement']['name']): string {
630
- if (t.isJSXIdentifier(name)) {
631
- return name.name;
632
- }
633
- if (t.isJSXMemberExpression(name)) {
634
- // Handle cases like <Form.Input />
635
- return this.getTagName(name.property);
636
- }
637
- return 'unknown';
638
- }
639
-
640
- /**
641
- * Extract attributes from JSX element
642
- */
643
- private extractAttributes(attributes: (t.JSXAttribute | t.JSXSpreadAttribute)[]): Record<string, string> {
644
- const result: Record<string, string> = {};
645
-
646
- for (const attr of attributes) {
647
- if (t.isJSXAttribute(attr) && t.isJSXIdentifier(attr.name)) {
648
- const name = attr.name.name;
649
- const value = this.getAttributeValue(attr.value);
650
- if (value !== null) {
651
- result[name] = value;
652
- }
653
- }
654
- }
655
-
656
- return result;
657
- }
658
-
659
- /**
660
- * Get attribute value
661
- */
662
- private getAttributeValue(value: t.JSXAttribute['value']): string | null {
663
- if (!value) return null;
664
-
665
- if (t.isStringLiteral(value)) {
666
- return value.value;
667
- }
668
-
669
- if (t.isJSXExpressionContainer(value)) {
670
- const expr = value.expression;
671
-
672
- // Handle string literals in expressions
673
- if (t.isStringLiteral(expr)) {
674
- return expr.value;
675
- }
676
-
677
- // Handle template literals
678
- if (t.isTemplateLiteral(expr) && expr.quasis.length === 1) {
679
- return expr.quasis[0].value.cooked || '';
680
- }
681
-
682
- // For complex expressions, return placeholder
683
- return '[expression]';
684
- }
685
-
686
- return null;
687
- }
688
-
689
- /**
690
- * Extract text content from JSX element
691
- * PHASE 1 ENHANCED: Extract text from dynamic expressions
692
- */
693
- private extractTextContent(node: t.JSXElement): string | null {
694
- const children = node.children;
695
- const textParts: string[] = [];
696
-
697
- for (const child of children) {
698
- if (t.isJSXText(child)) {
699
- const text = child.value.trim();
700
- if (text) textParts.push(text);
701
- }
702
-
703
- if (t.isJSXExpressionContainer(child)) {
704
- const expr = child.expression;
705
-
706
- // Static string literal
707
- if (t.isStringLiteral(expr)) {
708
- textParts.push(expr.value);
709
- }
710
-
711
- // PHASE 1: Extract text from ternary (condition ? 'Text A' : 'Text B')
712
- if (t.isConditionalExpression(expr)) {
713
- const texts: string[] = [];
714
-
715
- if (t.isStringLiteral(expr.consequent)) {
716
- texts.push(expr.consequent.value);
717
- }
718
-
719
- if (t.isStringLiteral(expr.alternate)) {
720
- texts.push(expr.alternate.value);
721
- }
722
-
723
- if (texts.length > 0) {
724
- textParts.push(texts.join(' / ')); // e.g., "Copy / Copied!"
725
- }
726
- }
727
-
728
- // PHASE 1: Extract text from logical OR (defaultValue || 'Fallback')
729
- if (t.isLogicalExpression(expr, { operator: '||' })) {
730
- if (t.isStringLiteral(expr.right)) {
731
- textParts.push(expr.right.value);
732
- }
733
- }
734
- }
735
- }
736
-
737
- return textParts.length > 0 ? textParts.join(' ') : null;
738
- }
739
-
740
- /**
741
- * Infer ARIA role from tag name
742
- */
743
- private inferRole(tagName: string): string | undefined {
744
- const roleMap: Record<string, string> = {
745
- 'button': 'button',
746
- 'input': 'textbox',
747
- 'textarea': 'textbox',
748
- 'select': 'combobox',
749
- 'a': 'link',
750
- 'form': 'form'
751
- };
752
-
753
- return roleMap[tagName.toLowerCase()];
754
- }
755
-
756
- /**
757
- * Resolve component import path to actual file path
758
- */
759
- private resolveComponentPath(importPath: string, sourceFile: string): string | null {
760
- try {
761
- // Handle different import path formats
762
- let resolvedPath: string;
763
-
764
- if (importPath.startsWith('@/')) {
765
- // Alias import: @/components/ChatInput → PROJECT_ROOT/components/ChatInput.tsx
766
- // Note: @ alias points to project root, not sourceRoot (app/)
767
- const projectRoot = path.resolve(this.config.sourceRoot, '..');
768
- const relativePath = importPath.replace('@/', '');
769
- resolvedPath = path.join(projectRoot, relativePath);
770
- } else if (importPath.startsWith('./') || importPath.startsWith('../')) {
771
- // Relative import: ./components/ChatInput
772
- const sourceDir = path.dirname(sourceFile);
773
- resolvedPath = path.join(sourceDir, importPath);
774
- } else {
775
- // Absolute or node_modules import - skip
776
- return null;
777
- }
778
-
779
- // Try different file extensions
780
- const extensions = ['.tsx', '.ts', '.jsx', '.js'];
781
- for (const ext of extensions) {
782
- const fullPath = resolvedPath + ext;
783
- if (fs.existsSync(fullPath)) {
784
- return fullPath;
785
- }
786
- }
787
-
788
- // Try index files
789
- for (const ext of extensions) {
790
- const indexPath = path.join(resolvedPath, `index${ext}`);
791
- if (fs.existsSync(indexPath)) {
792
- return indexPath;
793
- }
794
- }
795
-
796
- return null;
797
- } catch (error) {
798
- if (this.config.verbose) {
799
- console.log(` Could not resolve: ${importPath}`);
800
- }
801
- return null;
802
- }
803
- }
804
-
805
- /**
806
- * NEW: Recursively scan components at multiple depths
807
- * @param components - List of components to scan
808
- * @param currentDepth - Current depth level
809
- * @param maxDepth - Maximum depth to scan
810
- */
811
- private async scanComponentsRecursive(
812
- components: ComponentInfo[],
813
- currentDepth: number,
814
- maxDepth: number
815
- ): Promise<void> {
816
- if (currentDepth > maxDepth) {
817
- return; // Stop recursion at max depth
818
- }
819
-
820
- const startTime = Date.now();
821
- const componentsScanned: ComponentInfo[] = [];
822
-
823
- // Scan all components at current level
824
- for (const component of components) {
825
- const componentPath = this.resolveComponentPath(component.importPath, component.sourceFile);
826
-
827
- if (!componentPath) {
828
- continue; // Skip if path cannot be resolved
829
- }
830
-
831
- // Skip node_modules
832
- if (componentPath.includes('node_modules')) {
833
- if (this.config.verbose) {
834
- console.log(` Skipping node_modules: ${componentPath}`);
835
- }
836
- continue;
837
- }
838
-
839
- // Skip if already scanned (avoid duplicates)
840
- if (this.scannedComponentPaths.has(componentPath)) {
841
- continue;
842
- }
843
-
844
- // Mark as scanned
845
- this.scannedComponentPaths.add(componentPath);
846
-
847
- // Store the count of detected components before scanning
848
- const detectedCountBefore = this.detectedComponents.length;
849
-
850
- // Scan the component file
851
- await this.scanComponentFile(componentPath, component.name);
852
-
853
- // Track newly detected components from this scan
854
- const newlyDetected = this.detectedComponents.slice(detectedCountBefore);
855
- componentsScanned.push(...newlyDetected);
856
- }
857
-
858
- const elapsed = Date.now() - startTime;
859
- if (this.config.verbose && componentsScanned.length > 0) {
860
- console.log(` [Depth ${currentDepth}] Scanned ${components.length} component(s), found ${componentsScanned.length} sub-component(s) (${elapsed}ms)`);
861
- }
862
-
863
- // Recursively scan newly detected components at next depth
864
- if (componentsScanned.length > 0 && currentDepth < maxDepth) {
865
- await this.scanComponentsRecursive(componentsScanned, currentDepth + 1, maxDepth);
866
- }
867
- }
868
-
869
- /**
870
- * Scan component file and extract elements
871
- */
872
- private async scanComponentFile(componentPath: string, componentName: string): Promise<void> {
873
- if (this.config.verbose) {
874
- console.log(` Scanning component: ${componentName} at ${componentPath}`);
875
- }
876
-
877
- const code = fs.readFileSync(componentPath, 'utf-8');
878
-
879
- // Parse with Babel
880
- const ast = parser.parse(code, {
881
- sourceType: 'module',
882
- plugins: ['typescript', 'jsx']
883
- });
884
-
885
- const importMap = new Map<string, string>(); // Empty import map for component scanning
886
-
887
- // Traverse AST and extract JSX elements (no nested component scanning)
888
- traverse(ast, {
889
- JSXElement: (path: NodePath<t.JSXElement>) => {
890
- const openingElement = path.node.openingElement;
891
- const tagName = this.getTagName(openingElement.name);
892
-
893
- // Extract attributes and text
894
- const attributes = this.extractAttributes(openingElement.attributes);
895
- const textContent = this.extractTextContent(path.node);
896
-
897
- // Use smart filter (pass path for container detection)
898
- if (!this.shouldScanElement(tagName, attributes, textContent, path)) {
899
- return;
900
- }
901
-
902
- // Build UI Element with component context
903
- const element: UIElement = {
904
- key: `e${++this.elementCounter}`,
905
- tag: tagName.toLowerCase(),
906
- role: attributes['role'] || this.inferRole(tagName),
907
- name: attributes['name'],
908
- id: attributes['id'],
909
- placeholder: attributes['placeholder'],
910
- ariaLabel: attributes['aria-label'] || attributes['ariaLabel'],
911
- testId: attributes['data-testid'] || attributes['data-test-id'],
912
- text: textContent || attributes['value'],
913
- source: `${componentName}.tsx`, // Mark as from component
914
- props: attributes
915
- };
916
-
917
- // Add label if associated
918
- if (attributes['aria-labelledby']) {
919
- element.label = `[ref:${attributes['aria-labelledby']}]`;
920
- }
921
-
922
- this.elements.push(element);
923
- },
924
-
925
- // PHASE 1: Detect conditional rendering in components too
926
- JSXExpressionContainer: (path: NodePath<t.JSXExpressionContainer>) => {
927
- this.detectConditionalElements(path, `${componentName}.tsx`, componentPath, importMap);
928
- }
929
- });
930
- }
931
-
932
- /**
933
- * Generate hash for cache key
934
- */
935
- private generateHash(elements: UIElement[]): string {
936
- const crypto = require('crypto');
937
- const content = JSON.stringify(elements);
938
- return crypto.createHash('sha256').update(content).digest('hex');
939
- }
940
- }
941
-
942
- // ============================================================================
943
- // Factory Function
944
- // ============================================================================
945
-
946
- export async function buildUIModel(
947
- screenId: string,
948
- config: ScanConfig
949
- ): Promise<UIModel> {
950
- const scanner = new ReactScanner(config);
951
- return scanner.buildUIModel(screenId);
952
- }
953
-
954
- export async function discoverScreens(
955
- config: ScanConfig
956
- ): Promise<string[]> {
957
- const scanner = new ReactScanner(config);
958
- return scanner.discoverScreens();
959
- }