@morscherlab/mint-sdk 1.0.0-beta.2 → 1.0.0-beta.3

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 (399) hide show
  1. package/README.md +218 -6
  2. package/dist/__tests__/components/ActionItem.test.d.ts +1 -0
  3. package/dist/__tests__/components/AppAvatarMenu.test.d.ts +1 -0
  4. package/dist/__tests__/components/AppPageSelector.test.d.ts +1 -0
  5. package/dist/__tests__/components/AppPillNav.test.d.ts +1 -0
  6. package/dist/__tests__/components/AppPluginSwitcher.test.d.ts +1 -0
  7. package/dist/__tests__/components/AppToastContainer.test.d.ts +1 -0
  8. package/dist/__tests__/components/BaseRadioGroup.test.d.ts +1 -0
  9. package/dist/__tests__/components/BaseSelect.test.d.ts +1 -0
  10. package/dist/__tests__/components/BaseTabs.test.d.ts +1 -0
  11. package/dist/__tests__/components/BatchProgressList.test.d.ts +1 -0
  12. package/dist/__tests__/components/BioTemplateExperimentWorkspaceView.test.d.ts +1 -0
  13. package/dist/__tests__/components/BioTemplatePackWorkspaceView.test.d.ts +1 -0
  14. package/dist/__tests__/components/BioTemplatePresetWorkspaceView.test.d.ts +1 -0
  15. package/dist/__tests__/components/BioTemplateRenderer.test.d.ts +1 -0
  16. package/dist/__tests__/components/Breadcrumb.test.d.ts +1 -0
  17. package/dist/__tests__/components/CalendarGridPanel.test.d.ts +1 -0
  18. package/dist/__tests__/components/ConcentrationInput.test.d.ts +1 -0
  19. package/dist/__tests__/components/ControlWorkspaceView.test.d.ts +1 -0
  20. package/dist/__tests__/components/DatePicker.test.d.ts +1 -0
  21. package/dist/__tests__/components/DateTimePicker.test.d.ts +1 -0
  22. package/dist/__tests__/components/EmptyState.test.d.ts +1 -0
  23. package/dist/__tests__/components/ExperimentPopover.test.d.ts +1 -0
  24. package/dist/__tests__/components/FormBuilder.test.d.ts +1 -0
  25. package/dist/__tests__/components/FormCompatibility.test.d.ts +1 -0
  26. package/dist/__tests__/components/GroupAssigner.test.d.ts +1 -0
  27. package/dist/__tests__/components/GroupingModal.test.d.ts +1 -0
  28. package/dist/__tests__/components/MultiSelect.test.d.ts +1 -0
  29. package/dist/__tests__/components/ProtocolStepEditor.test.d.ts +1 -0
  30. package/dist/__tests__/components/ReagentList.test.d.ts +1 -0
  31. package/dist/__tests__/components/SampleHierarchyTree.test.d.ts +1 -0
  32. package/dist/__tests__/components/SampleSelector.test.d.ts +1 -0
  33. package/dist/__tests__/components/SegmentedControl.test.d.ts +1 -0
  34. package/dist/__tests__/components/SettingsButton.test.d.ts +1 -0
  35. package/dist/__tests__/components/SettingsModal.test.d.ts +1 -0
  36. package/dist/__tests__/components/TagsInput.test.d.ts +1 -0
  37. package/dist/__tests__/components/ThemeToggle.test.d.ts +1 -0
  38. package/dist/__tests__/components/TimePicker.test.d.ts +1 -0
  39. package/dist/__tests__/composables/useBioTemplatePackWorkspace.test.d.ts +1 -0
  40. package/dist/__tests__/composables/useBioTemplatePresetWorkspace.test.d.ts +1 -0
  41. package/dist/__tests__/composables/useBioTemplateWorkspace.test.d.ts +1 -0
  42. package/dist/__tests__/composables/useCalendarGrid.test.d.ts +1 -0
  43. package/dist/__tests__/composables/useControlSchema.test.d.ts +1 -0
  44. package/dist/__tests__/composables/useDebouncedWatch.test.d.ts +1 -0
  45. package/dist/__tests__/composables/useDropdownState.test.d.ts +1 -0
  46. package/dist/__tests__/composables/useEventListener.test.d.ts +1 -0
  47. package/dist/__tests__/composables/useExpansionSet.test.d.ts +1 -0
  48. package/dist/__tests__/composables/useExperimentData.test.d.ts +1 -0
  49. package/dist/__tests__/composables/useExperimentSelector.test.d.ts +1 -0
  50. package/dist/__tests__/composables/useGroupAssignment.test.d.ts +1 -0
  51. package/dist/__tests__/composables/useListSelection.test.d.ts +1 -0
  52. package/dist/__tests__/composables/usePluginClient.test.d.ts +1 -0
  53. package/dist/__tests__/composables/usePluginConfig.test.d.ts +1 -0
  54. package/dist/__tests__/composables/useRequestSyncState.test.d.ts +1 -0
  55. package/dist/__tests__/composables/useSampleGroups.test.d.ts +1 -0
  56. package/dist/__tests__/composables/useSelectionLimit.test.d.ts +1 -0
  57. package/dist/__tests__/composables/useSortedItems.test.d.ts +1 -0
  58. package/dist/__tests__/composables/useTemplateCollection.test.d.ts +1 -0
  59. package/dist/__tests__/composables/useTextSearch.test.d.ts +1 -0
  60. package/dist/__tests__/composables/useTheme.test.d.ts +1 -0
  61. package/dist/__tests__/composables/useTimeUtils.test.d.ts +1 -0
  62. package/dist/__tests__/docs/frontendDocsCatalog.test.d.ts +1 -0
  63. package/dist/__tests__/templates/templates.test.d.ts +1 -0
  64. package/dist/{auth-DsI0rQ7_.js → auth-QQj2kkze.js} +12 -5
  65. package/dist/auth-QQj2kkze.js.map +1 -0
  66. package/dist/components/ActionItem.vue.d.ts +32 -0
  67. package/dist/components/AppAvatarMenu.vue.d.ts +2 -7
  68. package/dist/components/AppPageSelector.vue.d.ts +3 -6
  69. package/dist/components/AppPillNav.vue.d.ts +2 -2
  70. package/dist/components/AppSidebar.vue.d.ts +56 -3
  71. package/dist/components/AppToastContainer.vue.d.ts +2 -0
  72. package/dist/components/AppTopBar.vue.d.ts +41 -10
  73. package/dist/components/BaseButton.vue.d.ts +1 -1
  74. package/dist/components/BaseCheckbox.vue.d.ts +1 -1
  75. package/dist/components/BaseInput.vue.d.ts +2 -2
  76. package/dist/components/BasePill.vue.d.ts +2 -2
  77. package/dist/components/BaseRadioGroup.vue.d.ts +3 -3
  78. package/dist/components/BaseSelect.vue.d.ts +3 -3
  79. package/dist/components/BaseTabs.vue.d.ts +2 -2
  80. package/dist/components/BaseTextarea.vue.d.ts +1 -1
  81. package/dist/components/BaseToggle.vue.d.ts +1 -1
  82. package/dist/components/BioTemplateExperimentWorkspaceView.vue.d.ts +117 -0
  83. package/dist/components/BioTemplatePackWorkspaceView.vue.d.ts +92 -0
  84. package/dist/components/BioTemplatePresetWorkspaceView.vue.d.ts +82 -0
  85. package/dist/components/BioTemplateRenderer.vue.d.ts +29 -0
  86. package/dist/components/Breadcrumb.vue.d.ts +2 -2
  87. package/dist/components/Calendar.vue.d.ts +1 -1
  88. package/dist/components/CalendarGridPanel.vue.d.ts +25 -0
  89. package/dist/components/CollapsibleCard.vue.d.ts +1 -1
  90. package/dist/components/ConcentrationInput.vue.d.ts +2 -2
  91. package/dist/components/ConfirmDialog.vue.d.ts +2 -2
  92. package/dist/components/ControlWorkspaceView.vue.d.ts +130 -0
  93. package/dist/components/DatePicker.vue.d.ts +1 -1
  94. package/dist/components/DateTimePicker.vue.d.ts +3 -3
  95. package/dist/components/Divider.vue.d.ts +1 -1
  96. package/dist/components/DropdownButton.vue.d.ts +3 -3
  97. package/dist/components/EmptyState.vue.d.ts +1 -2
  98. package/dist/components/ExperimentDataViewer.vue.d.ts +1 -1
  99. package/dist/components/ExperimentTimeline.vue.d.ts +2 -2
  100. package/dist/components/FileUploader.vue.d.ts +1 -1
  101. package/dist/components/FitPanel.vue.d.ts +1 -1
  102. package/dist/components/FormActions.vue.d.ts +4 -4
  103. package/dist/components/FormBuilder.vue.d.ts +22 -8
  104. package/dist/components/FormFieldRenderer.vue.d.ts +7 -10
  105. package/dist/components/FormSection.vue.d.ts +11 -24
  106. package/dist/components/FormulaInput.vue.d.ts +2 -2
  107. package/dist/components/MoleculeInput.vue.d.ts +2 -2
  108. package/dist/components/MultiSelect.vue.d.ts +3 -3
  109. package/dist/components/NumberInput.vue.d.ts +1 -1
  110. package/dist/components/ProgressBar.vue.d.ts +1 -1
  111. package/dist/components/ProtocolStepEditor.vue.d.ts +3 -1
  112. package/dist/components/RackEditor.vue.d.ts +2 -2
  113. package/dist/components/SampleLegend.vue.d.ts +2 -2
  114. package/dist/components/ScheduleCalendar.vue.d.ts +2 -2
  115. package/dist/components/SegmentedControl.vue.d.ts +2 -2
  116. package/dist/components/SequenceInput.vue.d.ts +3 -3
  117. package/dist/components/SettingsButton.vue.d.ts +2 -2
  118. package/dist/components/SettingsModal.vue.d.ts +13 -5
  119. package/dist/components/StatusIndicator.vue.d.ts +1 -1
  120. package/dist/components/TagsInput.vue.d.ts +3 -2
  121. package/dist/components/TimePicker.vue.d.ts +3 -3
  122. package/dist/components/TimeRangeInput.vue.d.ts +1 -1
  123. package/dist/components/UnitInput.vue.d.ts +2 -2
  124. package/dist/components/WellPlate.vue.d.ts +8 -8
  125. package/dist/components/index.d.ts +11 -1
  126. package/dist/components/index.js +3 -3
  127. package/dist/components/internal/FormFieldRendererInternal.vue.d.ts +31 -0
  128. package/dist/components/internal/FormSectionRenderer.vue.d.ts +43 -0
  129. package/dist/{components-_XqPEhP9.js → components-D_Sr0adg.js} +7290 -6518
  130. package/dist/components-D_Sr0adg.js.map +1 -0
  131. package/dist/composables/index.d.ts +21 -2
  132. package/dist/composables/index.js +4 -3
  133. package/dist/composables/platformContextHelpers.d.ts +14 -0
  134. package/dist/composables/useBioTemplateComponents.d.ts +20 -0
  135. package/dist/composables/useBioTemplateControls.d.ts +6 -0
  136. package/dist/composables/useBioTemplatePackWorkspace.d.ts +45 -0
  137. package/dist/composables/useBioTemplatePresetWorkspace.d.ts +74 -0
  138. package/dist/composables/useBioTemplateWorkspace.d.ts +50 -0
  139. package/dist/composables/useCalendarGrid.d.ts +26 -0
  140. package/dist/composables/useControlSchema.d.ts +321 -0
  141. package/dist/composables/useDebouncedWatch.d.ts +20 -0
  142. package/dist/composables/useDropdownState.d.ts +19 -0
  143. package/dist/composables/useEventListener.d.ts +13 -0
  144. package/dist/composables/useExpansionSet.d.ts +21 -0
  145. package/dist/composables/useExperimentData.d.ts +10 -0
  146. package/dist/composables/useExperimentSave.d.ts +31 -2
  147. package/dist/composables/useExperimentSelector.d.ts +20 -0
  148. package/dist/composables/useForm.d.ts +2 -0
  149. package/dist/composables/useGroupAssignment.d.ts +31 -0
  150. package/dist/composables/useListSelection.d.ts +35 -0
  151. package/dist/composables/usePlatformContext.d.ts +21 -3
  152. package/dist/composables/usePluginApi.d.ts +7 -14
  153. package/dist/composables/usePluginClient.d.ts +109 -0
  154. package/dist/composables/usePluginConfig.d.ts +12 -0
  155. package/dist/composables/useRequestSyncState.d.ts +34 -0
  156. package/dist/composables/useSampleGroups.d.ts +32 -0
  157. package/dist/composables/useSelectionLimit.d.ts +17 -0
  158. package/dist/composables/useSortedItems.d.ts +32 -0
  159. package/dist/composables/useTemplateCollection.d.ts +58 -0
  160. package/dist/composables/useTextSearch.d.ts +18 -0
  161. package/dist/composables/useTimeUtils.d.ts +8 -0
  162. package/dist/{composables-tiZqLu1M.js → composables-C3dpXQN5.js} +228 -146
  163. package/dist/composables-C3dpXQN5.js.map +1 -0
  164. package/dist/index.d.ts +12 -3
  165. package/dist/index.js +6 -5
  166. package/dist/install.d.ts +7 -2
  167. package/dist/install.js +2 -2
  168. package/dist/install.js.map +1 -1
  169. package/dist/stores/index.js +1 -1
  170. package/dist/stores/settings.d.ts +4 -1
  171. package/dist/styles.css +5235 -5977
  172. package/dist/templates/adapters.d.ts +43 -0
  173. package/dist/templates/builders.d.ts +63 -0
  174. package/dist/templates/catalog.d.ts +188 -0
  175. package/dist/templates/componentBindings.d.ts +58 -0
  176. package/dist/templates/controlSchemas.d.ts +25 -0
  177. package/dist/templates/index.d.ts +15 -0
  178. package/dist/templates/index.js +2 -0
  179. package/dist/templates/lookup.d.ts +4 -0
  180. package/dist/templates/packs.d.ts +18 -0
  181. package/dist/templates/presets.d.ts +90 -0
  182. package/dist/templates/types.d.ts +531 -0
  183. package/dist/templates-50NPjaxL.js +9333 -0
  184. package/dist/templates-50NPjaxL.js.map +1 -0
  185. package/dist/types/components.d.ts +26 -4
  186. package/dist/types/form-builder.d.ts +6 -8
  187. package/dist/types/index.d.ts +2 -2
  188. package/dist/types/platform.d.ts +7 -1
  189. package/dist/useScheduleDrag-D4oWdh41.js +4371 -0
  190. package/dist/useScheduleDrag-D4oWdh41.js.map +1 -0
  191. package/dist/utils/formModelSync.d.ts +5 -0
  192. package/dist/utils/items.d.ts +8 -0
  193. package/dist/utils/options.d.ts +6 -0
  194. package/dist/utils/pluginIcon.d.ts +9 -0
  195. package/package.json +7 -2
  196. package/src/__tests__/components/ActionItem.test.ts +99 -0
  197. package/src/__tests__/components/AppAvatarMenu.test.ts +27 -0
  198. package/src/__tests__/components/AppPageSelector.test.ts +134 -0
  199. package/src/__tests__/components/AppPillNav.test.ts +78 -0
  200. package/src/__tests__/components/AppPluginSwitcher.test.ts +44 -0
  201. package/src/__tests__/components/AppSidebar.test.ts +370 -0
  202. package/src/__tests__/components/AppToastContainer.test.ts +48 -0
  203. package/src/__tests__/components/AppTopBar.test.ts +383 -0
  204. package/src/__tests__/components/BaseRadioGroup.test.ts +25 -0
  205. package/src/__tests__/components/BaseSelect.test.ts +21 -0
  206. package/src/__tests__/components/BaseTabs.test.ts +25 -0
  207. package/src/__tests__/components/BatchProgressList.test.ts +52 -0
  208. package/src/__tests__/components/BioTemplateExperimentWorkspaceView.test.ts +153 -0
  209. package/src/__tests__/components/BioTemplatePackWorkspaceView.test.ts +161 -0
  210. package/src/__tests__/components/BioTemplatePresetWorkspaceView.test.ts +281 -0
  211. package/src/__tests__/components/BioTemplateRenderer.test.ts +71 -0
  212. package/src/__tests__/components/Breadcrumb.test.ts +23 -0
  213. package/src/__tests__/components/CalendarGridPanel.test.ts +36 -0
  214. package/src/__tests__/components/ConcentrationInput.test.ts +45 -0
  215. package/src/__tests__/components/ControlWorkspaceView.test.ts +1031 -0
  216. package/src/__tests__/components/DataFrame.test.ts +11 -0
  217. package/src/__tests__/components/DatePicker.test.ts +45 -0
  218. package/src/__tests__/components/DateTimePicker.test.ts +48 -0
  219. package/src/__tests__/components/DropdownButton.test.ts +23 -0
  220. package/src/__tests__/components/EmptyState.test.ts +23 -0
  221. package/src/__tests__/components/ExperimentPopover.test.ts +56 -0
  222. package/src/__tests__/components/FormBuilder.test.ts +296 -0
  223. package/src/__tests__/components/FormCompatibility.test.ts +94 -0
  224. package/src/__tests__/components/GroupAssigner.test.ts +30 -0
  225. package/src/__tests__/components/GroupingModal.test.ts +73 -0
  226. package/src/__tests__/components/MultiSelect.test.ts +48 -0
  227. package/src/__tests__/components/ProtocolStepEditor.test.ts +33 -0
  228. package/src/__tests__/components/ReagentList.test.ts +82 -0
  229. package/src/__tests__/components/SampleHierarchyTree.test.ts +53 -0
  230. package/src/__tests__/components/SampleSelector.test.ts +60 -0
  231. package/src/__tests__/components/SegmentedControl.test.ts +24 -0
  232. package/src/__tests__/components/SettingsButton.test.ts +44 -0
  233. package/src/__tests__/components/SettingsModal.test.ts +296 -0
  234. package/src/__tests__/components/TagsInput.test.ts +75 -0
  235. package/src/__tests__/components/ThemeToggle.test.ts +47 -0
  236. package/src/__tests__/components/TimePicker.test.ts +38 -0
  237. package/src/__tests__/composables/useBioTemplatePackWorkspace.test.ts +122 -0
  238. package/src/__tests__/composables/useBioTemplatePresetWorkspace.test.ts +199 -0
  239. package/src/__tests__/composables/useBioTemplateWorkspace.test.ts +99 -0
  240. package/src/__tests__/composables/useCalendarGrid.test.ts +38 -0
  241. package/src/__tests__/composables/useControlSchema.test.ts +919 -0
  242. package/src/__tests__/composables/useDebouncedWatch.test.ts +93 -0
  243. package/src/__tests__/composables/useDropdownState.test.ts +95 -0
  244. package/src/__tests__/composables/useEventListener.test.ts +116 -0
  245. package/src/__tests__/composables/useExpansionSet.test.ts +62 -0
  246. package/src/__tests__/composables/useExperimentData.test.ts +4 -0
  247. package/src/__tests__/composables/useExperimentSave.test.ts +203 -8
  248. package/src/__tests__/composables/useExperimentSelector.test.ts +164 -0
  249. package/src/__tests__/composables/useForm.test.ts +58 -0
  250. package/src/__tests__/composables/useFormBuilder.test.ts +77 -0
  251. package/src/__tests__/composables/useGroupAssignment.test.ts +73 -0
  252. package/src/__tests__/composables/useListSelection.test.ts +66 -0
  253. package/src/__tests__/composables/usePluginClient.test.ts +444 -0
  254. package/src/__tests__/composables/usePluginConfig.test.ts +5 -0
  255. package/src/__tests__/composables/useRequestSyncState.test.ts +92 -0
  256. package/src/__tests__/composables/useSampleGroups.test.ts +66 -0
  257. package/src/__tests__/composables/useSelectionLimit.test.ts +41 -0
  258. package/src/__tests__/composables/useSortedItems.test.ts +87 -0
  259. package/src/__tests__/composables/useTemplateCollection.test.ts +147 -0
  260. package/src/__tests__/composables/useTextSearch.test.ts +55 -0
  261. package/src/__tests__/composables/useTheme.test.ts +91 -0
  262. package/src/__tests__/composables/useTimeUtils.test.ts +35 -0
  263. package/src/__tests__/docs/frontendDocsCatalog.test.ts +229 -0
  264. package/src/__tests__/fixtures/templates/dose-response.json +81 -0
  265. package/src/__tests__/fixtures/templates/plate-map.json +54 -0
  266. package/src/__tests__/fixtures/templates/qpcr-plate.json +96 -0
  267. package/src/__tests__/fixtures/templates/sample-sheet.json +71 -0
  268. package/src/__tests__/templates/templates.test.ts +1043 -0
  269. package/src/components/ActionItem.vue +82 -0
  270. package/src/components/AppAvatarMenu.vue +15 -69
  271. package/src/components/AppLayout.story.vue +25 -25
  272. package/src/components/AppPageSelector.vue +63 -94
  273. package/src/components/AppPillNav.vue +44 -39
  274. package/src/components/AppPluginSwitcher.vue +41 -145
  275. package/src/components/AppSidebar.story.vue +94 -0
  276. package/src/components/AppSidebar.vue +187 -12
  277. package/src/components/{ToastNotification.story.vue → AppToastContainer.story.vue} +6 -6
  278. package/src/components/AppToastContainer.vue +62 -0
  279. package/src/components/AppTopBar.story.vue +7 -30
  280. package/src/components/AppTopBar.vue +251 -57
  281. package/src/components/BaseModal.vue +3 -5
  282. package/src/components/BaseRadioGroup.vue +7 -3
  283. package/src/components/BaseSelect.vue +11 -7
  284. package/src/components/BaseTabs.vue +6 -4
  285. package/src/components/BatchProgressList.vue +5 -8
  286. package/src/components/BioTemplateExperimentWorkspaceView.story.vue +123 -0
  287. package/src/components/BioTemplateExperimentWorkspaceView.vue +337 -0
  288. package/src/components/BioTemplatePackWorkspaceView.story.vue +107 -0
  289. package/src/components/BioTemplatePackWorkspaceView.vue +176 -0
  290. package/src/components/BioTemplatePresetWorkspaceView.story.vue +151 -0
  291. package/src/components/BioTemplatePresetWorkspaceView.vue +392 -0
  292. package/src/components/BioTemplateRenderer.story.vue +57 -0
  293. package/src/components/BioTemplateRenderer.vue +269 -0
  294. package/src/components/Breadcrumb.vue +14 -8
  295. package/src/components/CalendarGridPanel.vue +120 -0
  296. package/src/components/ConcentrationInput.vue +27 -64
  297. package/src/components/ControlWorkspaceView.story.vue +336 -0
  298. package/src/components/ControlWorkspaceView.vue +347 -0
  299. package/src/components/DataFrame.vue +34 -50
  300. package/src/components/DatePicker.vue +59 -192
  301. package/src/components/DateTimePicker.vue +50 -171
  302. package/src/components/DropdownButton.vue +14 -32
  303. package/src/components/EmptyState.vue +4 -2
  304. package/src/components/ExperimentPopover.vue +5 -22
  305. package/src/components/FormBuilder.vue +124 -27
  306. package/src/components/FormFieldRenderer.vue +15 -38
  307. package/src/components/FormSection.vue +20 -73
  308. package/src/components/GroupAssigner.vue +24 -56
  309. package/src/components/GroupingModal.story.vue +3 -3
  310. package/src/components/GroupingModal.vue +30 -391
  311. package/src/components/MultiSelect.vue +17 -12
  312. package/src/components/PlateMapEditor.vue +3 -8
  313. package/src/components/PluginIcon.vue +2 -22
  314. package/src/components/ProtocolStepEditor.vue +13 -22
  315. package/src/components/ReagentList.vue +25 -33
  316. package/src/components/SampleHierarchyTree.vue +12 -23
  317. package/src/components/SampleSelector.vue +42 -122
  318. package/src/components/SegmentedControl.vue +7 -3
  319. package/src/components/SettingsButton.story.vue +1 -1
  320. package/src/components/SettingsButton.vue +15 -27
  321. package/src/components/SettingsModal.story.vue +1 -1
  322. package/src/components/SettingsModal.vue +120 -29
  323. package/src/components/TagsInput.vue +29 -14
  324. package/src/components/ThemeToggle.vue +9 -7
  325. package/src/components/TimePicker.vue +19 -41
  326. package/src/components/ToastNotification.vue +4 -57
  327. package/src/components/Tooltip.vue +7 -12
  328. package/src/components/WellEditPopup.vue +3 -8
  329. package/src/components/WellPlate.vue +4 -10
  330. package/src/components/index.ts +11 -1
  331. package/src/components/internal/FormFieldRendererInternal.vue +50 -0
  332. package/src/components/internal/FormSectionRenderer.vue +78 -0
  333. package/src/composables/index.ts +212 -0
  334. package/src/composables/platformContextHelpers.ts +74 -0
  335. package/src/composables/useBioTemplateComponents.ts +93 -0
  336. package/src/composables/useBioTemplateControls.ts +41 -0
  337. package/src/composables/useBioTemplatePackWorkspace.ts +181 -0
  338. package/src/composables/useBioTemplatePresetWorkspace.ts +337 -0
  339. package/src/composables/useBioTemplateWorkspace.ts +139 -0
  340. package/src/composables/useCalendarGrid.ts +140 -0
  341. package/src/composables/useControlSchema.ts +1274 -0
  342. package/src/composables/useDebouncedWatch.ts +119 -0
  343. package/src/composables/useDropdownState.ts +83 -0
  344. package/src/composables/useEventListener.ts +111 -0
  345. package/src/composables/useExpansionSet.ts +117 -0
  346. package/src/composables/useExperimentData.ts +20 -11
  347. package/src/composables/useExperimentSave.ts +202 -50
  348. package/src/composables/useExperimentSelector.ts +86 -72
  349. package/src/composables/useForm.ts +49 -4
  350. package/src/composables/useFormBuilder.ts +93 -42
  351. package/src/composables/useGroupAssignment.ts +148 -0
  352. package/src/composables/useListSelection.ts +158 -0
  353. package/src/composables/usePluginApi.ts +7 -14
  354. package/src/composables/usePluginClient.ts +425 -0
  355. package/src/composables/usePluginConfig.ts +34 -13
  356. package/src/composables/useRequestSyncState.ts +126 -0
  357. package/src/composables/useSampleGroups.ts +126 -0
  358. package/src/composables/useSelectionLimit.ts +57 -0
  359. package/src/composables/useSortedItems.ts +118 -0
  360. package/src/composables/useTemplateCollection.ts +229 -0
  361. package/src/composables/useTextSearch.ts +60 -0
  362. package/src/composables/useTheme.ts +2 -28
  363. package/src/composables/useTimeUtils.ts +26 -2
  364. package/src/composables/useWellPlateEditor.ts +13 -9
  365. package/src/index.ts +224 -4
  366. package/src/install.ts +11 -4
  367. package/src/stores/settings.ts +13 -9
  368. package/src/styles/components/app-page-selector.css +23 -0
  369. package/src/styles/components/app-pill-nav.css +7 -0
  370. package/src/styles/components/app-top-bar.css +34 -0
  371. package/src/styles/components/concentration-input.css +3 -142
  372. package/src/styles/components/empty-state.css +0 -16
  373. package/src/styles/components/settings-button.css +3 -66
  374. package/src/styles/components/theme-toggle.css +3 -66
  375. package/src/styles/index.css +0 -1
  376. package/src/templates/adapters.ts +785 -0
  377. package/src/templates/builders.ts +2149 -0
  378. package/src/templates/catalog.ts +245 -0
  379. package/src/templates/componentBindings.ts +615 -0
  380. package/src/templates/controlSchemas.ts +718 -0
  381. package/src/templates/index.ts +314 -0
  382. package/src/templates/lookup.ts +18 -0
  383. package/src/templates/packs.ts +156 -0
  384. package/src/templates/presets.ts +146 -0
  385. package/src/templates/types.ts +668 -0
  386. package/src/types/components.ts +41 -4
  387. package/src/types/form-builder.ts +7 -2
  388. package/src/types/index.ts +14 -0
  389. package/src/types/platform.ts +7 -1
  390. package/src/utils/formModelSync.ts +52 -0
  391. package/src/utils/items.ts +28 -0
  392. package/src/utils/options.ts +23 -0
  393. package/src/utils/pluginIcon.ts +30 -0
  394. package/dist/auth-DsI0rQ7_.js.map +0 -1
  395. package/dist/components-_XqPEhP9.js.map +0 -1
  396. package/dist/composables-tiZqLu1M.js.map +0 -1
  397. package/dist/useScheduleDrag-CA9sGNJG.js +0 -7181
  398. package/dist/useScheduleDrag-CA9sGNJG.js.map +0 -1
  399. package/src/styles/components/grouping-modal.css +0 -323
@@ -305,6 +305,17 @@ describe('DataFrame', () => {
305
305
  expect(rows).toHaveLength(0)
306
306
  })
307
307
 
308
+ it('should treat whitespace-only search as empty', async () => {
309
+ const wrapper = mount(DataFrame, {
310
+ props: { data: mockData, columns: mockColumns, searchable: true },
311
+ })
312
+ const searchInput = wrapper.find('.mint-dataframe__search')
313
+ await searchInput.setValue(' ')
314
+
315
+ const rows = wrapper.findAll('.mint-dataframe__row')
316
+ expect(rows).toHaveLength(3)
317
+ })
318
+
308
319
  it('should be case-insensitive', async () => {
309
320
  const wrapper = mount(DataFrame, {
310
321
  props: { data: mockData, columns: mockColumns, searchable: true },
@@ -0,0 +1,45 @@
1
+ import { mount } from '@vue/test-utils'
2
+ import { afterEach, describe, expect, it } from 'vitest'
3
+ import { nextTick } from 'vue'
4
+ import DatePicker from '../../components/DatePicker.vue'
5
+
6
+ function dropdown(): HTMLElement | null {
7
+ return document.body.querySelector('.mint-date-picker__dropdown')
8
+ }
9
+
10
+ describe('DatePicker', () => {
11
+ afterEach(() => {
12
+ document.body.innerHTML = ''
13
+ })
14
+
15
+ it('uses shared dropdown state with teleported calendar content', async () => {
16
+ const wrapper = mount(DatePicker, {
17
+ props: {
18
+ modelValue: '2026-04-22',
19
+ },
20
+ attachTo: document.body,
21
+ })
22
+
23
+ await wrapper.get('input').trigger('click')
24
+ expect(dropdown()).not.toBeNull()
25
+
26
+ document.dispatchEvent(new KeyboardEvent('keydown', { key: 'Escape' }))
27
+ await nextTick()
28
+ expect(dropdown()).toBeNull()
29
+
30
+ await wrapper.get('input').trigger('click')
31
+ expect(dropdown()).not.toBeNull()
32
+
33
+ const nextButton = document.body.querySelector<HTMLButtonElement>('[aria-label="Next month"]')
34
+ expect(nextButton).not.toBeNull()
35
+ nextButton?.dispatchEvent(new MouseEvent('click', { bubbles: true }))
36
+ await nextTick()
37
+ expect(dropdown()).not.toBeNull()
38
+
39
+ document.body.dispatchEvent(new MouseEvent('click', { bubbles: true }))
40
+ await nextTick()
41
+ expect(dropdown()).toBeNull()
42
+
43
+ wrapper.unmount()
44
+ })
45
+ })
@@ -0,0 +1,48 @@
1
+ import { mount } from '@vue/test-utils'
2
+ import { nextTick } from 'vue'
3
+ import { describe, expect, it } from 'vitest'
4
+ import CalendarGridPanel from '../../components/CalendarGridPanel.vue'
5
+ import DateTimePicker from '../../components/DateTimePicker.vue'
6
+
7
+ describe('DateTimePicker', () => {
8
+ it('uses the shared calendar panel and keeps time selection working', async () => {
9
+ const wrapper = mount(DateTimePicker, {
10
+ props: {
11
+ modelValue: '2026-04-22T14:30',
12
+ },
13
+ })
14
+
15
+ await wrapper.find('input').trigger('click')
16
+ await nextTick()
17
+
18
+ expect(wrapper.findComponent(CalendarGridPanel).exists()).toBe(true)
19
+ expect(wrapper.find('.mint-date-picker__day--selected').text()).toBe('22')
20
+
21
+ const timeButton = wrapper.findAll('.mint-datetime-picker__time-chip')
22
+ .find((button) => button.text() === '09:00')
23
+ expect(timeButton).toBeDefined()
24
+
25
+ await timeButton?.trigger('click')
26
+
27
+ expect(wrapper.emitted('update:modelValue')?.at(-1)).toEqual(['2026-04-22T09:00'])
28
+ })
29
+
30
+ it('uses shared dropdown state for Escape close behavior', async () => {
31
+ const wrapper = mount(DateTimePicker, {
32
+ props: {
33
+ modelValue: '2026-04-22T14:30',
34
+ },
35
+ attachTo: document.body,
36
+ })
37
+
38
+ await wrapper.find('input').trigger('click')
39
+ expect(wrapper.find('.mint-datetime-picker__dropdown').exists()).toBe(true)
40
+
41
+ document.dispatchEvent(new KeyboardEvent('keydown', { key: 'Escape' }))
42
+ await nextTick()
43
+
44
+ expect(wrapper.find('.mint-datetime-picker__dropdown').exists()).toBe(false)
45
+
46
+ wrapper.unmount()
47
+ })
48
+ })
@@ -84,6 +84,29 @@ describe('DropdownButton', () => {
84
84
  expect(wrapper.emitted('select')).toHaveLength(1)
85
85
  expect(wrapper.emitted('select')?.[0]).toEqual([mockOptions[1]])
86
86
  })
87
+
88
+ it('accepts string and number shorthand options', async () => {
89
+ const wrapper = mount(DropdownButton, {
90
+ props: {
91
+ modelValue: 96,
92
+ options: ['custom', 96, { value: 'log', label: 'Log scale' }],
93
+ },
94
+ })
95
+
96
+ expect(wrapper.find('.mint-dropdown-button__label').text()).toBe('96')
97
+
98
+ await wrapper.find('.mint-dropdown-button__trigger').trigger('click')
99
+ expect(wrapper.findAll('.mint-dropdown-button__option').map(option => option.text())).toEqual([
100
+ 'custom',
101
+ '96',
102
+ 'Log scale',
103
+ ])
104
+
105
+ await wrapper.findAll('.mint-dropdown-button__option')[0].trigger('click')
106
+
107
+ expect(wrapper.emitted('update:modelValue')?.[0]).toEqual(['custom'])
108
+ expect(wrapper.emitted('select')?.[0]).toEqual([{ value: 'custom', label: 'custom' }])
109
+ })
87
110
  })
88
111
 
89
112
  describe('toggle open/close', () => {
@@ -0,0 +1,23 @@
1
+ import { mount } from '@vue/test-utils'
2
+ import { describe, expect, it } from 'vitest'
3
+ import EmptyState from '../../components/EmptyState.vue'
4
+
5
+ describe('EmptyState', () => {
6
+ it('renders action through BaseButton and emits action clicks', async () => {
7
+ const wrapper = mount(EmptyState, {
8
+ props: {
9
+ title: 'No experiments',
10
+ actionLabel: 'Create experiment',
11
+ },
12
+ })
13
+
14
+ const button = wrapper.find('.mint-button')
15
+ expect(button.exists()).toBe(true)
16
+ expect(button.classes()).toContain('mint-button--primary')
17
+ expect(button.text()).toBe('Create experiment')
18
+
19
+ await button.trigger('click')
20
+
21
+ expect(wrapper.emitted('action')).toHaveLength(1)
22
+ })
23
+ })
@@ -0,0 +1,56 @@
1
+ import { mount } from '@vue/test-utils'
2
+ import { afterEach, describe, expect, it } from 'vitest'
3
+ import ExperimentPopover from '../../components/ExperimentPopover.vue'
4
+
5
+ function mountPopover(props = {}) {
6
+ return mount(ExperimentPopover, {
7
+ props,
8
+ attachTo: document.body,
9
+ global: {
10
+ stubs: {
11
+ ConfirmDialog: {
12
+ template: '<div data-test="confirm-dialog" />',
13
+ },
14
+ },
15
+ },
16
+ })
17
+ }
18
+
19
+ describe('ExperimentPopover', () => {
20
+ afterEach(() => {
21
+ document.body.innerHTML = ''
22
+ })
23
+
24
+ it('opens and closes the popover on trigger and outside click', async () => {
25
+ const wrapper = mountPopover({
26
+ experimentName: 'Dose response run',
27
+ experimentCode: 'EXP-001',
28
+ })
29
+
30
+ expect(wrapper.find('.mint-experiment-popover__panel').exists()).toBe(false)
31
+
32
+ await wrapper.find('.mint-experiment-popover__trigger').trigger('click')
33
+ expect(wrapper.find('.mint-experiment-popover__panel').exists()).toBe(true)
34
+
35
+ document.body.click()
36
+ await wrapper.vm.$nextTick()
37
+ expect(wrapper.find('.mint-experiment-popover__panel').exists()).toBe(false)
38
+
39
+ wrapper.unmount()
40
+ })
41
+
42
+ it('emits select and closes when changing experiment context', async () => {
43
+ const wrapper = mountPopover({
44
+ experimentName: 'Dose response run',
45
+ experimentCode: 'EXP-001',
46
+ })
47
+
48
+ await wrapper.find('.mint-experiment-popover__trigger').trigger('click')
49
+ await wrapper.find('.mint-experiment-popover__change-btn').trigger('click')
50
+
51
+ expect(wrapper.emitted('select')).toHaveLength(1)
52
+ expect(wrapper.find('.mint-experiment-popover__panel').exists()).toBe(false)
53
+
54
+ wrapper.unmount()
55
+ })
56
+ })
@@ -0,0 +1,296 @@
1
+ import { flushPromises, mount } from '@vue/test-utils'
2
+ import { nextTick } from 'vue'
3
+ import { describe, expect, it } from 'vitest'
4
+ import FormBuilder from '../../components/FormBuilder.vue'
5
+ import FormActions from '../../components/FormActions.vue'
6
+ import { defineControlModel, useControlWorkspace, type ControlSchema } from '../../composables/useControlSchema'
7
+ import type { FormSchema } from '../../types/form-builder'
8
+
9
+ const controls: ControlSchema = {
10
+ threshold: {
11
+ label: 'Threshold',
12
+ default: 0.05,
13
+ min: 0,
14
+ max: 1,
15
+ section: 'parameters',
16
+ sectionLabel: 'Parameters',
17
+ },
18
+ method: {
19
+ label: 'Method',
20
+ default: 'linear',
21
+ options: ['linear', 'logistic'],
22
+ section: 'parameters',
23
+ },
24
+ notes: {
25
+ label: 'Notes',
26
+ placeholder: 'Optional notes',
27
+ section: 'metadata',
28
+ sectionLabel: 'Metadata',
29
+ },
30
+ }
31
+
32
+ describe('FormBuilder', () => {
33
+ it('renders compact controls without a prebuilt FormSchema', () => {
34
+ const wrapper = mount(FormBuilder, {
35
+ props: { controls },
36
+ })
37
+
38
+ expect(wrapper.find('.mint-form-builder').exists()).toBe(true)
39
+ expect(wrapper.text()).toContain('Parameters')
40
+ expect(wrapper.text()).toContain('Metadata')
41
+ expect((wrapper.find('input').element as HTMLInputElement).value).toBe('0.05')
42
+ expect(wrapper.find('select').exists()).toBe(true)
43
+ })
44
+
45
+ it('merges compact control defaults with incoming model values', async () => {
46
+ const wrapper = mount(FormBuilder, {
47
+ props: {
48
+ controls,
49
+ modelValue: { threshold: 0.25 },
50
+ },
51
+ })
52
+
53
+ await wrapper.find('input').setValue('0.5')
54
+
55
+ expect(wrapper.emitted('update:modelValue')?.at(-1)?.[0]).toMatchObject({
56
+ threshold: 0.5,
57
+ method: 'linear',
58
+ notes: '',
59
+ })
60
+ })
61
+
62
+ it('uses controlOptions initialValues over compact control defaults', () => {
63
+ const wrapper = mount(FormBuilder, {
64
+ props: {
65
+ controls,
66
+ controlOptions: {
67
+ initialValues: {
68
+ threshold: 0.35,
69
+ method: 'logistic',
70
+ },
71
+ },
72
+ },
73
+ })
74
+
75
+ expect((wrapper.find('input').element as HTMLInputElement).value).toBe('0.35')
76
+ expect((wrapper.find('select').element as HTMLSelectElement).value).toBe('logistic')
77
+ })
78
+
79
+ it('syncs external modelValue changes into rendered fields', async () => {
80
+ const wrapper = mount(FormBuilder, {
81
+ props: {
82
+ controls,
83
+ modelValue: { threshold: 0.25 },
84
+ },
85
+ })
86
+
87
+ await wrapper.setProps({ modelValue: { threshold: 0.75, method: 'logistic' } })
88
+
89
+ expect((wrapper.find('input').element as HTMLInputElement).value).toBe('0.75')
90
+ expect((wrapper.find('select').element as HTMLSelectElement).value).toBe('logistic')
91
+ })
92
+
93
+ it('syncs compact controls schema changes into rendered fields and submit data', async () => {
94
+ const thresholdOnly: ControlSchema = {
95
+ threshold: controls.threshold,
96
+ }
97
+
98
+ const wrapper = mount(FormBuilder, {
99
+ props: {
100
+ controls: thresholdOnly,
101
+ },
102
+ })
103
+
104
+ await wrapper.find('input').setValue('0.5')
105
+ await wrapper.setProps({
106
+ controls: {
107
+ threshold: controls.threshold,
108
+ method: controls.method,
109
+ },
110
+ })
111
+
112
+ expect(wrapper.find('select').exists()).toBe(true)
113
+ expect((wrapper.find('input').element as HTMLInputElement).value).toBe('0.5')
114
+ expect((wrapper.find('select').element as HTMLSelectElement).value).toBe('linear')
115
+
116
+ wrapper.findComponent(FormActions).vm.$emit('submit')
117
+ await flushPromises()
118
+
119
+ expect(wrapper.emitted('submit')?.at(-1)?.[0]).toMatchObject({
120
+ threshold: 0.5,
121
+ method: 'linear',
122
+ })
123
+ })
124
+
125
+ it('removes stale field data when compact controls are removed', async () => {
126
+ const wrapper = mount(FormBuilder, {
127
+ props: { controls },
128
+ })
129
+
130
+ await wrapper.setProps({
131
+ controls: {
132
+ method: controls.method,
133
+ },
134
+ })
135
+
136
+ expect(wrapper.find('input').exists()).toBe(false)
137
+ wrapper.findComponent(FormActions).vm.$emit('submit')
138
+ await flushPromises()
139
+
140
+ const submitted = wrapper.emitted('submit')?.at(-1)?.[0] as Record<string, unknown>
141
+ expect(submitted).toMatchObject({ method: 'linear' })
142
+ expect(submitted).not.toHaveProperty('threshold')
143
+ })
144
+
145
+ it('renders fields from a complete control model binding', async () => {
146
+ const model = defineControlModel({
147
+ views: {
148
+ run: {
149
+ label: 'Run',
150
+ sections: {
151
+ parameters: {
152
+ label: 'Parameters',
153
+ controls: {
154
+ threshold: { label: 'Threshold', default: 0.05, min: 0, max: 1 },
155
+ method: { label: 'Method', default: 'linear', options: ['linear', 'logistic'] },
156
+ },
157
+ },
158
+ },
159
+ },
160
+ },
161
+ })
162
+ const wrapper = mount(FormBuilder, {
163
+ props: {
164
+ model,
165
+ modelValue: { threshold: 0.25 },
166
+ },
167
+ })
168
+
169
+ expect(wrapper.text()).toContain('Parameters')
170
+ expect((wrapper.find('input').element as HTMLInputElement).value).toBe('0.25')
171
+ expect((wrapper.find('select').element as HTMLSelectElement).value).toBe('linear')
172
+
173
+ await wrapper.find('input').setValue('0.5')
174
+
175
+ expect(wrapper.emitted('update:modelValue')?.at(-1)?.[0]).toMatchObject({
176
+ threshold: 0.5,
177
+ method: 'linear',
178
+ })
179
+ })
180
+
181
+ it('uses initialValues from a complete control model binding', () => {
182
+ const model = defineControlModel({
183
+ initialValues: {
184
+ threshold: 0.25,
185
+ method: 'logistic',
186
+ },
187
+ views: {
188
+ run: {
189
+ label: 'Run',
190
+ sections: {
191
+ parameters: {
192
+ label: 'Parameters',
193
+ controls: {
194
+ threshold: { label: 'Threshold', default: 0.05, min: 0, max: 1 },
195
+ method: { label: 'Method', default: 'linear', options: ['linear', 'logistic'] },
196
+ },
197
+ },
198
+ },
199
+ },
200
+ },
201
+ })
202
+ const wrapper = mount(FormBuilder, {
203
+ props: {
204
+ model,
205
+ controlOptions: {
206
+ initialValues: {
207
+ method: 'linear',
208
+ },
209
+ },
210
+ },
211
+ })
212
+
213
+ expect((wrapper.find('input').element as HTMLInputElement).value).toBe('0.25')
214
+ expect((wrapper.find('select').element as HTMLSelectElement).value).toBe('linear')
215
+ })
216
+
217
+ it('ignores external modelValue keys that are not part of the form schema', async () => {
218
+ const wrapper = mount(FormBuilder, {
219
+ props: {
220
+ controls,
221
+ modelValue: { threshold: 0.25, externalContext: 'experiment-1' },
222
+ },
223
+ })
224
+
225
+ await wrapper.find('input').setValue('0.5')
226
+ await wrapper.setProps({
227
+ modelValue: { threshold: 0.5, method: 'linear', externalContext: 'experiment-2' },
228
+ })
229
+
230
+ expect((wrapper.find('input').element as HTMLInputElement).value).toBe('0.5')
231
+ expect((wrapper.findComponent(FormBuilder).vm as unknown as { form: { dirty: Record<string, boolean> } }).form.dirty.threshold).toBe(true)
232
+ })
233
+
234
+ it('works with direct useControlWorkspace form bindings', async () => {
235
+ const workspace = useControlWorkspace({
236
+ threshold: 0.05,
237
+ method: ['linear', 'logistic'],
238
+ })
239
+ const wrapper = mount(FormBuilder, {
240
+ props: workspace.form,
241
+ })
242
+
243
+ await wrapper.find('input').setValue('0.5')
244
+ expect(workspace.values.threshold).toBe(0.5)
245
+
246
+ workspace.setValues({ method: 'logistic' })
247
+ await nextTick()
248
+ expect((wrapper.find('select').element as HTMLSelectElement).value).toBe('logistic')
249
+ })
250
+
251
+ it('uses explicit schema when both schema and controls are provided', () => {
252
+ const schema: FormSchema = {
253
+ sections: [
254
+ {
255
+ id: 'explicit',
256
+ title: 'Explicit Schema',
257
+ fields: [
258
+ {
259
+ name: 'explicitField',
260
+ label: 'Explicit Field',
261
+ type: 'text',
262
+ defaultValue: 'from schema',
263
+ },
264
+ ],
265
+ },
266
+ ],
267
+ }
268
+
269
+ const wrapper = mount(FormBuilder, {
270
+ props: { controls, schema },
271
+ })
272
+
273
+ expect(wrapper.text()).toContain('Explicit Schema')
274
+ expect(wrapper.text()).not.toContain('Parameters')
275
+ expect((wrapper.find('input').element as HTMLInputElement).value).toBe('from schema')
276
+ })
277
+
278
+ it('can hide the default actions for embedded forms', () => {
279
+ const wrapper = mount(FormBuilder, {
280
+ props: {
281
+ controls,
282
+ showActions: false,
283
+ },
284
+ })
285
+
286
+ expect(wrapper.find('.mint-form-actions').exists()).toBe(false)
287
+ })
288
+
289
+ it('renders default actions by default', () => {
290
+ const wrapper = mount(FormBuilder, {
291
+ props: { controls },
292
+ })
293
+
294
+ expect(wrapper.find('.mint-form-actions').exists()).toBe(true)
295
+ })
296
+ })
@@ -0,0 +1,94 @@
1
+ import { mount } from '@vue/test-utils'
2
+ import { describe, expect, it } from 'vitest'
3
+ import FormFieldRenderer from '../../components/FormFieldRenderer.vue'
4
+ import FormSection from '../../components/FormSection.vue'
5
+ import { useFormBuilder } from '../../composables/useFormBuilder'
6
+ import type { FormSchema } from '../../types/form-builder'
7
+
8
+ const schema: FormSchema = {
9
+ sections: [
10
+ {
11
+ id: 'parameters',
12
+ title: 'Parameters',
13
+ fields: [
14
+ { name: 'threshold', label: 'Threshold', type: 'text', defaultValue: '0.05' },
15
+ ],
16
+ },
17
+ ],
18
+ }
19
+
20
+ describe('deprecated form compatibility components', () => {
21
+ it('keeps FormFieldRenderer rendering through the internal renderer', () => {
22
+ const builder = useFormBuilder(schema)
23
+ const field = schema.sections[0].fields[0]
24
+
25
+ const wrapper = mount(FormFieldRenderer, {
26
+ props: {
27
+ field,
28
+ resolvedProps: builder.getResolvedFieldProps(field),
29
+ form: builder.form,
30
+ },
31
+ })
32
+
33
+ expect(wrapper.text()).toContain('Threshold')
34
+ expect(wrapper.find('input').exists()).toBe(true)
35
+ })
36
+
37
+ it('keeps FormFieldRenderer field slot forwarding', () => {
38
+ const builder = useFormBuilder(schema)
39
+ const field = schema.sections[0].fields[0]
40
+
41
+ const wrapper = mount(FormFieldRenderer, {
42
+ props: {
43
+ field,
44
+ resolvedProps: builder.getResolvedFieldProps(field),
45
+ form: builder.form,
46
+ },
47
+ slots: {
48
+ 'field:threshold': '<div class="custom-field">Custom threshold</div>',
49
+ },
50
+ })
51
+
52
+ expect(wrapper.find('.custom-field').text()).toBe('Custom threshold')
53
+ expect(wrapper.find('input').exists()).toBe(false)
54
+ })
55
+
56
+ it('keeps FormSection section and field slot forwarding', () => {
57
+ const builder = useFormBuilder(schema)
58
+ const section = schema.sections[0]
59
+
60
+ const wrapper = mount(FormSection, {
61
+ props: {
62
+ section,
63
+ builder,
64
+ },
65
+ slots: {
66
+ 'field:threshold': '<div class="custom-field">Custom threshold</div>',
67
+ 'section:parameters:after': '<div class="section-after">After section</div>',
68
+ },
69
+ })
70
+
71
+ expect(wrapper.text()).toContain('Parameters')
72
+ expect(wrapper.find('.custom-field').text()).toBe('Custom threshold')
73
+ expect(wrapper.find('.section-after').text()).toBe('After section')
74
+ expect(wrapper.find('input').exists()).toBe(false)
75
+ })
76
+
77
+ it('keeps FormSection whole-section slot forwarding', () => {
78
+ const builder = useFormBuilder(schema)
79
+ const section = schema.sections[0]
80
+
81
+ const wrapper = mount(FormSection, {
82
+ props: {
83
+ section,
84
+ builder,
85
+ },
86
+ slots: {
87
+ 'section:parameters': '<div class="custom-section">Custom section</div>',
88
+ },
89
+ })
90
+
91
+ expect(wrapper.find('.custom-section').text()).toBe('Custom section')
92
+ expect(wrapper.text()).not.toContain('Threshold')
93
+ })
94
+ })
@@ -0,0 +1,30 @@
1
+ import { mount } from '@vue/test-utils'
2
+ import { describe, expect, it } from 'vitest'
3
+ import GroupAssigner from '../../components/GroupAssigner.vue'
4
+ import type { GroupItem } from '../../types'
5
+
6
+ const groups: GroupItem[] = [
7
+ { name: 'control-a', color: '#3B82F6', count: 3 },
8
+ { name: 'treatment-a', color: '#F43F5E', count: 4 },
9
+ ]
10
+
11
+ describe('GroupAssigner', () => {
12
+ it('uses shared assignment state for counts, validation, and emitted updates', async () => {
13
+ const wrapper = mount(GroupAssigner, {
14
+ props: {
15
+ groups,
16
+ group1: ['control-a'],
17
+ group2: [],
18
+ minPerGroup: 1,
19
+ },
20
+ })
21
+
22
+ expect(wrapper.text()).toContain('3 samples')
23
+ expect(wrapper.text()).toContain('0 samples')
24
+ expect(wrapper.text()).toContain('Add 1 more to Treatment')
25
+
26
+ await wrapper.find('.mint-group-assigner__pill-remove').trigger('click')
27
+ expect(wrapper.emitted('update:group1')?.[0]).toEqual([[]])
28
+ expect(wrapper.emitted('update:group2')).toBeUndefined()
29
+ })
30
+ })