create-appraisejs 0.3.1-alpha.0 → 0.3.1-alpha.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 (503) hide show
  1. package/README.md +17 -17
  2. package/dist/cli.e2e.test.d.ts +2 -2
  3. package/dist/cli.e2e.test.js +74 -72
  4. package/dist/config.d.ts.map +1 -1
  5. package/dist/config.js.map +1 -1
  6. package/dist/config.test.d.ts +2 -2
  7. package/dist/config.test.js +64 -64
  8. package/dist/copy-template.test.d.ts +2 -2
  9. package/dist/copy-template.test.js +70 -70
  10. package/dist/download-repo.test.d.ts +2 -2
  11. package/dist/download-repo.test.js +15 -13
  12. package/dist/install.test.d.ts +2 -2
  13. package/dist/install.test.js +115 -114
  14. package/dist/prepare-template-utils.d.ts.map +1 -1
  15. package/dist/prepare-template-utils.js.map +1 -1
  16. package/dist/prompts.d.ts.map +1 -1
  17. package/dist/prompts.js +2 -2
  18. package/dist/prompts.js.map +1 -1
  19. package/dist/prompts.test.d.ts +2 -2
  20. package/dist/prompts.test.js +54 -54
  21. package/dist/sync-templates-utils.d.ts.map +1 -1
  22. package/dist/sync-templates-utils.js +1 -5
  23. package/dist/sync-templates-utils.js.map +1 -1
  24. package/package.json +1 -1
  25. package/templates/blank/.editorconfig +11 -0
  26. package/templates/blank/.gitattributes +21 -0
  27. package/templates/blank/.gitconfig.appraise +14 -0
  28. package/templates/blank/.prettierrc +13 -0
  29. package/templates/blank/.vscode/settings.json +2 -0
  30. package/templates/blank/README.md +12 -12
  31. package/templates/blank/components.json +24 -24
  32. package/templates/blank/cucumber.mjs +16 -16
  33. package/templates/blank/e2e/README.md +56 -0
  34. package/templates/blank/e2e/apply-migrations.mjs +76 -0
  35. package/templates/blank/e2e/appraise-smoke.spec.ts +78 -0
  36. package/templates/blank/e2e/authoring.spec.ts +53 -0
  37. package/templates/blank/e2e/crud-configuration.spec.ts +138 -0
  38. package/templates/blank/e2e/crud-tests.spec.ts +76 -0
  39. package/templates/blank/e2e/helpers/forms.ts +186 -0
  40. package/templates/blank/e2e/helpers/navigation.ts +27 -0
  41. package/templates/blank/e2e/helpers/table.ts +45 -0
  42. package/templates/blank/e2e/helpers/test-data.ts +549 -0
  43. package/templates/blank/e2e/helpers/ui.ts +41 -0
  44. package/templates/blank/e2e/navigation.spec.ts +101 -0
  45. package/templates/blank/e2e/runs-and-reports.spec.ts +82 -0
  46. package/templates/blank/e2e/settings-sync.spec.ts +39 -0
  47. package/templates/blank/e2e/start-server.mjs +61 -0
  48. package/templates/blank/eslint.config.mjs +4 -4
  49. package/templates/blank/gitignore +0 -3
  50. package/templates/blank/next-env.d.ts +1 -1
  51. package/templates/blank/package-lock.json +58 -102
  52. package/templates/blank/package.json +10 -9
  53. package/templates/blank/packages/cucumber-runtime/package.json +13 -13
  54. package/templates/blank/packages/cucumber-runtime/src/cache.util.ts +93 -93
  55. package/templates/blank/packages/cucumber-runtime/src/cli.ts +77 -68
  56. package/templates/blank/packages/cucumber-runtime/src/environment.util.ts +21 -21
  57. package/templates/blank/packages/cucumber-runtime/src/executor.ts +32 -32
  58. package/templates/blank/packages/cucumber-runtime/src/index.ts +17 -17
  59. package/templates/blank/packages/cucumber-runtime/src/locator.util.ts +234 -234
  60. package/templates/blank/packages/cucumber-runtime/src/parameter-types.ts +7 -7
  61. package/templates/blank/packages/cucumber-runtime/src/random-data.util.ts +35 -35
  62. package/templates/blank/packages/cucumber-runtime/src/types.ts +13 -13
  63. package/templates/blank/packages/cucumber-runtime/src/world.ts +44 -44
  64. package/templates/blank/packages/cucumber-runtime/tsconfig.json +11 -11
  65. package/templates/blank/packages/locator-picker-companion/dist/cli.js +1 -1
  66. package/templates/blank/packages/locator-picker-companion/dist/index.d.ts +3 -3
  67. package/templates/blank/packages/locator-picker-companion/dist/index.js +3 -3
  68. package/templates/blank/packages/locator-picker-companion/dist/injected-picker-script.js +13 -3
  69. package/templates/blank/packages/locator-picker-companion/dist/selector-generator.d.ts +6 -3
  70. package/templates/blank/packages/locator-picker-companion/dist/selector-generator.js +233 -229
  71. package/templates/blank/packages/locator-picker-companion/src/cli.ts +1 -7
  72. package/templates/blank/packages/locator-picker-companion/src/injected-picker-script.ts +26 -5
  73. package/templates/blank/packages/locator-picker-companion/src/launcher.ts +1 -3
  74. package/templates/blank/packages/locator-picker-companion/src/session-file.ts +4 -15
  75. package/templates/blank/packages/locator-picker-companion/src/types.ts +1 -8
  76. package/templates/blank/playwright.config.ts +46 -0
  77. package/templates/blank/postcss.config.mjs +8 -8
  78. package/templates/blank/prisma/dev.db +0 -0
  79. package/templates/blank/scripts/configure-git-line-endings.mjs +91 -0
  80. package/templates/blank/scripts/install-template-step.ts +5 -1
  81. package/templates/blank/scripts/lib/step-matcher.test.ts +3 -3
  82. package/templates/blank/scripts/lib/step-matcher.ts +5 -1
  83. package/templates/blank/scripts/lib/template-step-installer.test.ts +4 -2
  84. package/templates/blank/scripts/lib/template-step-installer.ts +16 -4
  85. package/templates/blank/scripts/lib/template-step-registry.test.ts +1 -1
  86. package/templates/blank/scripts/lib/template-step-registry.ts +4 -1
  87. package/templates/blank/scripts/run-fallow-commit.mjs +4 -6
  88. package/templates/blank/scripts/sync-all.ts +7 -6
  89. package/templates/blank/scripts/sync-appraise-base-template.ts +84 -77
  90. package/templates/blank/scripts/sync-environments.ts +47 -44
  91. package/templates/blank/scripts/sync-locator-groups.ts +67 -76
  92. package/templates/blank/scripts/sync-locators.ts +50 -53
  93. package/templates/blank/scripts/sync-modules.ts +42 -42
  94. package/templates/blank/scripts/sync-tags.ts +51 -51
  95. package/templates/blank/scripts/sync-template-step-groups.ts +34 -32
  96. package/templates/blank/scripts/sync-template-steps.ts +97 -97
  97. package/templates/blank/scripts/sync-test-cases.ts +65 -74
  98. package/templates/blank/scripts/sync-test-suites.ts +85 -80
  99. package/templates/blank/src/actions/server-action-boundary.test.ts +1 -7
  100. package/templates/blank/src/actions/settings/sync-actions.ts +3 -1
  101. package/templates/blank/src/actions/tags/tag-actions.ts +1 -7
  102. package/templates/blank/src/actions/template-test-case/template-test-case-actions.ts +3 -1
  103. package/templates/blank/src/app/(base)/environments/create/page.tsx +28 -28
  104. package/templates/blank/src/app/(base)/environments/environment-form.tsx +3 -2
  105. package/templates/blank/src/app/(base)/environments/environment-helpers.ts +4 -1
  106. package/templates/blank/src/app/(base)/layout.tsx +10 -10
  107. package/templates/blank/src/app/(base)/locator-groups/locator-group-form.tsx +5 -3
  108. package/templates/blank/src/app/(base)/locator-groups/locator-group-table-columns.tsx +77 -77
  109. package/templates/blank/src/app/(base)/locator-groups/locator-group-table.tsx +28 -28
  110. package/templates/blank/src/app/(base)/locator-groups/modify/[id]/page.tsx +46 -46
  111. package/templates/blank/src/app/(base)/locator-groups/page.tsx +9 -19
  112. package/templates/blank/src/app/(base)/locators/create/create-locator-workspace-helpers.test.ts +3 -6
  113. package/templates/blank/src/app/(base)/locators/create/create-locator-workspace-response.ts +22 -5
  114. package/templates/blank/src/app/(base)/locators/create/create-locator-workspace-state.ts +5 -1
  115. package/templates/blank/src/app/(base)/locators/create/use-locator-workspace.ts +7 -2
  116. package/templates/blank/src/app/(base)/locators/locator-file-sync.ts +1 -5
  117. package/templates/blank/src/app/(base)/locators/locator-helpers.ts +2 -12
  118. package/templates/blank/src/app/(base)/locators/locator-table-columns.tsx +59 -60
  119. package/templates/blank/src/app/(base)/modules/module-form.tsx +3 -1
  120. package/templates/blank/src/app/(base)/reports/overview-chart.tsx +49 -49
  121. package/templates/blank/src/app/(base)/reports/test-cases/test-cases-metric-table-columns.tsx +114 -115
  122. package/templates/blank/src/app/(base)/reports/test-cases/test-cases-metric-table.tsx +27 -27
  123. package/templates/blank/src/app/(base)/reports/test-suites/test-suites-metric-table-columns.tsx +80 -79
  124. package/templates/blank/src/app/(base)/reports/test-suites/test-suites-metric-table.tsx +29 -27
  125. package/templates/blank/src/app/(base)/settings/settings-sync-panel-helpers.test.tsx +4 -6
  126. package/templates/blank/src/app/(base)/settings/settings-sync-panel.tsx +22 -23
  127. package/templates/blank/src/app/(base)/tags/tag-form.test.tsx +2 -10
  128. package/templates/blank/src/app/(base)/tags/tag-form.tsx +5 -3
  129. package/templates/blank/src/app/(base)/tags/tag-table-columns.tsx +63 -63
  130. package/templates/blank/src/app/(base)/tags/tag-table.tsx +29 -29
  131. package/templates/blank/src/app/(base)/template-step-groups/create/page.tsx +28 -28
  132. package/templates/blank/src/app/(base)/template-step-groups/template-step-group-form.tsx +3 -1
  133. package/templates/blank/src/app/(base)/template-step-groups/template-step-group-helpers.ts +4 -1
  134. package/templates/blank/src/app/(base)/template-steps/paramChip.tsx +6 -10
  135. package/templates/blank/src/app/(base)/template-steps/template-step-form-helpers.ts +4 -2
  136. package/templates/blank/src/app/(base)/template-steps/template-step-helpers.test.ts +19 -1
  137. package/templates/blank/src/app/(base)/template-steps/template-step-helpers.ts +1 -5
  138. package/templates/blank/src/app/(base)/template-steps/template-step-row-guards.ts +1 -5
  139. package/templates/blank/src/app/(base)/template-steps/template-step-types.ts +1 -5
  140. package/templates/blank/src/app/(base)/template-test-cases/create-template-test-case-page-data.test.ts +4 -1
  141. package/templates/blank/src/app/(base)/template-test-cases/create-template-test-case-page-data.ts +1 -2
  142. package/templates/blank/src/app/(base)/template-test-cases/modify/[id]/page.tsx +1 -0
  143. package/templates/blank/src/app/(base)/template-test-cases/template-test-case-flow.test.tsx +0 -1
  144. package/templates/blank/src/app/(base)/template-test-cases/template-test-case-form.tsx +3 -1
  145. package/templates/blank/src/app/(base)/template-test-cases/template-test-case-table-columns.tsx +76 -76
  146. package/templates/blank/src/app/(base)/template-test-cases/template-test-case-table.tsx +32 -32
  147. package/templates/blank/src/app/(base)/test-cases/create/page.tsx +1 -5
  148. package/templates/blank/src/app/(base)/test-cases/create-from-template/create-from-template-selection-helpers.ts +1 -4
  149. package/templates/blank/src/app/(base)/test-cases/create-from-template/page.test.tsx +2 -2
  150. package/templates/blank/src/app/(base)/test-cases/inline-tag-creation-dialog.tsx +1 -7
  151. package/templates/blank/src/app/(base)/test-cases/inline-test-suite-creation-dialog.tsx +1 -7
  152. package/templates/blank/src/app/(base)/test-cases/modify/[id]/page.tsx +1 -5
  153. package/templates/blank/src/app/(base)/test-cases/test-case-form.tsx +63 -78
  154. package/templates/blank/src/app/(base)/test-cases/test-case-route-helpers.test.ts +2 -7
  155. package/templates/blank/src/app/(base)/test-suites/editable-test-suite-helpers.ts +1 -6
  156. package/templates/blank/src/app/(base)/test-suites/test-suite-form.test.tsx +2 -10
  157. package/templates/blank/src/app/(dashboard-components)/data-card-grid.tsx +10 -10
  158. package/templates/blank/src/app/api/test-runs/[runId]/download/route.ts +1 -4
  159. package/templates/blank/src/app/api/test-runs/[runId]/trace/[testCaseId]/route.test.ts +18 -9
  160. package/templates/blank/src/assets/icons/empty-tube.tsx +23 -23
  161. package/templates/blank/src/assets/icons/tube-plus.tsx +29 -29
  162. package/templates/blank/src/components/data-visualization/info-card.tsx +56 -70
  163. package/templates/blank/src/components/data-visualization/info-grid.tsx +22 -22
  164. package/templates/blank/src/components/diagram/add-node-prompt-node.tsx +1 -1
  165. package/templates/blank/src/components/diagram/dynamic-parameters-helpers.ts +4 -3
  166. package/templates/blank/src/components/diagram/dynamic-parameters.test.tsx +4 -1
  167. package/templates/blank/src/components/diagram/dynamic-parameters.tsx +0 -1
  168. package/templates/blank/src/components/diagram/flow-diagram-block-dialog.test.tsx +26 -0
  169. package/templates/blank/src/components/diagram/flow-diagram-block-dialog.tsx +3 -1
  170. package/templates/blank/src/components/diagram/flow-diagram-connection-guards.ts +24 -6
  171. package/templates/blank/src/components/diagram/flow-diagram-helpers.test.ts +27 -0
  172. package/templates/blank/src/components/diagram/flow-diagram-node-search.tsx +65 -65
  173. package/templates/blank/src/components/diagram/flow-diagram-toolbar.tsx +1 -3
  174. package/templates/blank/src/components/diagram/flow-diagram-view.tsx +1 -7
  175. package/templates/blank/src/components/diagram/flow-node-data-helpers.ts +1 -1
  176. package/templates/blank/src/components/diagram/node-form-fields-content.tsx +9 -2
  177. package/templates/blank/src/components/diagram/node-form-helpers.test.ts +3 -1
  178. package/templates/blank/src/components/diagram/node-form-helpers.ts +1 -4
  179. package/templates/blank/src/components/diagram/node-form-template-step-selection.ts +1 -4
  180. package/templates/blank/src/components/diagram/node-form.test.tsx +25 -0
  181. package/templates/blank/src/components/diagram/node-form.tsx +2 -12
  182. package/templates/blank/src/components/diagram/options-header-node.test.tsx +4 -1
  183. package/templates/blank/src/components/diagram/options-header-node.tsx +140 -130
  184. package/templates/blank/src/components/diagram/use-flow-diagram.ts +1 -2
  185. package/templates/blank/src/components/form/error-message.tsx +7 -7
  186. package/templates/blank/src/components/loading-skeleton/form/button-skeleton.tsx +8 -8
  187. package/templates/blank/src/components/loading-skeleton/form/text-input-skeleton.tsx +8 -8
  188. package/templates/blank/src/components/loading-skeleton/visualization/table-skeleton.tsx +14 -14
  189. package/templates/blank/src/components/navigation/entity-search-command.tsx +3 -9
  190. package/templates/blank/src/components/navigation/nav-link.tsx +60 -60
  191. package/templates/blank/src/components/test-case/test-case-form-helpers.test.ts +1 -5
  192. package/templates/blank/src/components/test-case/test-case-form-helpers.ts +1 -5
  193. package/templates/blank/src/components/test-case/test-case-picker-helpers.ts +8 -7
  194. package/templates/blank/src/components/test-case/test-case-picker.tsx +2 -6
  195. package/templates/blank/src/components/test-run/log-viewer.test.tsx +25 -0
  196. package/templates/blank/src/components/test-run/log-viewer.tsx +2 -2
  197. package/templates/blank/src/components/test-run/test-run-details-guards.ts +10 -1
  198. package/templates/blank/src/components/test-run/test-run-details.test.tsx +9 -19
  199. package/templates/blank/src/components/test-run/use-log-viewer.ts +2 -8
  200. package/templates/blank/src/components/test-run/use-test-run-header.ts +1 -4
  201. package/templates/blank/src/components/test-suite/test-suite-picker.tsx +1 -1
  202. package/templates/blank/src/components/theme/theme-provider.tsx +7 -1
  203. package/templates/blank/src/components/typography/page-header-subtitle.tsx +7 -7
  204. package/templates/blank/src/components/typography/page-header.tsx +7 -7
  205. package/templates/blank/src/components/ui/chart.tsx +2 -8
  206. package/templates/blank/src/components/ui/command.tsx +2 -5
  207. package/templates/blank/src/components/ui/data-table-column-header.tsx +3 -3
  208. package/templates/blank/src/components/ui/data-table.test.tsx +1 -8
  209. package/templates/blank/src/components/ui/dialog.tsx +1 -4
  210. package/templates/blank/src/components/ui/select.tsx +2 -2
  211. package/templates/blank/src/components/ui/separator.tsx +5 -1
  212. package/templates/blank/src/components/ui/skeleton.tsx +7 -7
  213. package/templates/blank/src/components/ui/table.tsx +1 -1
  214. package/templates/blank/src/components/ui/toaster.tsx +26 -26
  215. package/templates/blank/src/components/ui/tooltip.tsx +1 -1
  216. package/templates/blank/src/constants/form-opts/locator-group-form-opts.ts +1 -1
  217. package/templates/blank/src/lib/locator-group-file-utils.ts +5 -1
  218. package/templates/blank/src/lib/locator-picker/session-manager.ts +1 -2
  219. package/templates/blank/src/lib/metrics/metric-calculator.test.ts +105 -6
  220. package/templates/blank/src/lib/metrics/metric-calculator.ts +39 -40
  221. package/templates/blank/src/lib/sync/sync-executor.ts +1 -4
  222. package/templates/blank/src/lib/sync/sync-pending-counts.test.ts +201 -7
  223. package/templates/blank/src/lib/sync/sync-pending-counts.ts +81 -13
  224. package/templates/blank/src/lib/template-automation-paths.ts +1 -1
  225. package/templates/blank/src/lib/template-sync-utils.d.ts +14 -7
  226. package/templates/blank/src/lib/test-case-utils.ts +6 -6
  227. package/templates/blank/src/lib/test-run/log-formatter.ts +82 -83
  228. package/templates/blank/src/lib/test-run/report-parser.ts +352 -352
  229. package/templates/blank/src/lib/transformers/gherkin-converter.ts +42 -42
  230. package/templates/blank/src/lib/transformers/key-to-icon-transformer.tsx +95 -95
  231. package/templates/blank/src/lib/utils/node-param-validation.ts +81 -81
  232. package/templates/blank/src/lib/utils/template-step-file-generator.ts +5 -3
  233. package/templates/blank/src/lib/utils/template-step-file-manager-intelligent.ts +11 -1
  234. package/templates/blank/src/services/dashboard/dashboard-service.test.ts +28 -1
  235. package/templates/blank/src/services/dashboard/dashboard-service.ts +2 -0
  236. package/templates/blank/src/services/locator/locator-service.test.ts +1 -3
  237. package/templates/blank/src/services/locator/locator-service.ts +3 -1
  238. package/templates/blank/src/services/locator/locator-sync-utils.test.ts +1 -4
  239. package/templates/blank/src/services/module/module-service.ts +1 -4
  240. package/templates/blank/src/services/report/report-service.test.ts +70 -6
  241. package/templates/blank/src/services/report/report-service.ts +52 -9
  242. package/templates/blank/src/services/shared/errors.ts +2 -4
  243. package/templates/blank/src/services/template-step/template-step-service.ts +3 -1
  244. package/templates/blank/src/services/template-step-group/template-step-group-service.ts +3 -1
  245. package/templates/blank/src/services/test-case/test-case-service.ts +0 -5
  246. package/templates/blank/src/services/test-run/test-run-service.test.ts +3 -6
  247. package/templates/blank/src/services/test-run/test-run-service.ts +12 -7
  248. package/templates/blank/src/services/test-suite/test-suite-service.test.ts +1 -5
  249. package/templates/blank/src/services/test-suite/test-suite-service.ts +2 -4
  250. package/templates/blank/src/types/form/actionHandler.ts +1 -5
  251. package/templates/blank/tsconfig.json +8 -8
  252. package/templates/default/packages/locator-picker-companion/dist/cli.d.ts +1 -1
  253. package/templates/default/packages/locator-picker-companion/dist/cli.js +313 -279
  254. package/templates/default/packages/locator-picker-companion/dist/index.d.ts +3 -3
  255. package/templates/default/packages/locator-picker-companion/dist/index.js +3 -3
  256. package/templates/default/packages/locator-picker-companion/dist/injected-picker-script.d.ts +1 -1
  257. package/templates/default/packages/locator-picker-companion/dist/injected-picker-script.js +526 -481
  258. package/templates/default/packages/locator-picker-companion/dist/launcher.d.ts +13 -10
  259. package/templates/default/packages/locator-picker-companion/dist/launcher.js +47 -48
  260. package/templates/default/packages/locator-picker-companion/dist/selector-generator.d.ts +6 -3
  261. package/templates/default/packages/locator-picker-companion/dist/selector-generator.js +233 -229
  262. package/templates/default/packages/locator-picker-companion/dist/session-file.d.ts +30 -22
  263. package/templates/default/packages/locator-picker-companion/dist/session-file.js +116 -104
  264. package/templates/default/packages/locator-picker-companion/dist/types.d.ts +25 -25
  265. package/templates/default/packages/locator-picker-companion/dist/types.js +1 -1
  266. package/templates/starter/.editorconfig +11 -0
  267. package/templates/starter/.gitattributes +21 -0
  268. package/templates/starter/.gitconfig.appraise +14 -0
  269. package/templates/starter/.prettierrc +13 -0
  270. package/templates/starter/.vscode/settings.json +2 -0
  271. package/templates/starter/README.md +12 -12
  272. package/templates/starter/automation/steps/actions/click.step.ts +56 -58
  273. package/templates/starter/automation/steps/actions/hover.step.ts +25 -27
  274. package/templates/starter/automation/steps/actions/input.step.ts +85 -108
  275. package/templates/starter/automation/steps/actions/navigation.step.ts +67 -70
  276. package/templates/starter/automation/steps/actions/random_data.step.ts +149 -143
  277. package/templates/starter/automation/steps/actions/store.step.ts +69 -1
  278. package/templates/starter/automation/steps/actions/wait.step.ts +84 -1
  279. package/templates/starter/automation/steps/validations/active_state_assertion.step.ts +28 -30
  280. package/templates/starter/automation/steps/validations/navigation_assertion.step.ts +20 -22
  281. package/templates/starter/automation/steps/validations/text_assertion.step.ts +67 -107
  282. package/templates/starter/automation/steps/validations/visibility_assertion.step.ts +24 -26
  283. package/templates/starter/components.json +24 -24
  284. package/templates/starter/cucumber.mjs +16 -16
  285. package/templates/starter/e2e/README.md +56 -0
  286. package/templates/starter/e2e/apply-migrations.mjs +76 -0
  287. package/templates/starter/e2e/appraise-smoke.spec.ts +78 -0
  288. package/templates/starter/e2e/authoring.spec.ts +53 -0
  289. package/templates/starter/e2e/crud-configuration.spec.ts +138 -0
  290. package/templates/starter/e2e/crud-tests.spec.ts +76 -0
  291. package/templates/starter/e2e/helpers/forms.ts +186 -0
  292. package/templates/starter/e2e/helpers/navigation.ts +27 -0
  293. package/templates/starter/e2e/helpers/table.ts +45 -0
  294. package/templates/starter/e2e/helpers/test-data.ts +549 -0
  295. package/templates/starter/e2e/helpers/ui.ts +41 -0
  296. package/templates/starter/e2e/navigation.spec.ts +101 -0
  297. package/templates/starter/e2e/runs-and-reports.spec.ts +82 -0
  298. package/templates/starter/e2e/settings-sync.spec.ts +39 -0
  299. package/templates/starter/e2e/start-server.mjs +61 -0
  300. package/templates/starter/eslint.config.mjs +4 -4
  301. package/templates/starter/gitignore +0 -3
  302. package/templates/starter/next-env.d.ts +1 -1
  303. package/templates/starter/package-lock.json +58 -102
  304. package/templates/starter/package.json +10 -9
  305. package/templates/starter/packages/cucumber-runtime/package.json +13 -13
  306. package/templates/starter/packages/cucumber-runtime/src/cache.util.ts +93 -93
  307. package/templates/starter/packages/cucumber-runtime/src/cli.ts +77 -68
  308. package/templates/starter/packages/cucumber-runtime/src/environment.util.ts +21 -21
  309. package/templates/starter/packages/cucumber-runtime/src/executor.ts +32 -32
  310. package/templates/starter/packages/cucumber-runtime/src/index.ts +17 -17
  311. package/templates/starter/packages/cucumber-runtime/src/locator.util.ts +234 -234
  312. package/templates/starter/packages/cucumber-runtime/src/parameter-types.ts +7 -7
  313. package/templates/starter/packages/cucumber-runtime/src/random-data.util.ts +35 -35
  314. package/templates/starter/packages/cucumber-runtime/src/types.ts +13 -13
  315. package/templates/starter/packages/cucumber-runtime/src/world.ts +44 -44
  316. package/templates/starter/packages/cucumber-runtime/tsconfig.json +11 -11
  317. package/templates/starter/packages/locator-picker-companion/dist/cli.js +1 -1
  318. package/templates/starter/packages/locator-picker-companion/dist/index.d.ts +3 -3
  319. package/templates/starter/packages/locator-picker-companion/dist/index.js +3 -3
  320. package/templates/starter/packages/locator-picker-companion/dist/injected-picker-script.js +13 -3
  321. package/templates/starter/packages/locator-picker-companion/dist/selector-generator.d.ts +6 -3
  322. package/templates/starter/packages/locator-picker-companion/dist/selector-generator.js +233 -229
  323. package/templates/starter/packages/locator-picker-companion/src/cli.ts +1 -7
  324. package/templates/starter/packages/locator-picker-companion/src/injected-picker-script.ts +26 -5
  325. package/templates/starter/packages/locator-picker-companion/src/launcher.ts +1 -3
  326. package/templates/starter/packages/locator-picker-companion/src/session-file.ts +4 -15
  327. package/templates/starter/packages/locator-picker-companion/src/types.ts +1 -8
  328. package/templates/starter/playwright.config.ts +46 -0
  329. package/templates/starter/postcss.config.mjs +8 -8
  330. package/templates/starter/prisma/dev.db +0 -0
  331. package/templates/starter/scripts/configure-git-line-endings.mjs +91 -0
  332. package/templates/starter/scripts/install-template-step.ts +5 -1
  333. package/templates/starter/scripts/lib/step-matcher.test.ts +3 -3
  334. package/templates/starter/scripts/lib/step-matcher.ts +5 -1
  335. package/templates/starter/scripts/lib/template-step-installer.test.ts +4 -2
  336. package/templates/starter/scripts/lib/template-step-installer.ts +16 -4
  337. package/templates/starter/scripts/lib/template-step-registry.test.ts +1 -1
  338. package/templates/starter/scripts/lib/template-step-registry.ts +4 -1
  339. package/templates/starter/scripts/run-fallow-commit.mjs +4 -6
  340. package/templates/starter/scripts/sync-all.ts +7 -6
  341. package/templates/starter/scripts/sync-appraise-base-template.ts +84 -77
  342. package/templates/starter/scripts/sync-environments.ts +47 -44
  343. package/templates/starter/scripts/sync-locator-groups.ts +67 -76
  344. package/templates/starter/scripts/sync-locators.ts +50 -53
  345. package/templates/starter/scripts/sync-modules.ts +42 -42
  346. package/templates/starter/scripts/sync-tags.ts +51 -51
  347. package/templates/starter/scripts/sync-template-step-groups.ts +34 -32
  348. package/templates/starter/scripts/sync-template-steps.ts +97 -97
  349. package/templates/starter/scripts/sync-test-cases.ts +65 -74
  350. package/templates/starter/scripts/sync-test-suites.ts +85 -80
  351. package/templates/starter/src/actions/server-action-boundary.test.ts +1 -7
  352. package/templates/starter/src/actions/settings/sync-actions.ts +3 -1
  353. package/templates/starter/src/actions/tags/tag-actions.ts +1 -7
  354. package/templates/starter/src/actions/template-test-case/template-test-case-actions.ts +3 -1
  355. package/templates/starter/src/app/(base)/environments/create/page.tsx +28 -28
  356. package/templates/starter/src/app/(base)/environments/environment-form.tsx +3 -2
  357. package/templates/starter/src/app/(base)/environments/environment-helpers.ts +4 -1
  358. package/templates/starter/src/app/(base)/layout.tsx +10 -10
  359. package/templates/starter/src/app/(base)/locator-groups/locator-group-form.tsx +5 -3
  360. package/templates/starter/src/app/(base)/locator-groups/locator-group-table-columns.tsx +77 -77
  361. package/templates/starter/src/app/(base)/locator-groups/locator-group-table.tsx +28 -28
  362. package/templates/starter/src/app/(base)/locator-groups/modify/[id]/page.tsx +46 -46
  363. package/templates/starter/src/app/(base)/locator-groups/page.tsx +9 -19
  364. package/templates/starter/src/app/(base)/locators/create/create-locator-workspace-helpers.test.ts +3 -6
  365. package/templates/starter/src/app/(base)/locators/create/create-locator-workspace-response.ts +22 -5
  366. package/templates/starter/src/app/(base)/locators/create/create-locator-workspace-state.ts +5 -1
  367. package/templates/starter/src/app/(base)/locators/create/use-locator-workspace.ts +7 -2
  368. package/templates/starter/src/app/(base)/locators/locator-file-sync.ts +1 -5
  369. package/templates/starter/src/app/(base)/locators/locator-helpers.ts +2 -12
  370. package/templates/starter/src/app/(base)/locators/locator-table-columns.tsx +59 -60
  371. package/templates/starter/src/app/(base)/modules/module-form.tsx +3 -1
  372. package/templates/starter/src/app/(base)/reports/overview-chart.tsx +49 -49
  373. package/templates/starter/src/app/(base)/reports/test-cases/test-cases-metric-table-columns.tsx +114 -115
  374. package/templates/starter/src/app/(base)/reports/test-cases/test-cases-metric-table.tsx +27 -27
  375. package/templates/starter/src/app/(base)/reports/test-suites/test-suites-metric-table-columns.tsx +80 -79
  376. package/templates/starter/src/app/(base)/reports/test-suites/test-suites-metric-table.tsx +29 -27
  377. package/templates/starter/src/app/(base)/settings/settings-sync-panel-helpers.test.tsx +4 -6
  378. package/templates/starter/src/app/(base)/settings/settings-sync-panel.tsx +22 -23
  379. package/templates/starter/src/app/(base)/tags/tag-form.test.tsx +2 -10
  380. package/templates/starter/src/app/(base)/tags/tag-form.tsx +5 -3
  381. package/templates/starter/src/app/(base)/tags/tag-table-columns.tsx +63 -63
  382. package/templates/starter/src/app/(base)/tags/tag-table.tsx +29 -29
  383. package/templates/starter/src/app/(base)/template-step-groups/create/page.tsx +28 -28
  384. package/templates/starter/src/app/(base)/template-step-groups/template-step-group-form.tsx +3 -1
  385. package/templates/starter/src/app/(base)/template-step-groups/template-step-group-helpers.ts +4 -1
  386. package/templates/starter/src/app/(base)/template-steps/paramChip.tsx +6 -10
  387. package/templates/starter/src/app/(base)/template-steps/template-step-form-helpers.ts +4 -2
  388. package/templates/starter/src/app/(base)/template-steps/template-step-helpers.test.ts +19 -1
  389. package/templates/starter/src/app/(base)/template-steps/template-step-helpers.ts +1 -5
  390. package/templates/starter/src/app/(base)/template-steps/template-step-row-guards.ts +1 -5
  391. package/templates/starter/src/app/(base)/template-steps/template-step-types.ts +1 -5
  392. package/templates/starter/src/app/(base)/template-test-cases/create-template-test-case-page-data.test.ts +4 -1
  393. package/templates/starter/src/app/(base)/template-test-cases/create-template-test-case-page-data.ts +1 -2
  394. package/templates/starter/src/app/(base)/template-test-cases/modify/[id]/page.tsx +1 -0
  395. package/templates/starter/src/app/(base)/template-test-cases/template-test-case-flow.test.tsx +0 -1
  396. package/templates/starter/src/app/(base)/template-test-cases/template-test-case-form.tsx +3 -1
  397. package/templates/starter/src/app/(base)/template-test-cases/template-test-case-table-columns.tsx +76 -76
  398. package/templates/starter/src/app/(base)/template-test-cases/template-test-case-table.tsx +32 -32
  399. package/templates/starter/src/app/(base)/test-cases/create/page.tsx +1 -5
  400. package/templates/starter/src/app/(base)/test-cases/create-from-template/create-from-template-selection-helpers.ts +1 -4
  401. package/templates/starter/src/app/(base)/test-cases/create-from-template/page.test.tsx +2 -2
  402. package/templates/starter/src/app/(base)/test-cases/inline-tag-creation-dialog.tsx +1 -7
  403. package/templates/starter/src/app/(base)/test-cases/inline-test-suite-creation-dialog.tsx +1 -7
  404. package/templates/starter/src/app/(base)/test-cases/modify/[id]/page.tsx +1 -5
  405. package/templates/starter/src/app/(base)/test-cases/test-case-form.tsx +63 -78
  406. package/templates/starter/src/app/(base)/test-cases/test-case-route-helpers.test.ts +2 -7
  407. package/templates/starter/src/app/(base)/test-suites/editable-test-suite-helpers.ts +1 -6
  408. package/templates/starter/src/app/(base)/test-suites/test-suite-form.test.tsx +2 -10
  409. package/templates/starter/src/app/(dashboard-components)/data-card-grid.tsx +10 -10
  410. package/templates/starter/src/app/api/test-runs/[runId]/download/route.ts +1 -4
  411. package/templates/starter/src/app/api/test-runs/[runId]/trace/[testCaseId]/route.test.ts +18 -9
  412. package/templates/starter/src/assets/icons/empty-tube.tsx +23 -23
  413. package/templates/starter/src/assets/icons/tube-plus.tsx +29 -29
  414. package/templates/starter/src/components/data-visualization/info-card.tsx +56 -70
  415. package/templates/starter/src/components/data-visualization/info-grid.tsx +22 -22
  416. package/templates/starter/src/components/diagram/add-node-prompt-node.tsx +1 -1
  417. package/templates/starter/src/components/diagram/dynamic-parameters-helpers.ts +4 -3
  418. package/templates/starter/src/components/diagram/dynamic-parameters.test.tsx +4 -1
  419. package/templates/starter/src/components/diagram/dynamic-parameters.tsx +0 -1
  420. package/templates/starter/src/components/diagram/flow-diagram-block-dialog.test.tsx +26 -0
  421. package/templates/starter/src/components/diagram/flow-diagram-block-dialog.tsx +3 -1
  422. package/templates/starter/src/components/diagram/flow-diagram-connection-guards.ts +24 -6
  423. package/templates/starter/src/components/diagram/flow-diagram-helpers.test.ts +27 -0
  424. package/templates/starter/src/components/diagram/flow-diagram-node-search.tsx +65 -65
  425. package/templates/starter/src/components/diagram/flow-diagram-toolbar.tsx +1 -3
  426. package/templates/starter/src/components/diagram/flow-diagram-view.tsx +1 -7
  427. package/templates/starter/src/components/diagram/flow-node-data-helpers.ts +1 -1
  428. package/templates/starter/src/components/diagram/node-form-fields-content.tsx +9 -2
  429. package/templates/starter/src/components/diagram/node-form-helpers.test.ts +3 -1
  430. package/templates/starter/src/components/diagram/node-form-helpers.ts +1 -4
  431. package/templates/starter/src/components/diagram/node-form-template-step-selection.ts +1 -4
  432. package/templates/starter/src/components/diagram/node-form.test.tsx +25 -0
  433. package/templates/starter/src/components/diagram/node-form.tsx +2 -12
  434. package/templates/starter/src/components/diagram/options-header-node.test.tsx +4 -1
  435. package/templates/starter/src/components/diagram/options-header-node.tsx +140 -130
  436. package/templates/starter/src/components/diagram/use-flow-diagram.ts +1 -2
  437. package/templates/starter/src/components/form/error-message.tsx +7 -7
  438. package/templates/starter/src/components/loading-skeleton/form/button-skeleton.tsx +8 -8
  439. package/templates/starter/src/components/loading-skeleton/form/text-input-skeleton.tsx +8 -8
  440. package/templates/starter/src/components/loading-skeleton/visualization/table-skeleton.tsx +14 -14
  441. package/templates/starter/src/components/navigation/entity-search-command.tsx +3 -9
  442. package/templates/starter/src/components/navigation/nav-link.tsx +60 -60
  443. package/templates/starter/src/components/test-case/test-case-form-helpers.test.ts +1 -5
  444. package/templates/starter/src/components/test-case/test-case-form-helpers.ts +1 -5
  445. package/templates/starter/src/components/test-case/test-case-picker-helpers.ts +8 -7
  446. package/templates/starter/src/components/test-case/test-case-picker.tsx +2 -6
  447. package/templates/starter/src/components/test-run/log-viewer.test.tsx +25 -0
  448. package/templates/starter/src/components/test-run/log-viewer.tsx +2 -2
  449. package/templates/starter/src/components/test-run/test-run-details-guards.ts +10 -1
  450. package/templates/starter/src/components/test-run/test-run-details.test.tsx +9 -19
  451. package/templates/starter/src/components/test-run/use-log-viewer.ts +2 -8
  452. package/templates/starter/src/components/test-run/use-test-run-header.ts +1 -4
  453. package/templates/starter/src/components/test-suite/test-suite-picker.tsx +1 -1
  454. package/templates/starter/src/components/theme/theme-provider.tsx +7 -1
  455. package/templates/starter/src/components/typography/page-header-subtitle.tsx +7 -7
  456. package/templates/starter/src/components/typography/page-header.tsx +7 -7
  457. package/templates/starter/src/components/ui/chart.tsx +2 -8
  458. package/templates/starter/src/components/ui/command.tsx +2 -5
  459. package/templates/starter/src/components/ui/data-table-column-header.tsx +3 -3
  460. package/templates/starter/src/components/ui/data-table.test.tsx +1 -8
  461. package/templates/starter/src/components/ui/dialog.tsx +1 -4
  462. package/templates/starter/src/components/ui/select.tsx +2 -2
  463. package/templates/starter/src/components/ui/separator.tsx +5 -1
  464. package/templates/starter/src/components/ui/skeleton.tsx +7 -7
  465. package/templates/starter/src/components/ui/table.tsx +1 -1
  466. package/templates/starter/src/components/ui/toaster.tsx +26 -26
  467. package/templates/starter/src/components/ui/tooltip.tsx +1 -1
  468. package/templates/starter/src/constants/form-opts/locator-group-form-opts.ts +1 -1
  469. package/templates/starter/src/lib/locator-group-file-utils.ts +5 -1
  470. package/templates/starter/src/lib/locator-picker/session-manager.ts +1 -2
  471. package/templates/starter/src/lib/metrics/metric-calculator.test.ts +105 -6
  472. package/templates/starter/src/lib/metrics/metric-calculator.ts +39 -40
  473. package/templates/starter/src/lib/sync/sync-executor.ts +1 -4
  474. package/templates/starter/src/lib/sync/sync-pending-counts.test.ts +201 -7
  475. package/templates/starter/src/lib/sync/sync-pending-counts.ts +81 -13
  476. package/templates/starter/src/lib/template-automation-paths.ts +1 -1
  477. package/templates/starter/src/lib/template-sync-utils.d.ts +14 -7
  478. package/templates/starter/src/lib/test-case-utils.ts +6 -6
  479. package/templates/starter/src/lib/test-run/log-formatter.ts +82 -83
  480. package/templates/starter/src/lib/test-run/report-parser.ts +352 -352
  481. package/templates/starter/src/lib/transformers/gherkin-converter.ts +42 -42
  482. package/templates/starter/src/lib/transformers/key-to-icon-transformer.tsx +95 -95
  483. package/templates/starter/src/lib/utils/node-param-validation.ts +81 -81
  484. package/templates/starter/src/lib/utils/template-step-file-generator.ts +5 -3
  485. package/templates/starter/src/lib/utils/template-step-file-manager-intelligent.ts +11 -1
  486. package/templates/starter/src/services/dashboard/dashboard-service.test.ts +28 -1
  487. package/templates/starter/src/services/dashboard/dashboard-service.ts +2 -0
  488. package/templates/starter/src/services/locator/locator-service.test.ts +1 -3
  489. package/templates/starter/src/services/locator/locator-service.ts +3 -1
  490. package/templates/starter/src/services/locator/locator-sync-utils.test.ts +1 -4
  491. package/templates/starter/src/services/module/module-service.ts +1 -4
  492. package/templates/starter/src/services/report/report-service.test.ts +70 -6
  493. package/templates/starter/src/services/report/report-service.ts +52 -9
  494. package/templates/starter/src/services/shared/errors.ts +2 -4
  495. package/templates/starter/src/services/template-step/template-step-service.ts +3 -1
  496. package/templates/starter/src/services/template-step-group/template-step-group-service.ts +3 -1
  497. package/templates/starter/src/services/test-case/test-case-service.ts +0 -5
  498. package/templates/starter/src/services/test-run/test-run-service.test.ts +3 -6
  499. package/templates/starter/src/services/test-run/test-run-service.ts +12 -7
  500. package/templates/starter/src/services/test-suite/test-suite-service.test.ts +1 -5
  501. package/templates/starter/src/services/test-suite/test-suite-service.ts +2 -4
  502. package/templates/starter/src/types/form/actionHandler.ts +1 -5
  503. package/templates/starter/tsconfig.json +8 -8
@@ -1,288 +1,300 @@
1
1
  export function installLocatorPickerOverlay() {
2
- var _a;
3
- const globalState = window;
4
- const ROOT_ID = 'appraise-locator-picker-root';
5
- const STYLE_ID = 'appraise-locator-picker-style';
6
- const HOVER_CLASS = 'appraise-locator-picker-hover';
7
- const ACTIVE_CLASS = 'appraise-locator-picker-active';
8
- const state = (_a = globalState.__appraiseLocatorPickerState) !== null && _a !== void 0 ? _a : (globalState.__appraiseLocatorPickerState = {
9
- picking: false,
10
- submitting: false,
11
- hoveredElement: null,
12
- preview: null,
13
- error: '',
14
- elements: {},
15
- });
16
- function stopOverlayEvent(event) {
17
- event.preventDefault();
18
- event.stopPropagation();
19
- }
20
- function normalizeText(value) {
21
- return (value !== null && value !== void 0 ? value : '').replace(/\s+/g, ' ').trim();
22
- }
23
- function escapeForCss(value) {
24
- return value.replace(/[^a-zA-Z0-9_-]/g, match => `\\${match}`);
25
- }
26
- function escapeForSelectorText(value) {
27
- return value.replace(/\\/g, '\\\\').replace(/"/g, '\\"');
28
- }
29
- function normalizeRoute(value) {
30
- if (!value) {
31
- return '/';
32
- }
33
- try {
34
- return new URL(value).pathname || '/';
35
- }
36
- catch (_a) {
37
- return value.startsWith('/') ? value : `/${value}`;
38
- }
2
+ var _a
3
+ const globalState = window
4
+ const ROOT_ID = 'appraise-locator-picker-root'
5
+ const STYLE_ID = 'appraise-locator-picker-style'
6
+ const HOVER_CLASS = 'appraise-locator-picker-hover'
7
+ const ACTIVE_CLASS = 'appraise-locator-picker-active'
8
+ const state =
9
+ (_a = globalState.__appraiseLocatorPickerState) !== null && _a !== void 0
10
+ ? _a
11
+ : (globalState.__appraiseLocatorPickerState = {
12
+ picking: false,
13
+ submitting: false,
14
+ hoveredElement: null,
15
+ preview: null,
16
+ error: '',
17
+ elements: {},
18
+ })
19
+ function stopOverlayEvent(event) {
20
+ event.preventDefault()
21
+ event.stopPropagation()
22
+ }
23
+ function normalizeText(value) {
24
+ return (value !== null && value !== void 0 ? value : '').replace(/\s+/g, ' ').trim()
25
+ }
26
+ function escapeForCss(value) {
27
+ return value.replace(/[^a-zA-Z0-9_-]/g, match => `\\${match}`)
28
+ }
29
+ function escapeForSelectorText(value) {
30
+ return value.replace(/\\/g, '\\\\').replace(/"/g, '\\"')
31
+ }
32
+ function normalizeRoute(value) {
33
+ if (!value) {
34
+ return '/'
39
35
  }
40
- function isLikelyStableIdentifier(value) {
41
- if (!value) {
42
- return false;
43
- }
44
- const normalized = value.trim();
45
- if (!normalized || normalized.length > 120) {
46
- return false;
47
- }
48
- return !/\d{4,}/.test(normalized) && !/[A-Fa-f0-9]{8,}/.test(normalized);
36
+ try {
37
+ return new URL(value).pathname || '/'
38
+ } catch (_a) {
39
+ return value.startsWith('/') ? value : `/${value}`
49
40
  }
50
- function isFormControl(element) {
51
- return ['input', 'textarea', 'select'].includes(element.tagName.toLowerCase());
41
+ }
42
+ function isLikelyStableIdentifier(value) {
43
+ if (!value) {
44
+ return false
52
45
  }
53
- function getLabelText(element) {
54
- if (!isFormControl(element)) {
55
- return '';
56
- }
57
- const input = element;
58
- if (input.labels && input.labels.length > 0) {
59
- return normalizeText(Array.from(input.labels)
60
- .map(label => label.textContent || '')
61
- .join(' '));
62
- }
63
- return '';
46
+ const normalized = value.trim()
47
+ if (!normalized || normalized.length > 120) {
48
+ return false
64
49
  }
65
- function getAccessibleName(element) {
66
- const ariaLabel = normalizeText(element.getAttribute('aria-label'));
67
- if (ariaLabel) {
68
- return ariaLabel;
69
- }
70
- const labelledBy = normalizeText(element.getAttribute('aria-labelledby'));
71
- if (labelledBy) {
72
- const text = labelledBy
73
- .split(/\s+/)
74
- .map(id => { var _a; return ((_a = document.getElementById(id)) === null || _a === void 0 ? void 0 : _a.textContent) || ''; })
75
- .join(' ');
76
- const normalized = normalizeText(text);
77
- if (normalized) {
78
- return normalized;
79
- }
80
- }
81
- const labelText = getLabelText(element);
82
- if (labelText) {
83
- return labelText;
84
- }
85
- const alt = normalizeText(element.getAttribute('alt'));
86
- if (alt) {
87
- return alt;
88
- }
89
- const title = normalizeText(element.getAttribute('title'));
90
- if (title) {
91
- return title;
92
- }
93
- const placeholder = normalizeText(element.getAttribute('placeholder'));
94
- if (placeholder) {
95
- return placeholder;
96
- }
97
- return normalizeText(element.textContent).slice(0, 240);
50
+ return !/\d{4,}/.test(normalized) && !/[A-Fa-f0-9]{8,}/.test(normalized)
51
+ }
52
+ function isFormControl(element) {
53
+ return ['input', 'textarea', 'select'].includes(element.tagName.toLowerCase())
54
+ }
55
+ function getLabelText(element) {
56
+ if (!isFormControl(element)) {
57
+ return ''
98
58
  }
99
- function getRole(element) {
100
- const explicitRole = normalizeText(element.getAttribute('role'));
101
- if (explicitRole) {
102
- return explicitRole;
103
- }
104
- const tagName = element.tagName.toLowerCase();
105
- if (tagName === 'button') {
106
- return 'button';
107
- }
108
- if (tagName === 'a' && element.hasAttribute('href')) {
109
- return 'link';
110
- }
111
- if (tagName === 'textarea') {
112
- return 'textbox';
113
- }
114
- if (tagName === 'select') {
115
- return 'combobox';
116
- }
117
- if (tagName === 'img') {
118
- return 'img';
119
- }
120
- if (tagName === 'input') {
121
- const type = (element.getAttribute('type') || 'text').toLowerCase();
122
- if (['button', 'submit', 'reset'].includes(type)) {
123
- return 'button';
124
- }
125
- if (type === 'checkbox') {
126
- return 'checkbox';
127
- }
128
- if (type === 'radio') {
129
- return 'radio';
130
- }
131
- return 'textbox';
132
- }
133
- return '';
134
- }
135
- function countRoleNameMatches(role, accessibleName) {
136
- let count = 0;
137
- for (const candidate of Array.from(document.querySelectorAll('*'))) {
138
- if (getRole(candidate) !== role) {
139
- continue;
140
- }
141
- if (getAccessibleName(candidate) !== accessibleName) {
142
- continue;
143
- }
144
- count += 1;
145
- if (count > 1) {
146
- break;
147
- }
148
- }
149
- return count;
150
- }
151
- function buildCssSelector(element) {
152
- const tagName = element.tagName.toLowerCase();
153
- const name = element.getAttribute('name');
154
- const type = element.getAttribute('type');
155
- const ariaLabel = element.getAttribute('aria-label');
156
- const parts = [tagName];
157
- if (isLikelyStableIdentifier(name)) {
158
- parts.push(`[name="${escapeForSelectorText(name)}"]`);
159
- }
160
- if (isLikelyStableIdentifier(type)) {
161
- parts.push(`[type="${escapeForSelectorText(type)}"]`);
162
- }
163
- if (isLikelyStableIdentifier(ariaLabel)) {
164
- parts.push(`[aria-label="${escapeForSelectorText(ariaLabel)}"]`);
165
- }
166
- if (parts.length === 1) {
167
- const stableClasses = Array.from(element.classList || []).filter(isLikelyStableIdentifier).slice(0, 2);
168
- if (stableClasses.length > 0) {
169
- parts.push(...stableClasses.map(className => `.${escapeForCss(className)}`));
170
- }
171
- }
172
- return parts.length > 1 ? `css=${parts.join('')}` : '';
59
+ const input = element
60
+ if (input.labels && input.labels.length > 0) {
61
+ return normalizeText(
62
+ Array.from(input.labels)
63
+ .map(label => label.textContent || '')
64
+ .join(' '),
65
+ )
173
66
  }
174
- function buildXPathSelector(element, text) {
175
- const tagName = element.tagName.toLowerCase();
176
- const id = element.getAttribute('id');
177
- if (id) {
178
- return `xpath=//*[@id="${id.replace(/"/g, '\\"')}"]`;
179
- }
180
- if (text) {
181
- return `xpath=//${tagName}[normalize-space()="${text.replace(/"/g, '\\"')}"]`;
182
- }
183
- return `xpath=//${tagName}`;
184
- }
185
- function buildPrimarySelector(element) {
186
- for (const attributeName of ['data-testid', 'data-test', 'data-qa']) {
187
- const value = element.getAttribute(attributeName);
188
- if (isLikelyStableIdentifier(value)) {
189
- return {
190
- selector: `css=[${attributeName}="${escapeForSelectorText(value)}"]`,
191
- strategy: 'test-id',
192
- };
193
- }
194
- }
195
- const accessibleName = getAccessibleName(element);
196
- const role = getRole(element);
197
- if (role && accessibleName && countRoleNameMatches(role, accessibleName) === 1) {
198
- return {
199
- selector: `role=${role}[name="${escapeForSelectorText(accessibleName)}"]`,
200
- strategy: 'role',
201
- };
202
- }
203
- const labelText = getLabelText(element);
204
- if (labelText) {
205
- return {
206
- selector: `label="${escapeForSelectorText(labelText)}"`,
207
- strategy: 'label',
208
- };
209
- }
210
- const placeholder = normalizeText(element.getAttribute('placeholder'));
211
- if (placeholder) {
212
- return {
213
- selector: `placeholder="${escapeForSelectorText(placeholder)}"`,
214
- strategy: 'placeholder',
215
- };
216
- }
217
- const id = element.getAttribute('id');
218
- if (isLikelyStableIdentifier(id)) {
219
- return {
220
- selector: `css=#${escapeForCss(id)}`,
221
- strategy: 'id',
222
- };
223
- }
224
- const cssSelector = buildCssSelector(element);
225
- if (cssSelector) {
226
- return {
227
- selector: cssSelector,
228
- strategy: 'css',
229
- };
230
- }
231
- return {
232
- selector: buildXPathSelector(element, normalizeText(element.textContent).slice(0, 120)),
233
- strategy: 'xpath',
234
- };
67
+ return ''
68
+ }
69
+ function getAccessibleName(element) {
70
+ const ariaLabel = normalizeText(element.getAttribute('aria-label'))
71
+ if (ariaLabel) {
72
+ return ariaLabel
73
+ }
74
+ const labelledBy = normalizeText(element.getAttribute('aria-labelledby'))
75
+ if (labelledBy) {
76
+ const text = labelledBy
77
+ .split(/\s+/)
78
+ .map(id => {
79
+ var _a
80
+ return ((_a = document.getElementById(id)) === null || _a === void 0 ? void 0 : _a.textContent) || ''
81
+ })
82
+ .join(' ')
83
+ const normalized = normalizeText(text)
84
+ if (normalized) {
85
+ return normalized
86
+ }
87
+ }
88
+ const labelText = getLabelText(element)
89
+ if (labelText) {
90
+ return labelText
91
+ }
92
+ const alt = normalizeText(element.getAttribute('alt'))
93
+ if (alt) {
94
+ return alt
95
+ }
96
+ const title = normalizeText(element.getAttribute('title'))
97
+ if (title) {
98
+ return title
99
+ }
100
+ const placeholder = normalizeText(element.getAttribute('placeholder'))
101
+ if (placeholder) {
102
+ return placeholder
103
+ }
104
+ return normalizeText(element.textContent).slice(0, 240)
105
+ }
106
+ function getRole(element) {
107
+ const explicitRole = normalizeText(element.getAttribute('role'))
108
+ if (explicitRole) {
109
+ return explicitRole
110
+ }
111
+ const tagName = element.tagName.toLowerCase()
112
+ if (tagName === 'button') {
113
+ return 'button'
114
+ }
115
+ if (tagName === 'a' && element.hasAttribute('href')) {
116
+ return 'link'
117
+ }
118
+ if (tagName === 'textarea') {
119
+ return 'textbox'
120
+ }
121
+ if (tagName === 'select') {
122
+ return 'combobox'
123
+ }
124
+ if (tagName === 'img') {
125
+ return 'img'
126
+ }
127
+ if (tagName === 'input') {
128
+ const type = (element.getAttribute('type') || 'text').toLowerCase()
129
+ if (['button', 'submit', 'reset'].includes(type)) {
130
+ return 'button'
131
+ }
132
+ if (type === 'checkbox') {
133
+ return 'checkbox'
134
+ }
135
+ if (type === 'radio') {
136
+ return 'radio'
137
+ }
138
+ return 'textbox'
139
+ }
140
+ return ''
141
+ }
142
+ function countRoleNameMatches(role, accessibleName) {
143
+ let count = 0
144
+ for (const candidate of Array.from(document.querySelectorAll('*'))) {
145
+ if (getRole(candidate) !== role) {
146
+ continue
147
+ }
148
+ if (getAccessibleName(candidate) !== accessibleName) {
149
+ continue
150
+ }
151
+ count += 1
152
+ if (count > 1) {
153
+ break
154
+ }
155
+ }
156
+ return count
157
+ }
158
+ function buildCssSelector(element) {
159
+ const tagName = element.tagName.toLowerCase()
160
+ const name = element.getAttribute('name')
161
+ const type = element.getAttribute('type')
162
+ const ariaLabel = element.getAttribute('aria-label')
163
+ const parts = [tagName]
164
+ if (isLikelyStableIdentifier(name)) {
165
+ parts.push(`[name="${escapeForSelectorText(name)}"]`)
235
166
  }
236
- function buildPreviewPayload(element) {
237
- const selection = buildPrimarySelector(element);
167
+ if (isLikelyStableIdentifier(type)) {
168
+ parts.push(`[type="${escapeForSelectorText(type)}"]`)
169
+ }
170
+ if (isLikelyStableIdentifier(ariaLabel)) {
171
+ parts.push(`[aria-label="${escapeForSelectorText(ariaLabel)}"]`)
172
+ }
173
+ if (parts.length === 1) {
174
+ const stableClasses = Array.from(element.classList || [])
175
+ .filter(isLikelyStableIdentifier)
176
+ .slice(0, 2)
177
+ if (stableClasses.length > 0) {
178
+ parts.push(...stableClasses.map(className => `.${escapeForCss(className)}`))
179
+ }
180
+ }
181
+ return parts.length > 1 ? `css=${parts.join('')}` : ''
182
+ }
183
+ function buildXPathSelector(element, text) {
184
+ const tagName = element.tagName.toLowerCase()
185
+ const id = element.getAttribute('id')
186
+ if (id) {
187
+ return `xpath=//*[@id="${id.replace(/"/g, '\\"')}"]`
188
+ }
189
+ if (text) {
190
+ return `xpath=//${tagName}[normalize-space()="${text.replace(/"/g, '\\"')}"]`
191
+ }
192
+ return `xpath=//${tagName}`
193
+ }
194
+ function buildPrimarySelector(element) {
195
+ for (const attributeName of ['data-testid', 'data-test', 'data-qa']) {
196
+ const value = element.getAttribute(attributeName)
197
+ if (isLikelyStableIdentifier(value)) {
238
198
  return {
239
- selector: selection.selector,
240
- strategy: selection.strategy,
241
- currentUrl: window.location.href,
242
- pathname: normalizeRoute(window.location.href),
243
- pageTitle: document.title || '',
244
- tagName: element.tagName.toLowerCase(),
245
- text: normalizeText(element.textContent).slice(0, 240) || undefined,
246
- accessibleName: getAccessibleName(element) || undefined,
247
- };
248
- }
249
- function clearHover() {
250
- if (state.hoveredElement instanceof Element) {
251
- state.hoveredElement.classList.remove(HOVER_CLASS);
199
+ selector: `css=[${attributeName}="${escapeForSelectorText(value)}"]`,
200
+ strategy: 'test-id',
252
201
  }
253
- state.hoveredElement = null;
202
+ }
254
203
  }
255
- function setPicking(enabled) {
256
- state.picking = enabled;
257
- document.documentElement.classList.toggle(ACTIVE_CLASS, enabled);
258
- if (!enabled) {
259
- clearHover();
260
- }
261
- render();
204
+ const accessibleName = getAccessibleName(element)
205
+ const role = getRole(element)
206
+ if (role && accessibleName && countRoleNameMatches(role, accessibleName) === 1) {
207
+ return {
208
+ selector: `role=${role}[name="${escapeForSelectorText(accessibleName)}"]`,
209
+ strategy: 'role',
210
+ }
262
211
  }
263
- function resolveElement(target) {
264
- if (target instanceof Element) {
265
- return target;
266
- }
267
- if (target instanceof Node) {
268
- return target.parentElement;
269
- }
270
- return null;
212
+ const labelText = getLabelText(element)
213
+ if (labelText) {
214
+ return {
215
+ selector: `label="${escapeForSelectorText(labelText)}"`,
216
+ strategy: 'label',
217
+ }
271
218
  }
272
- function getMountTarget() {
273
- var _a, _b;
274
- return (_b = (_a = document.body) !== null && _a !== void 0 ? _a : document.documentElement) !== null && _b !== void 0 ? _b : null;
219
+ const placeholder = normalizeText(element.getAttribute('placeholder'))
220
+ if (placeholder) {
221
+ return {
222
+ selector: `placeholder="${escapeForSelectorText(placeholder)}"`,
223
+ strategy: 'placeholder',
224
+ }
275
225
  }
276
- function ensureStyle() {
277
- if (!document.documentElement) {
278
- return;
279
- }
280
- if (document.getElementById(STYLE_ID)) {
281
- return;
282
- }
283
- const style = document.createElement('style');
284
- style.id = STYLE_ID;
285
- style.textContent = `
226
+ const id = element.getAttribute('id')
227
+ if (isLikelyStableIdentifier(id)) {
228
+ return {
229
+ selector: `css=#${escapeForCss(id)}`,
230
+ strategy: 'id',
231
+ }
232
+ }
233
+ const cssSelector = buildCssSelector(element)
234
+ if (cssSelector) {
235
+ return {
236
+ selector: cssSelector,
237
+ strategy: 'css',
238
+ }
239
+ }
240
+ return {
241
+ selector: buildXPathSelector(element, normalizeText(element.textContent).slice(0, 120)),
242
+ strategy: 'xpath',
243
+ }
244
+ }
245
+ function buildPreviewPayload(element) {
246
+ const selection = buildPrimarySelector(element)
247
+ return {
248
+ selector: selection.selector,
249
+ strategy: selection.strategy,
250
+ currentUrl: window.location.href,
251
+ pathname: normalizeRoute(window.location.href),
252
+ pageTitle: document.title || '',
253
+ tagName: element.tagName.toLowerCase(),
254
+ text: normalizeText(element.textContent).slice(0, 240) || undefined,
255
+ accessibleName: getAccessibleName(element) || undefined,
256
+ }
257
+ }
258
+ function clearHover() {
259
+ if (state.hoveredElement instanceof Element) {
260
+ state.hoveredElement.classList.remove(HOVER_CLASS)
261
+ }
262
+ state.hoveredElement = null
263
+ }
264
+ function setPicking(enabled) {
265
+ state.picking = enabled
266
+ document.documentElement.classList.toggle(ACTIVE_CLASS, enabled)
267
+ if (!enabled) {
268
+ clearHover()
269
+ }
270
+ render()
271
+ }
272
+ function resolveElement(target) {
273
+ if (target instanceof Element) {
274
+ return target
275
+ }
276
+ if (target instanceof Node) {
277
+ return target.parentElement
278
+ }
279
+ return null
280
+ }
281
+ function getMountTarget() {
282
+ var _a, _b
283
+ return (_b = (_a = document.body) !== null && _a !== void 0 ? _a : document.documentElement) !== null &&
284
+ _b !== void 0
285
+ ? _b
286
+ : null
287
+ }
288
+ function ensureStyle() {
289
+ if (!document.documentElement) {
290
+ return
291
+ }
292
+ if (document.getElementById(STYLE_ID)) {
293
+ return
294
+ }
295
+ const style = document.createElement('style')
296
+ style.id = STYLE_ID
297
+ style.textContent = `
286
298
  .${HOVER_CLASS} {
287
299
  outline: 2px solid #0f766e !important;
288
300
  outline-offset: 2px !important;
@@ -400,216 +412,249 @@ export function installLocatorPickerOverlay() {
400
412
  font-size: 12px;
401
413
  line-height: 1.4;
402
414
  }
403
- `;
404
- document.documentElement.appendChild(style);
415
+ `
416
+ document.documentElement.appendChild(style)
417
+ }
418
+ function ensureRoot() {
419
+ var _a
420
+ const mountTarget = getMountTarget()
421
+ if (!mountTarget) {
422
+ return
405
423
  }
406
- function ensureRoot() {
407
- var _a;
408
- const mountTarget = getMountTarget();
409
- if (!mountTarget) {
410
- return;
411
- }
412
- if ((_a = state.elements.root) === null || _a === void 0 ? void 0 : _a.isConnected) {
413
- return;
414
- }
415
- const root = document.createElement('div');
416
- root.id = ROOT_ID;
417
- root.setAttribute('role', 'dialog');
418
- root.setAttribute('aria-label', 'Appraise locator picker');
419
- const heading = document.createElement('p');
420
- heading.className = 'appraise-picker-heading';
421
- heading.textContent = 'Appraise Locator Picker';
422
- const helperText = document.createElement('p');
423
- helperText.className = 'appraise-picker-copy';
424
- const statusText = document.createElement('p');
425
- statusText.className = 'appraise-picker-status';
426
- const previewCard = document.createElement('div');
427
- previewCard.className = 'appraise-picker-card';
428
- const selectorLabel = document.createElement('span');
429
- selectorLabel.className = 'appraise-picker-label';
430
- selectorLabel.textContent = 'Selector';
431
- const selectorValue = document.createElement('pre');
432
- selectorValue.className = 'appraise-picker-value';
433
- const metadataLabel = document.createElement('span');
434
- metadataLabel.className = 'appraise-picker-label';
435
- metadataLabel.textContent = 'Page';
436
- const metadataValue = document.createElement('p');
437
- metadataValue.className = 'appraise-picker-value';
438
- previewCard.appendChild(selectorLabel);
439
- previewCard.appendChild(selectorValue);
440
- previewCard.appendChild(metadataLabel);
441
- previewCard.appendChild(metadataValue);
442
- const errorValue = document.createElement('p');
443
- errorValue.className = 'appraise-picker-error';
444
- const actions = document.createElement('div');
445
- actions.className = 'appraise-picker-actions';
446
- const startButton = document.createElement('button');
447
- startButton.className = 'appraise-picker-primary';
448
- startButton.type = 'button';
449
- startButton.textContent = 'Start picking';
450
- startButton.addEventListener('click', event => {
451
- stopOverlayEvent(event);
452
- state.submitting = false;
453
- state.preview = null;
454
- state.error = '';
455
- setPicking(true);
456
- });
457
- const useButton = document.createElement('button');
458
- useButton.className = 'appraise-picker-primary';
459
- useButton.type = 'button';
460
- useButton.textContent = 'Use selector';
461
- useButton.addEventListener('click', event => {
462
- stopOverlayEvent(event);
463
- if (!state.preview || state.submitting || !globalState.__appraiseLocatorPickerConfirm) {
464
- return;
465
- }
466
- state.submitting = true;
467
- state.error = '';
468
- render();
469
- void globalState
470
- .__appraiseLocatorPickerConfirm(state.preview)
471
- .catch(error => {
472
- state.error = error instanceof Error ? error.message : 'Unable to use selector.';
473
- })
474
- .finally(() => {
475
- state.submitting = false;
476
- render();
477
- });
478
- });
479
- const pickAgainButton = document.createElement('button');
480
- pickAgainButton.className = 'appraise-picker-secondary';
481
- pickAgainButton.type = 'button';
482
- pickAgainButton.textContent = 'Pick again';
483
- pickAgainButton.addEventListener('click', event => {
484
- stopOverlayEvent(event);
485
- state.submitting = false;
486
- state.preview = null;
487
- state.error = '';
488
- setPicking(true);
489
- });
490
- const cancelButton = document.createElement('button');
491
- cancelButton.className = 'appraise-picker-secondary';
492
- cancelButton.type = 'button';
493
- cancelButton.textContent = 'Cancel';
494
- cancelButton.addEventListener('click', event => {
495
- stopOverlayEvent(event);
496
- if (state.submitting || !globalState.__appraiseLocatorPickerCancel) {
497
- return;
498
- }
499
- state.submitting = true;
500
- render();
501
- void globalState.__appraiseLocatorPickerCancel().catch(() => undefined);
502
- });
503
- actions.append(startButton, useButton, pickAgainButton, cancelButton);
504
- root.append(heading, helperText, statusText, previewCard, errorValue, actions);
505
- root.addEventListener('pointerdown', event => event.stopPropagation());
506
- root.addEventListener('click', event => event.stopPropagation());
507
- mountTarget.appendChild(root);
508
- state.elements = {
509
- root,
510
- startButton,
511
- useButton,
512
- pickAgainButton,
513
- cancelButton,
514
- helperText,
515
- statusText,
516
- selectorValue,
517
- metadataValue,
518
- errorValue,
519
- };
520
- }
521
- function render() {
522
- var _a, _b;
523
- ensureStyle();
524
- ensureRoot();
525
- document.documentElement.classList.toggle(ACTIVE_CLASS, state.picking);
526
- if (state.hoveredElement && !state.hoveredElement.isConnected) {
527
- state.hoveredElement = null;
528
- }
529
- const { helperText, statusText, selectorValue, metadataValue, errorValue, startButton, useButton, pickAgainButton, cancelButton } = state.elements;
530
- if (!helperText || !statusText || !selectorValue || !metadataValue || !errorValue || !startButton || !useButton || !pickAgainButton || !cancelButton) {
531
- return;
532
- }
533
- helperText.textContent = state.preview
534
- ? 'Review the generated selector, then confirm it or pick a different element.'
535
- : 'Start picking, hover to highlight, then click one element to preview a single generated selector.';
536
- if (state.picking) {
537
- statusText.textContent = 'Picking mode is on';
538
- }
539
- else if (state.preview) {
540
- statusText.textContent = 'Selection ready';
541
- }
542
- else {
543
- statusText.textContent = 'Waiting to start';
544
- }
545
- selectorValue.textContent = (_b = (_a = state.preview) === null || _a === void 0 ? void 0 : _a.selector) !== null && _b !== void 0 ? _b : 'No selector picked yet.';
546
- metadataValue.textContent = state.preview
547
- ? [state.preview.pageTitle || '(untitled page)', state.preview.currentUrl].filter(Boolean).join('\n')
548
- : 'The selected page metadata will appear here.';
549
- errorValue.textContent = state.error;
550
- startButton.style.display = state.preview ? 'none' : 'inline-flex';
551
- useButton.style.display = state.preview ? 'inline-flex' : 'none';
552
- pickAgainButton.style.display = state.preview ? 'inline-flex' : 'none';
553
- useButton.disabled = !state.preview || state.submitting;
554
- pickAgainButton.disabled = state.submitting;
555
- cancelButton.disabled = state.submitting;
556
- }
557
- function installOrRender() {
558
- render();
559
- }
560
- if (!globalState.__appraiseLocatorPickerDomListenersBound) {
561
- globalState.__appraiseLocatorPickerDomListenersBound = true;
562
- document.addEventListener('DOMContentLoaded', installOrRender);
563
- window.addEventListener('load', installOrRender);
564
- }
565
- if (globalState.__appraiseLocatorPickerInstalled) {
566
- installOrRender();
567
- return;
568
- }
569
- globalState.__appraiseLocatorPickerInstalled = true;
570
- document.addEventListener('mouseover', event => {
571
- if (!state.picking) {
572
- return;
573
- }
574
- const target = resolveElement(event.target);
575
- if (!target || target.closest(`#${ROOT_ID}`)) {
576
- return;
577
- }
578
- clearHover();
579
- target.classList.add(HOVER_CLASS);
580
- state.hoveredElement = target;
581
- }, true);
582
- document.addEventListener('mouseout', event => {
583
- if (!state.picking) {
584
- return;
585
- }
586
- const target = resolveElement(event.target);
587
- if (target instanceof Element) {
588
- target.classList.remove(HOVER_CLASS);
589
- }
590
- }, true);
591
- document.addEventListener('click', event => {
592
- if (!state.picking) {
593
- return;
594
- }
595
- const target = resolveElement(event.target);
596
- if (!target || target.closest(`#${ROOT_ID}`)) {
597
- return;
598
- }
599
- event.preventDefault();
600
- event.stopPropagation();
601
- event.stopImmediatePropagation();
602
- void Promise.resolve()
603
- .then(() => {
604
- const preview = buildPreviewPayload(target);
605
- state.preview = preview;
606
- state.error = '';
607
- setPicking(false);
424
+ if ((_a = state.elements.root) === null || _a === void 0 ? void 0 : _a.isConnected) {
425
+ return
426
+ }
427
+ const root = document.createElement('div')
428
+ root.id = ROOT_ID
429
+ root.setAttribute('role', 'dialog')
430
+ root.setAttribute('aria-label', 'Appraise locator picker')
431
+ const heading = document.createElement('p')
432
+ heading.className = 'appraise-picker-heading'
433
+ heading.textContent = 'Appraise Locator Picker'
434
+ const helperText = document.createElement('p')
435
+ helperText.className = 'appraise-picker-copy'
436
+ const statusText = document.createElement('p')
437
+ statusText.className = 'appraise-picker-status'
438
+ const previewCard = document.createElement('div')
439
+ previewCard.className = 'appraise-picker-card'
440
+ const selectorLabel = document.createElement('span')
441
+ selectorLabel.className = 'appraise-picker-label'
442
+ selectorLabel.textContent = 'Selector'
443
+ const selectorValue = document.createElement('pre')
444
+ selectorValue.className = 'appraise-picker-value'
445
+ const metadataLabel = document.createElement('span')
446
+ metadataLabel.className = 'appraise-picker-label'
447
+ metadataLabel.textContent = 'Page'
448
+ const metadataValue = document.createElement('p')
449
+ metadataValue.className = 'appraise-picker-value'
450
+ previewCard.appendChild(selectorLabel)
451
+ previewCard.appendChild(selectorValue)
452
+ previewCard.appendChild(metadataLabel)
453
+ previewCard.appendChild(metadataValue)
454
+ const errorValue = document.createElement('p')
455
+ errorValue.className = 'appraise-picker-error'
456
+ const actions = document.createElement('div')
457
+ actions.className = 'appraise-picker-actions'
458
+ const startButton = document.createElement('button')
459
+ startButton.className = 'appraise-picker-primary'
460
+ startButton.type = 'button'
461
+ startButton.textContent = 'Start picking'
462
+ startButton.addEventListener('click', event => {
463
+ stopOverlayEvent(event)
464
+ state.submitting = false
465
+ state.preview = null
466
+ state.error = ''
467
+ setPicking(true)
468
+ })
469
+ const useButton = document.createElement('button')
470
+ useButton.className = 'appraise-picker-primary'
471
+ useButton.type = 'button'
472
+ useButton.textContent = 'Use selector'
473
+ useButton.addEventListener('click', event => {
474
+ stopOverlayEvent(event)
475
+ if (!state.preview || state.submitting || !globalState.__appraiseLocatorPickerConfirm) {
476
+ return
477
+ }
478
+ state.submitting = true
479
+ state.error = ''
480
+ render()
481
+ void globalState
482
+ .__appraiseLocatorPickerConfirm(state.preview)
483
+ .catch(error => {
484
+ state.error = error instanceof Error ? error.message : 'Unable to use selector.'
485
+ })
486
+ .finally(() => {
487
+ state.submitting = false
488
+ render()
489
+ })
490
+ })
491
+ const pickAgainButton = document.createElement('button')
492
+ pickAgainButton.className = 'appraise-picker-secondary'
493
+ pickAgainButton.type = 'button'
494
+ pickAgainButton.textContent = 'Pick again'
495
+ pickAgainButton.addEventListener('click', event => {
496
+ stopOverlayEvent(event)
497
+ state.submitting = false
498
+ state.preview = null
499
+ state.error = ''
500
+ setPicking(true)
501
+ })
502
+ const cancelButton = document.createElement('button')
503
+ cancelButton.className = 'appraise-picker-secondary'
504
+ cancelButton.type = 'button'
505
+ cancelButton.textContent = 'Cancel'
506
+ cancelButton.addEventListener('click', event => {
507
+ stopOverlayEvent(event)
508
+ if (state.submitting || !globalState.__appraiseLocatorPickerCancel) {
509
+ return
510
+ }
511
+ state.submitting = true
512
+ render()
513
+ void globalState.__appraiseLocatorPickerCancel().catch(() => undefined)
514
+ })
515
+ actions.append(startButton, useButton, pickAgainButton, cancelButton)
516
+ root.append(heading, helperText, statusText, previewCard, errorValue, actions)
517
+ root.addEventListener('pointerdown', event => event.stopPropagation())
518
+ root.addEventListener('click', event => event.stopPropagation())
519
+ mountTarget.appendChild(root)
520
+ state.elements = {
521
+ root,
522
+ startButton,
523
+ useButton,
524
+ pickAgainButton,
525
+ cancelButton,
526
+ helperText,
527
+ statusText,
528
+ selectorValue,
529
+ metadataValue,
530
+ errorValue,
531
+ }
532
+ }
533
+ function render() {
534
+ var _a, _b
535
+ ensureStyle()
536
+ ensureRoot()
537
+ document.documentElement.classList.toggle(ACTIVE_CLASS, state.picking)
538
+ if (state.hoveredElement && !state.hoveredElement.isConnected) {
539
+ state.hoveredElement = null
540
+ }
541
+ const {
542
+ helperText,
543
+ statusText,
544
+ selectorValue,
545
+ metadataValue,
546
+ errorValue,
547
+ startButton,
548
+ useButton,
549
+ pickAgainButton,
550
+ cancelButton,
551
+ } = state.elements
552
+ if (
553
+ !helperText ||
554
+ !statusText ||
555
+ !selectorValue ||
556
+ !metadataValue ||
557
+ !errorValue ||
558
+ !startButton ||
559
+ !useButton ||
560
+ !pickAgainButton ||
561
+ !cancelButton
562
+ ) {
563
+ return
564
+ }
565
+ helperText.textContent = state.preview
566
+ ? 'Review the generated selector, then confirm it or pick a different element.'
567
+ : 'Start picking, hover to highlight, then click one element to preview a single generated selector.'
568
+ if (state.picking) {
569
+ statusText.textContent = 'Picking mode is on'
570
+ } else if (state.preview) {
571
+ statusText.textContent = 'Selection ready'
572
+ } else {
573
+ statusText.textContent = 'Waiting to start'
574
+ }
575
+ selectorValue.textContent =
576
+ (_b = (_a = state.preview) === null || _a === void 0 ? void 0 : _a.selector) !== null && _b !== void 0
577
+ ? _b
578
+ : 'No selector picked yet.'
579
+ metadataValue.textContent = state.preview
580
+ ? [state.preview.pageTitle || '(untitled page)', state.preview.currentUrl].filter(Boolean).join('\n')
581
+ : 'The selected page metadata will appear here.'
582
+ errorValue.textContent = state.error
583
+ startButton.style.display = state.preview ? 'none' : 'inline-flex'
584
+ useButton.style.display = state.preview ? 'inline-flex' : 'none'
585
+ pickAgainButton.style.display = state.preview ? 'inline-flex' : 'none'
586
+ useButton.disabled = !state.preview || state.submitting
587
+ pickAgainButton.disabled = state.submitting
588
+ cancelButton.disabled = state.submitting
589
+ }
590
+ function installOrRender() {
591
+ render()
592
+ }
593
+ if (!globalState.__appraiseLocatorPickerDomListenersBound) {
594
+ globalState.__appraiseLocatorPickerDomListenersBound = true
595
+ document.addEventListener('DOMContentLoaded', installOrRender)
596
+ window.addEventListener('load', installOrRender)
597
+ }
598
+ if (globalState.__appraiseLocatorPickerInstalled) {
599
+ installOrRender()
600
+ return
601
+ }
602
+ globalState.__appraiseLocatorPickerInstalled = true
603
+ document.addEventListener(
604
+ 'mouseover',
605
+ event => {
606
+ if (!state.picking) {
607
+ return
608
+ }
609
+ const target = resolveElement(event.target)
610
+ if (!target || target.closest(`#${ROOT_ID}`)) {
611
+ return
612
+ }
613
+ clearHover()
614
+ target.classList.add(HOVER_CLASS)
615
+ state.hoveredElement = target
616
+ },
617
+ true,
618
+ )
619
+ document.addEventListener(
620
+ 'mouseout',
621
+ event => {
622
+ if (!state.picking) {
623
+ return
624
+ }
625
+ const target = resolveElement(event.target)
626
+ if (target instanceof Element) {
627
+ target.classList.remove(HOVER_CLASS)
628
+ }
629
+ },
630
+ true,
631
+ )
632
+ document.addEventListener(
633
+ 'click',
634
+ event => {
635
+ if (!state.picking) {
636
+ return
637
+ }
638
+ const target = resolveElement(event.target)
639
+ if (!target || target.closest(`#${ROOT_ID}`)) {
640
+ return
641
+ }
642
+ event.preventDefault()
643
+ event.stopPropagation()
644
+ event.stopImmediatePropagation()
645
+ void Promise.resolve()
646
+ .then(() => {
647
+ const preview = buildPreviewPayload(target)
648
+ state.preview = preview
649
+ state.error = ''
650
+ setPicking(false)
651
+ })
652
+ .catch(error => {
653
+ state.error = error instanceof Error ? error.message : 'Unable to inspect that element.'
654
+ setPicking(false)
608
655
  })
609
- .catch(error => {
610
- state.error = error instanceof Error ? error.message : 'Unable to inspect that element.';
611
- setPicking(false);
612
- });
613
- }, true);
614
- installOrRender();
656
+ },
657
+ true,
658
+ )
659
+ installOrRender()
615
660
  }