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

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 (427) hide show
  1. package/README.md +225 -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/ComponentBindingRenderer.test.d.ts +1 -0
  19. package/dist/__tests__/components/ConcentrationInput.test.d.ts +1 -0
  20. package/dist/__tests__/components/ControlWorkspaceView.test.d.ts +1 -0
  21. package/dist/__tests__/components/DatePicker.test.d.ts +1 -0
  22. package/dist/__tests__/components/DateTimePicker.test.d.ts +1 -0
  23. package/dist/__tests__/components/DoseDesignWorkspaceView.test.d.ts +1 -0
  24. package/dist/__tests__/components/EmptyState.test.d.ts +1 -0
  25. package/dist/__tests__/components/ExperimentPopover.test.d.ts +1 -0
  26. package/dist/__tests__/components/FormBuilder.test.d.ts +1 -0
  27. package/dist/__tests__/components/GroupAssigner.test.d.ts +1 -0
  28. package/dist/__tests__/components/MultiSelect.test.d.ts +1 -0
  29. package/dist/__tests__/components/PluginWorkspaceView.test.d.ts +1 -0
  30. package/dist/__tests__/components/ProtocolStepEditor.test.d.ts +1 -0
  31. package/dist/__tests__/components/ReagentList.test.d.ts +1 -0
  32. package/dist/__tests__/components/SampleHierarchyTree.test.d.ts +1 -0
  33. package/dist/__tests__/components/SampleSelector.test.d.ts +1 -0
  34. package/dist/__tests__/components/SegmentedControl.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/experiment-utils.test.d.ts +1 -0
  40. package/dist/__tests__/composables/useApi.test.d.ts +1 -0
  41. package/dist/__tests__/composables/useBioTemplatePackWorkspace.test.d.ts +1 -0
  42. package/dist/__tests__/composables/useBioTemplatePresetWorkspace.test.d.ts +1 -0
  43. package/dist/__tests__/composables/useBioTemplateWorkspace.test.d.ts +1 -0
  44. package/dist/__tests__/composables/useCalendarGrid.test.d.ts +1 -0
  45. package/dist/__tests__/composables/useControlSchema.test.d.ts +1 -0
  46. package/dist/__tests__/composables/useDebouncedWatch.test.d.ts +1 -0
  47. package/dist/__tests__/composables/useDropdownState.test.d.ts +1 -0
  48. package/dist/__tests__/composables/useEventListener.test.d.ts +1 -0
  49. package/dist/__tests__/composables/useExpansionSet.test.d.ts +1 -0
  50. package/dist/__tests__/composables/useExperimentData.test.d.ts +1 -0
  51. package/dist/__tests__/composables/useExperimentSelector.test.d.ts +1 -0
  52. package/dist/__tests__/composables/useGroupAssignment.test.d.ts +1 -0
  53. package/dist/__tests__/composables/useListSelection.test.d.ts +1 -0
  54. package/dist/__tests__/composables/usePluginClient.test.d.ts +1 -0
  55. package/dist/__tests__/composables/usePluginConfig.test.d.ts +1 -0
  56. package/dist/__tests__/composables/useRequestSyncState.test.d.ts +1 -0
  57. package/dist/__tests__/composables/useSampleGroups.test.d.ts +1 -0
  58. package/dist/__tests__/composables/useSelectionLimit.test.d.ts +1 -0
  59. package/dist/__tests__/composables/useSortedItems.test.d.ts +1 -0
  60. package/dist/__tests__/composables/useTemplateCollection.test.d.ts +1 -0
  61. package/dist/__tests__/composables/useTextSearch.test.d.ts +1 -0
  62. package/dist/__tests__/composables/useTheme.test.d.ts +1 -0
  63. package/dist/__tests__/composables/useTimeUtils.test.d.ts +1 -0
  64. package/dist/__tests__/docs/frontendDocsCatalog.test.d.ts +1 -0
  65. package/dist/__tests__/templates/templates.test.d.ts +1 -0
  66. package/dist/{auth-DsI0rQ7_.js → auth-QQj2kkze.js} +12 -5
  67. package/dist/auth-QQj2kkze.js.map +1 -0
  68. package/dist/components/AppAvatarMenu.vue.d.ts +2 -7
  69. package/dist/components/AppContainer.vue.d.ts +1 -1
  70. package/dist/components/AppLayout.vue.d.ts +20 -1
  71. package/dist/components/AppSidebar.vue.d.ts +111 -6
  72. package/dist/components/AppTopBar.vue.d.ts +35 -22
  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 +119 -0
  83. package/dist/components/BioTemplatePackWorkspaceView.vue.d.ts +93 -0
  84. package/dist/components/BioTemplatePresetWorkspaceView.vue.d.ts +87 -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/CollapsibleCard.vue.d.ts +1 -1
  89. package/dist/components/ComponentBindingRenderer.vue.d.ts +44 -0
  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 +147 -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/DoseDesignWorkspaceView.vue.d.ts +149 -0
  97. package/dist/components/DropdownButton.vue.d.ts +3 -3
  98. package/dist/components/EmptyState.vue.d.ts +1 -2
  99. package/dist/components/ExperimentDataViewer.vue.d.ts +1 -1
  100. package/dist/components/ExperimentTimeline.vue.d.ts +2 -2
  101. package/dist/components/FileUploader.vue.d.ts +1 -1
  102. package/dist/components/FitPanel.vue.d.ts +1 -1
  103. package/dist/components/FormActions.vue.d.ts +4 -4
  104. package/dist/components/FormBuilder.vue.d.ts +31 -17
  105. package/dist/components/FormulaInput.vue.d.ts +2 -2
  106. package/dist/components/MoleculeInput.vue.d.ts +2 -2
  107. package/dist/components/MultiSelect.vue.d.ts +3 -3
  108. package/dist/components/NumberInput.vue.d.ts +1 -1
  109. package/dist/components/PlateMapEditor.vue.d.ts +1 -1
  110. package/dist/components/PluginWorkspaceView.vue.d.ts +310 -0
  111. package/dist/components/ProgressBar.vue.d.ts +1 -1
  112. package/dist/components/ProtocolStepEditor.vue.d.ts +3 -1
  113. package/dist/components/RackEditor.vue.d.ts +2 -2
  114. package/dist/components/SampleLegend.vue.d.ts +2 -2
  115. package/dist/components/ScheduleCalendar.vue.d.ts +2 -2
  116. package/dist/components/SegmentedControl.vue.d.ts +2 -2
  117. package/dist/components/SequenceInput.vue.d.ts +3 -3
  118. package/dist/components/SettingsModal.vue.d.ts +14 -6
  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 +6 -6
  125. package/dist/components/index.d.ts +9 -8
  126. package/dist/components/index.js +3 -3
  127. package/dist/components/{SettingsButton.vue.d.ts → internal/ActionItemInternal.vue.d.ts} +11 -9
  128. package/dist/components/{AppPageSelector.vue.d.ts → internal/AppPageSelectorInternal.vue.d.ts} +3 -6
  129. package/dist/components/{AppPillNav.vue.d.ts → internal/AppPillNavInternal.vue.d.ts} +4 -2
  130. package/dist/components/internal/CalendarGridPanelInternal.vue.d.ts +25 -0
  131. package/dist/components/{FormFieldRenderer.vue.d.ts → internal/FormFieldRendererInternal.vue.d.ts} +2 -2
  132. package/dist/components/{FormSection.vue.d.ts → internal/FormSectionRenderer.vue.d.ts} +7 -7
  133. package/dist/components/{WellEditPopup.vue.d.ts → internal/WellEditPopupInternal.vue.d.ts} +1 -1
  134. package/dist/{components-_XqPEhP9.js → components-BkGF4B4y.js} +9760 -8471
  135. package/dist/components-BkGF4B4y.js.map +1 -0
  136. package/dist/composables/experiment-utils.d.ts +8 -0
  137. package/dist/composables/index.d.ts +22 -5
  138. package/dist/composables/index.js +4 -3
  139. package/dist/composables/platformContextHelpers.d.ts +14 -0
  140. package/dist/composables/useAppExperiment.d.ts +31 -2
  141. package/dist/composables/useBioTemplateComponents.d.ts +22 -0
  142. package/dist/composables/useBioTemplateControls.d.ts +6 -0
  143. package/dist/composables/useBioTemplatePackWorkspace.d.ts +46 -0
  144. package/dist/composables/useBioTemplatePresetWorkspace.d.ts +75 -0
  145. package/dist/composables/useBioTemplateWorkspace.d.ts +51 -0
  146. package/dist/composables/useCalendarGrid.d.ts +26 -0
  147. package/dist/composables/useControlSchema.d.ts +343 -0
  148. package/dist/composables/useDebouncedWatch.d.ts +20 -0
  149. package/dist/composables/useDropdownState.d.ts +19 -0
  150. package/dist/composables/useEventListener.d.ts +13 -0
  151. package/dist/composables/useExpansionSet.d.ts +21 -0
  152. package/dist/composables/useExperimentData.d.ts +10 -0
  153. package/dist/composables/useExperimentSave.d.ts +31 -2
  154. package/dist/composables/useExperimentSelector.d.ts +20 -0
  155. package/dist/composables/useForm.d.ts +2 -0
  156. package/dist/composables/useGroupAssignment.d.ts +31 -0
  157. package/dist/composables/useListSelection.d.ts +35 -0
  158. package/dist/composables/usePlatformContext.d.ts +21 -3
  159. package/dist/composables/usePluginClient.d.ts +112 -0
  160. package/dist/composables/usePluginConfig.d.ts +12 -0
  161. package/dist/composables/useRequestSyncState.d.ts +34 -0
  162. package/dist/composables/useSampleGroups.d.ts +32 -0
  163. package/dist/composables/useSelectionLimit.d.ts +17 -0
  164. package/dist/composables/useSortedItems.d.ts +32 -0
  165. package/dist/composables/useTemplateCollection.d.ts +58 -0
  166. package/dist/composables/useTextSearch.d.ts +18 -0
  167. package/dist/composables/useTimeUtils.d.ts +8 -0
  168. package/dist/{composables-tiZqLu1M.js → composables-CHsME9H1.js} +240 -146
  169. package/dist/composables-CHsME9H1.js.map +1 -0
  170. package/dist/index.d.ts +6 -4
  171. package/dist/index.js +6 -5
  172. package/dist/install.d.ts +7 -2
  173. package/dist/install.js +2 -2
  174. package/dist/install.js.map +1 -1
  175. package/dist/stores/index.js +1 -1
  176. package/dist/stores/settings.d.ts +4 -1
  177. package/dist/styles.css +4746 -5514
  178. package/dist/templates/adapters.d.ts +43 -0
  179. package/dist/templates/builders.d.ts +63 -0
  180. package/dist/templates/catalog.d.ts +188 -0
  181. package/dist/templates/componentBindings.d.ts +71 -0
  182. package/dist/templates/controlSchemas.d.ts +25 -0
  183. package/dist/templates/index.d.ts +15 -0
  184. package/dist/templates/index.js +2 -0
  185. package/dist/templates/lookup.d.ts +4 -0
  186. package/dist/templates/packs.d.ts +18 -0
  187. package/dist/templates/presets.d.ts +90 -0
  188. package/dist/templates/types.d.ts +531 -0
  189. package/dist/templates-B5jmTWuk.js +9388 -0
  190. package/dist/templates-B5jmTWuk.js.map +1 -0
  191. package/dist/types/components.d.ts +26 -23
  192. package/dist/types/form-builder.d.ts +6 -8
  193. package/dist/types/index.d.ts +2 -2
  194. package/dist/types/platform.d.ts +7 -1
  195. package/dist/useScheduleDrag-BgzpQT53.js +4414 -0
  196. package/dist/useScheduleDrag-BgzpQT53.js.map +1 -0
  197. package/dist/utils/formModelSync.d.ts +5 -0
  198. package/dist/utils/items.d.ts +8 -0
  199. package/dist/utils/options.d.ts +6 -0
  200. package/dist/utils/pluginIcon.d.ts +9 -0
  201. package/package.json +7 -2
  202. package/src/__tests__/components/ActionItem.test.ts +99 -0
  203. package/src/__tests__/components/AppAvatarMenu.test.ts +27 -0
  204. package/src/__tests__/components/AppLayout.test.ts +44 -0
  205. package/src/__tests__/components/AppPageSelector.test.ts +134 -0
  206. package/src/__tests__/components/AppPillNav.test.ts +125 -0
  207. package/src/__tests__/components/AppPluginSwitcher.test.ts +44 -0
  208. package/src/__tests__/components/AppSidebar.test.ts +496 -0
  209. package/src/__tests__/components/AppToastContainer.test.ts +37 -0
  210. package/src/__tests__/components/AppTopBar.test.ts +455 -9
  211. package/src/__tests__/components/BaseRadioGroup.test.ts +25 -0
  212. package/src/__tests__/components/BaseSelect.test.ts +21 -0
  213. package/src/__tests__/components/BaseTabs.test.ts +25 -0
  214. package/src/__tests__/components/BatchProgressList.test.ts +52 -0
  215. package/src/__tests__/components/BioTemplateExperimentWorkspaceView.test.ts +159 -0
  216. package/src/__tests__/components/BioTemplatePackWorkspaceView.test.ts +175 -0
  217. package/src/__tests__/components/BioTemplatePresetWorkspaceView.test.ts +306 -0
  218. package/src/__tests__/components/BioTemplateRenderer.test.ts +71 -0
  219. package/src/__tests__/components/Breadcrumb.test.ts +23 -0
  220. package/src/__tests__/components/CalendarGridPanel.test.ts +36 -0
  221. package/src/__tests__/components/ComponentBindingRenderer.test.ts +161 -0
  222. package/src/__tests__/components/ConcentrationInput.test.ts +45 -0
  223. package/src/__tests__/components/ControlWorkspaceView.test.ts +1102 -0
  224. package/src/__tests__/components/DataFrame.test.ts +11 -0
  225. package/src/__tests__/components/DatePicker.test.ts +45 -0
  226. package/src/__tests__/components/DateTimePicker.test.ts +48 -0
  227. package/src/__tests__/components/DoseDesignWorkspaceView.test.ts +185 -0
  228. package/src/__tests__/components/DropdownButton.test.ts +23 -0
  229. package/src/__tests__/components/EmptyState.test.ts +23 -0
  230. package/src/__tests__/components/ExperimentPopover.test.ts +56 -0
  231. package/src/__tests__/components/FormBuilder.test.ts +296 -0
  232. package/src/__tests__/components/GroupAssigner.test.ts +30 -0
  233. package/src/__tests__/components/MultiSelect.test.ts +48 -0
  234. package/src/__tests__/components/PluginWorkspaceView.test.ts +548 -0
  235. package/src/__tests__/components/ProtocolStepEditor.test.ts +33 -0
  236. package/src/__tests__/components/ReagentList.test.ts +82 -0
  237. package/src/__tests__/components/SampleHierarchyTree.test.ts +53 -0
  238. package/src/__tests__/components/SampleSelector.test.ts +60 -0
  239. package/src/__tests__/components/SegmentedControl.test.ts +24 -0
  240. package/src/__tests__/components/SettingsModal.test.ts +296 -0
  241. package/src/__tests__/components/TagsInput.test.ts +75 -0
  242. package/src/__tests__/components/ThemeToggle.test.ts +47 -0
  243. package/src/__tests__/components/TimePicker.test.ts +38 -0
  244. package/src/__tests__/composables/experiment-utils.test.ts +30 -0
  245. package/src/__tests__/composables/useApi.test.ts +30 -0
  246. package/src/__tests__/composables/useAppExperiment.test.ts +100 -1
  247. package/src/__tests__/composables/useBioTemplatePackWorkspace.test.ts +125 -0
  248. package/src/__tests__/composables/useBioTemplatePresetWorkspace.test.ts +199 -0
  249. package/src/__tests__/composables/useBioTemplateWorkspace.test.ts +104 -0
  250. package/src/__tests__/composables/useCalendarGrid.test.ts +38 -0
  251. package/src/__tests__/composables/useControlSchema.test.ts +1033 -0
  252. package/src/__tests__/composables/useDebouncedWatch.test.ts +93 -0
  253. package/src/__tests__/composables/useDropdownState.test.ts +95 -0
  254. package/src/__tests__/composables/useEventListener.test.ts +116 -0
  255. package/src/__tests__/composables/useExpansionSet.test.ts +62 -0
  256. package/src/__tests__/composables/useExperimentData.test.ts +4 -0
  257. package/src/__tests__/composables/useExperimentSave.test.ts +203 -8
  258. package/src/__tests__/composables/useExperimentSelector.test.ts +164 -0
  259. package/src/__tests__/composables/useForm.test.ts +58 -0
  260. package/src/__tests__/composables/useFormBuilder.test.ts +77 -0
  261. package/src/__tests__/composables/useGroupAssignment.test.ts +73 -0
  262. package/src/__tests__/composables/useListSelection.test.ts +66 -0
  263. package/src/__tests__/composables/usePluginClient.test.ts +541 -0
  264. package/src/__tests__/composables/usePluginConfig.test.ts +5 -0
  265. package/src/__tests__/composables/useRequestSyncState.test.ts +92 -0
  266. package/src/__tests__/composables/useSampleGroups.test.ts +66 -0
  267. package/src/__tests__/composables/useSelectionLimit.test.ts +41 -0
  268. package/src/__tests__/composables/useSortedItems.test.ts +87 -0
  269. package/src/__tests__/composables/useTemplateCollection.test.ts +147 -0
  270. package/src/__tests__/composables/useTextSearch.test.ts +55 -0
  271. package/src/__tests__/composables/useTheme.test.ts +91 -0
  272. package/src/__tests__/composables/useTimeUtils.test.ts +35 -0
  273. package/src/__tests__/docs/frontendDocsCatalog.test.ts +324 -0
  274. package/src/__tests__/fixtures/templates/dose-response.json +81 -0
  275. package/src/__tests__/fixtures/templates/plate-map.json +54 -0
  276. package/src/__tests__/fixtures/templates/qpcr-plate.json +96 -0
  277. package/src/__tests__/fixtures/templates/sample-sheet.json +71 -0
  278. package/src/__tests__/templates/templates.test.ts +1055 -0
  279. package/src/components/AppAvatarMenu.vue +15 -69
  280. package/src/components/AppLayout.story.vue +64 -25
  281. package/src/components/AppLayout.vue +83 -2
  282. package/src/components/AppPluginSwitcher.vue +41 -145
  283. package/src/components/AppSidebar.story.vue +203 -1
  284. package/src/components/AppSidebar.vue +320 -25
  285. package/src/components/{ToastNotification.story.vue → AppToastContainer.story.vue} +6 -6
  286. package/src/components/{ToastNotification.vue → AppToastContainer.vue} +1 -1
  287. package/src/components/AppTopBar.story.vue +7 -33
  288. package/src/components/AppTopBar.vue +104 -300
  289. package/src/components/BaseModal.vue +3 -5
  290. package/src/components/BaseRadioGroup.vue +7 -3
  291. package/src/components/BaseSelect.vue +11 -7
  292. package/src/components/BaseTabs.vue +6 -4
  293. package/src/components/BatchProgressList.vue +5 -8
  294. package/src/components/BioTemplateExperimentWorkspaceView.story.vue +123 -0
  295. package/src/components/BioTemplateExperimentWorkspaceView.vue +343 -0
  296. package/src/components/BioTemplatePackWorkspaceView.story.vue +107 -0
  297. package/src/components/BioTemplatePackWorkspaceView.vue +177 -0
  298. package/src/components/BioTemplatePresetWorkspaceView.story.vue +163 -0
  299. package/src/components/BioTemplatePresetWorkspaceView.vue +401 -0
  300. package/src/components/BioTemplateRenderer.story.vue +57 -0
  301. package/src/components/BioTemplateRenderer.vue +57 -0
  302. package/src/components/Breadcrumb.vue +14 -8
  303. package/src/components/ComponentBindingRenderer.story.vue +57 -0
  304. package/src/components/ComponentBindingRenderer.vue +308 -0
  305. package/src/components/ConcentrationInput.vue +27 -64
  306. package/src/components/ControlWorkspaceView.story.vue +347 -0
  307. package/src/components/ControlWorkspaceView.vue +378 -0
  308. package/src/components/DataFrame.vue +34 -50
  309. package/src/components/DatePicker.vue +59 -192
  310. package/src/components/DateTimePicker.vue +50 -171
  311. package/src/components/DoseDesignWorkspaceView.story.vue +77 -0
  312. package/src/components/DoseDesignWorkspaceView.vue +255 -0
  313. package/src/components/DropdownButton.vue +14 -32
  314. package/src/components/EmptyState.vue +4 -2
  315. package/src/components/ExperimentPopover.vue +7 -28
  316. package/src/components/ExperimentSelectorModal.vue +6 -5
  317. package/src/components/FormBuilder.story.vue +190 -0
  318. package/src/components/FormBuilder.vue +124 -27
  319. package/src/components/GroupAssigner.vue +24 -56
  320. package/src/components/MultiSelect.vue +17 -12
  321. package/src/components/PlateMapEditor.vue +3 -8
  322. package/src/components/PluginIcon.vue +2 -22
  323. package/src/components/PluginWorkspaceView.story.vue +334 -0
  324. package/src/components/PluginWorkspaceView.vue +708 -0
  325. package/src/components/ProtocolStepEditor.vue +13 -22
  326. package/src/components/ReagentList.vue +25 -33
  327. package/src/components/SampleHierarchyTree.vue +12 -23
  328. package/src/components/SampleSelector.vue +42 -122
  329. package/src/components/SegmentedControl.vue +7 -3
  330. package/src/components/SettingsModal.story.vue +88 -1
  331. package/src/components/SettingsModal.vue +120 -29
  332. package/src/components/TagsInput.vue +29 -14
  333. package/src/components/ThemeToggle.vue +9 -7
  334. package/src/components/TimePicker.vue +19 -41
  335. package/src/components/Tooltip.vue +7 -12
  336. package/src/components/WellPlate.vue +6 -12
  337. package/src/components/index.ts +9 -8
  338. package/src/components/internal/ActionItemInternal.vue +82 -0
  339. package/src/components/internal/AppPageSelectorInternal.vue +128 -0
  340. package/src/components/internal/AppPillNavInternal.vue +194 -0
  341. package/src/components/internal/CalendarGridPanelInternal.vue +120 -0
  342. package/src/components/{FormFieldRenderer.vue → internal/FormFieldRendererInternal.vue} +4 -12
  343. package/src/components/{FormSection.vue → internal/FormSectionRenderer.vue} +6 -18
  344. package/src/components/{WellEditPopup.vue → internal/WellEditPopupInternal.vue} +5 -10
  345. package/src/composables/experiment-utils.ts +26 -0
  346. package/src/composables/index.ts +229 -3
  347. package/src/composables/platformContextHelpers.ts +74 -0
  348. package/src/composables/useApi.ts +9 -2
  349. package/src/composables/useAppExperiment.ts +85 -13
  350. package/src/composables/useBioTemplateComponents.ts +105 -0
  351. package/src/composables/useBioTemplateControls.ts +41 -0
  352. package/src/composables/useBioTemplatePackWorkspace.ts +185 -0
  353. package/src/composables/useBioTemplatePresetWorkspace.ts +326 -0
  354. package/src/composables/useBioTemplateWorkspace.ts +141 -0
  355. package/src/composables/useCalendarGrid.ts +140 -0
  356. package/src/composables/useControlSchema.ts +1362 -0
  357. package/src/composables/useDebouncedWatch.ts +119 -0
  358. package/src/composables/useDropdownState.ts +83 -0
  359. package/src/composables/useEventListener.ts +111 -0
  360. package/src/composables/useExpansionSet.ts +117 -0
  361. package/src/composables/useExperimentData.ts +20 -11
  362. package/src/composables/useExperimentSave.ts +202 -50
  363. package/src/composables/useExperimentSelector.ts +86 -72
  364. package/src/composables/useForm.ts +49 -4
  365. package/src/composables/useFormBuilder.ts +93 -42
  366. package/src/composables/useGroupAssignment.ts +148 -0
  367. package/src/composables/useListSelection.ts +158 -0
  368. package/src/composables/usePluginClient.ts +466 -0
  369. package/src/composables/usePluginConfig.ts +34 -13
  370. package/src/composables/useRequestSyncState.ts +126 -0
  371. package/src/composables/useSampleGroups.ts +126 -0
  372. package/src/composables/useSelectionLimit.ts +57 -0
  373. package/src/composables/useSortedItems.ts +118 -0
  374. package/src/composables/useTemplateCollection.ts +229 -0
  375. package/src/composables/useTextSearch.ts +60 -0
  376. package/src/composables/useTheme.ts +2 -28
  377. package/src/composables/useTimeUtils.ts +26 -2
  378. package/src/composables/useWellPlateEditor.ts +13 -9
  379. package/src/index.ts +11 -348
  380. package/src/install.ts +11 -4
  381. package/src/stores/settings.ts +13 -9
  382. package/src/styles/components/app-layout.css +82 -0
  383. package/src/styles/components/app-page-selector.css +23 -0
  384. package/src/styles/components/app-pill-nav.css +77 -0
  385. package/src/styles/components/app-sidebar.css +119 -0
  386. package/src/styles/components/app-top-bar.css +0 -201
  387. package/src/styles/components/concentration-input.css +3 -142
  388. package/src/styles/components/empty-state.css +0 -16
  389. package/src/styles/components/theme-toggle.css +3 -66
  390. package/src/styles/index.css +0 -2
  391. package/src/templates/adapters.ts +785 -0
  392. package/src/templates/builders.ts +2149 -0
  393. package/src/templates/catalog.ts +245 -0
  394. package/src/templates/componentBindings.ts +653 -0
  395. package/src/templates/controlSchemas.ts +718 -0
  396. package/src/templates/index.ts +318 -0
  397. package/src/templates/lookup.ts +18 -0
  398. package/src/templates/packs.ts +156 -0
  399. package/src/templates/presets.ts +146 -0
  400. package/src/templates/types.ts +668 -0
  401. package/src/types/components.ts +39 -27
  402. package/src/types/form-builder.ts +7 -2
  403. package/src/types/index.ts +13 -3
  404. package/src/types/platform.ts +7 -1
  405. package/src/utils/formModelSync.ts +52 -0
  406. package/src/utils/items.ts +28 -0
  407. package/src/utils/options.ts +23 -0
  408. package/src/utils/pluginIcon.ts +30 -0
  409. package/dist/__tests__/composables/usePluginApi.test.d.ts +0 -13
  410. package/dist/auth-DsI0rQ7_.js.map +0 -1
  411. package/dist/components/GroupingModal.vue.d.ts +0 -12
  412. package/dist/components-_XqPEhP9.js.map +0 -1
  413. package/dist/composables/usePluginApi.d.ts +0 -29
  414. package/dist/composables-tiZqLu1M.js.map +0 -1
  415. package/dist/useScheduleDrag-CA9sGNJG.js +0 -7181
  416. package/dist/useScheduleDrag-CA9sGNJG.js.map +0 -1
  417. package/src/__tests__/composables/usePluginApi.test.ts +0 -81
  418. package/src/components/AppPageSelector.vue +0 -159
  419. package/src/components/AppPillNav.vue +0 -66
  420. package/src/components/GroupingModal.story.vue +0 -52
  421. package/src/components/GroupingModal.vue +0 -422
  422. package/src/components/SettingsButton.story.vue +0 -58
  423. package/src/components/SettingsButton.vue +0 -76
  424. package/src/composables/usePluginApi.ts +0 -39
  425. package/src/styles/components/grouping-modal.css +0 -323
  426. package/src/styles/components/settings-button.css +0 -94
  427. /package/dist/components/{ToastNotification.vue.d.ts → AppToastContainer.vue.d.ts} +0 -0
@@ -0,0 +1,1102 @@
1
+ import { mount } from '@vue/test-utils'
2
+ import { createPinia } from 'pinia'
3
+ import { computed, h, nextTick, ref } from 'vue'
4
+ import { describe, expect, it, vi } from 'vitest'
5
+ import AppLayout from '../../components/AppLayout.vue'
6
+ import AppSidebar from '../../components/AppSidebar.vue'
7
+ import AppTopBar from '../../components/AppTopBar.vue'
8
+ import ControlWorkspaceView from '../../components/ControlWorkspaceView.vue'
9
+ import FormActions from '../../components/FormActions.vue'
10
+ import FormBuilder from '../../components/FormBuilder.vue'
11
+ import {
12
+ defineControlModel,
13
+ defineControls,
14
+ defineDoseDesignControlModel,
15
+ useControlWorkspace,
16
+ } from '../../composables/useControlSchema'
17
+
18
+ vi.mock('../../composables/usePlatformContext', () => ({
19
+ usePlatformContext: vi.fn(() => ({
20
+ isIntegrated: computed(() => false),
21
+ context: ref({ isIntegrated: false, theme: 'system' }),
22
+ plugin: computed(() => undefined),
23
+ user: computed(() => undefined),
24
+ theme: computed(() => 'system' as const),
25
+ features: computed(() => undefined),
26
+ navigate: vi.fn(),
27
+ notify: vi.fn(),
28
+ sendToPlatform: vi.fn(),
29
+ })),
30
+ }))
31
+
32
+ vi.mock('../../composables/useTheme', () => ({
33
+ useTheme: vi.fn(() => ({
34
+ isDark: ref(false),
35
+ theme: ref('light'),
36
+ toggleTheme: vi.fn(),
37
+ })),
38
+ }))
39
+
40
+ function createWorkspace() {
41
+ return useControlWorkspace(defineControls({
42
+ threshold: {
43
+ type: 'number',
44
+ default: 0.05,
45
+ section: 'parameters',
46
+ view: 'analysis',
47
+ },
48
+ chartScale: {
49
+ default: 'linear',
50
+ options: ['linear', 'log'],
51
+ section: 'display',
52
+ view: 'results',
53
+ },
54
+ }), {
55
+ views: {
56
+ analysis: { label: 'Run' },
57
+ results: { label: 'Results' },
58
+ },
59
+ })
60
+ }
61
+
62
+ function mountView(workspace = createWorkspace()) {
63
+ return mount(ControlWorkspaceView, {
64
+ props: {
65
+ workspace,
66
+ title: 'Analysis Workspace',
67
+ },
68
+ global: {
69
+ plugins: [createPinia()],
70
+ stubs: {
71
+ 'router-link': {
72
+ template: '<a><slot /></a>',
73
+ },
74
+ BaseModal: {
75
+ template: '<div><slot /></div>',
76
+ },
77
+ },
78
+ },
79
+ })
80
+ }
81
+
82
+ const globalOptions = {
83
+ plugins: [createPinia()],
84
+ stubs: {
85
+ 'router-link': {
86
+ template: '<a><slot /></a>',
87
+ },
88
+ BaseModal: {
89
+ template: '<div><slot /></div>',
90
+ },
91
+ },
92
+ }
93
+
94
+ describe('ControlWorkspaceView', () => {
95
+ it('renders AppTopBar, AppSidebar, and FormBuilder from one workspace object', () => {
96
+ const workspace = createWorkspace()
97
+ const wrapper = mountView(workspace)
98
+
99
+ expect(wrapper.findComponent(AppTopBar).props('title')).toBe('Analysis Workspace')
100
+ expect('tabs' in wrapper.findComponent(AppTopBar).props()).toBe(false)
101
+ expect(wrapper.findComponent(AppTopBar).props('pillNav')).toEqual([
102
+ { id: 'analysis', label: 'Run' },
103
+ { id: 'results', label: 'Results' },
104
+ ])
105
+ expect(wrapper.findComponent(AppTopBar).props('currentPillId')).toBe('analysis')
106
+ expect(wrapper.findComponent(AppLayout).props('responsiveSidebar')).toBe(true)
107
+ expect(wrapper.findComponent(AppSidebar).props('activeView')).toBe('analysis')
108
+ expect(wrapper.findComponent(AppSidebar).props('variant')).toBe('analysis')
109
+ expect(wrapper.findComponent(FormBuilder).props('modelValue')).toMatchObject(workspace.values)
110
+ })
111
+
112
+ it('lets callers override the generated sidebar chrome', () => {
113
+ const wrapper = mount(ControlWorkspaceView, {
114
+ props: {
115
+ workspace: createWorkspace(),
116
+ sidebarVariant: 'default',
117
+ responsiveSidebar: false,
118
+ sidebarTitle: 'Peak Picking',
119
+ sidebarSubtitle: 'Current experiment',
120
+ sidebarBadge: 3,
121
+ },
122
+ global: globalOptions,
123
+ })
124
+
125
+ expect(wrapper.findComponent(AppLayout).props('responsiveSidebar')).toBe(false)
126
+ expect(wrapper.findComponent(AppSidebar).props('variant')).toBe('default')
127
+ expect(wrapper.findComponent(AppSidebar).props('title')).toBe('Peak Picking')
128
+ expect(wrapper.findComponent(AppSidebar).props('subtitle')).toBe('Current experiment')
129
+ expect(wrapper.findComponent(AppSidebar).props('badge')).toBe(3)
130
+ })
131
+
132
+ it('can create the workspace internally from direct controls', async () => {
133
+ const controls = defineControls({
134
+ threshold: {
135
+ type: 'number',
136
+ default: 0.05,
137
+ section: 'parameters',
138
+ view: 'analysis',
139
+ },
140
+ chartScale: {
141
+ default: 'linear',
142
+ options: ['linear', 'log'],
143
+ section: 'display',
144
+ view: 'results',
145
+ },
146
+ })
147
+
148
+ const wrapper = mount(ControlWorkspaceView, {
149
+ props: {
150
+ controls,
151
+ controlOptions: {
152
+ views: {
153
+ analysis: { label: 'Run' },
154
+ results: { label: 'Results' },
155
+ },
156
+ },
157
+ title: 'Direct Workspace',
158
+ },
159
+ global: globalOptions,
160
+ })
161
+
162
+ expect(wrapper.findComponent(AppTopBar).props('title')).toBe('Direct Workspace')
163
+ expect(wrapper.findComponent(AppTopBar).props('pillNav')).toEqual([
164
+ { id: 'analysis', label: 'Run' },
165
+ { id: 'results', label: 'Results' },
166
+ ])
167
+ expect(wrapper.findComponent(AppSidebar).props('activeView')).toBe('analysis')
168
+ expect(wrapper.findComponent(FormBuilder).props('modelValue')).toMatchObject({
169
+ threshold: 0.05,
170
+ chartScale: 'linear',
171
+ })
172
+
173
+ wrapper.findComponent(AppTopBar).vm.$emit('pill-select', { id: 'results', label: 'Results' })
174
+ await nextTick()
175
+
176
+ expect(wrapper.findComponent(AppTopBar).props('currentPillId')).toBe('results')
177
+ expect(wrapper.findComponent(AppSidebar).props('activeView')).toBe('results')
178
+ })
179
+
180
+ it('accepts a nested control model through direct v-bind props', () => {
181
+ const model = defineControlModel({
182
+ views: {
183
+ design: {
184
+ label: 'Design',
185
+ sections: {
186
+ dose: {
187
+ label: 'Dose design',
188
+ controls: {
189
+ replicates: {
190
+ type: 'number',
191
+ default: 3,
192
+ min: 1,
193
+ },
194
+ },
195
+ },
196
+ },
197
+ },
198
+ results: {
199
+ label: 'Results',
200
+ section: 'plots',
201
+ sectionLabel: 'Plots',
202
+ controls: {
203
+ chartScale: ['linear', 'log'],
204
+ },
205
+ },
206
+ },
207
+ })
208
+
209
+ const wrapper = mount(ControlWorkspaceView, {
210
+ props: {
211
+ ...model,
212
+ title: 'Nested Model Workspace',
213
+ },
214
+ global: globalOptions,
215
+ })
216
+
217
+ expect(wrapper.findComponent(AppTopBar).props('pillNav')).toEqual([
218
+ { id: 'design', label: 'Design' },
219
+ { id: 'results', label: 'Results' },
220
+ ])
221
+ expect(wrapper.findComponent(AppSidebar).props('activeView')).toBe('design')
222
+ expect(wrapper.findComponent(AppSidebar).props('panels')).toMatchObject({
223
+ design: [{ id: 'design-dose', label: 'Dose design' }],
224
+ results: [{ id: 'plots', label: 'Plots' }],
225
+ })
226
+ expect(wrapper.findComponent(FormBuilder).props('modelValue')).toMatchObject({
227
+ replicates: 3,
228
+ chartScale: 'linear',
229
+ })
230
+ })
231
+
232
+ it('accepts a nested control model through one model prop', () => {
233
+ const model = defineControlModel({
234
+ views: {
235
+ design: {
236
+ label: 'Design',
237
+ sections: {
238
+ dose: {
239
+ label: 'Dose design',
240
+ controls: {
241
+ replicates: {
242
+ type: 'number',
243
+ default: 3,
244
+ min: 1,
245
+ },
246
+ },
247
+ },
248
+ },
249
+ },
250
+ results: {
251
+ label: 'Results',
252
+ section: 'plots',
253
+ sectionLabel: 'Plots',
254
+ controls: {
255
+ chartScale: ['linear', 'log'],
256
+ },
257
+ },
258
+ },
259
+ })
260
+
261
+ const wrapper = mount(ControlWorkspaceView, {
262
+ props: {
263
+ model,
264
+ title: 'Model Workspace',
265
+ },
266
+ global: globalOptions,
267
+ })
268
+
269
+ expect(wrapper.findComponent(AppTopBar).props('pillNav')).toEqual([
270
+ { id: 'design', label: 'Design' },
271
+ { id: 'results', label: 'Results' },
272
+ ])
273
+ expect(wrapper.findComponent(AppSidebar).props('panels')).toMatchObject({
274
+ design: [{ id: 'design-dose', label: 'Dose design' }],
275
+ results: [{ id: 'plots', label: 'Plots' }],
276
+ })
277
+ expect(wrapper.findComponent(FormBuilder).props('modelValue')).toMatchObject({
278
+ replicates: 3,
279
+ chartScale: 'linear',
280
+ })
281
+ })
282
+
283
+ it('merges model and prop controlOptions initial values', () => {
284
+ const model = defineControlModel({
285
+ initialValues: {
286
+ replicates: 4,
287
+ chartScale: 'log',
288
+ },
289
+ views: {
290
+ design: {
291
+ label: 'Design',
292
+ sections: {
293
+ dose: {
294
+ label: 'Dose design',
295
+ controls: {
296
+ replicates: {
297
+ type: 'number',
298
+ default: 3,
299
+ min: 1,
300
+ },
301
+ },
302
+ },
303
+ },
304
+ },
305
+ results: {
306
+ label: 'Results',
307
+ section: 'plots',
308
+ sectionLabel: 'Plots',
309
+ controls: {
310
+ chartScale: ['linear', 'log'],
311
+ },
312
+ },
313
+ },
314
+ })
315
+
316
+ const wrapper = mount(ControlWorkspaceView, {
317
+ props: {
318
+ model,
319
+ controlOptions: {
320
+ initialValues: {
321
+ chartScale: 'linear',
322
+ },
323
+ },
324
+ },
325
+ global: globalOptions,
326
+ })
327
+
328
+ expect(wrapper.findComponent(FormBuilder).props('modelValue')).toMatchObject({
329
+ replicates: 4,
330
+ chartScale: 'linear',
331
+ })
332
+ })
333
+
334
+ it('accepts a raw nested control model through one model prop', () => {
335
+ const wrapper = mount(ControlWorkspaceView, {
336
+ props: {
337
+ model: {
338
+ views: {
339
+ setup: {
340
+ label: 'Setup',
341
+ sections: {
342
+ parameters: {
343
+ label: 'Parameters',
344
+ controls: {
345
+ threshold: { type: 'number', default: 0.05 },
346
+ normalize: true,
347
+ },
348
+ },
349
+ },
350
+ },
351
+ review: {
352
+ label: 'Review',
353
+ controls: {
354
+ chartScale: ['linear', 'log'],
355
+ },
356
+ },
357
+ },
358
+ },
359
+ title: 'Raw Model Workspace',
360
+ },
361
+ global: globalOptions,
362
+ })
363
+
364
+ expect(wrapper.findComponent(AppTopBar).props('title')).toBe('Raw Model Workspace')
365
+ expect(wrapper.findComponent(AppTopBar).props('pillNav')).toEqual([
366
+ { id: 'setup', label: 'Setup' },
367
+ { id: 'review', label: 'Review' },
368
+ ])
369
+ expect(wrapper.findComponent(AppSidebar).props('panels')).toMatchObject({
370
+ setup: [{ id: 'setup-parameters', label: 'Parameters' }],
371
+ review: [{ id: 'review-controls', label: 'Review' }],
372
+ })
373
+ expect(wrapper.findComponent(FormBuilder).props('modelValue')).toMatchObject({
374
+ threshold: 0.05,
375
+ normalize: true,
376
+ chartScale: 'linear',
377
+ })
378
+ })
379
+
380
+ it('resolves component props from a dose design model prop', () => {
381
+ const model = defineDoseDesignControlModel({
382
+ selectedWells: ['A1', 'A2'],
383
+ includeMolecularWeight: true,
384
+ })
385
+
386
+ const wrapper = mount(ControlWorkspaceView, {
387
+ props: {
388
+ model,
389
+ },
390
+ slots: {
391
+ default: ({ componentBindings, componentPropsById }) => h(
392
+ 'pre',
393
+ { class: 'component-props-by-id' },
394
+ JSON.stringify({ componentBindings, componentPropsById }),
395
+ ),
396
+ },
397
+ global: globalOptions,
398
+ })
399
+
400
+ expect(JSON.parse(wrapper.find('.component-props-by-id').text())).toEqual({
401
+ componentBindings: [
402
+ {
403
+ id: 'plate',
404
+ component: 'WellPlate',
405
+ props: {
406
+ modelValue: ['A1', 'A2'],
407
+ format: 96,
408
+ disabled: false,
409
+ },
410
+ },
411
+ {
412
+ id: 'dose',
413
+ component: 'DoseCalculator',
414
+ props: {
415
+ mode: 'serial',
416
+ targetWells: ['A1', 'A2'],
417
+ disabled: false,
418
+ molecularWeight: 300,
419
+ },
420
+ },
421
+ ],
422
+ componentPropsById: {
423
+ plate: {
424
+ modelValue: ['A1', 'A2'],
425
+ format: 96,
426
+ disabled: false,
427
+ },
428
+ dose: {
429
+ mode: 'serial',
430
+ targetWells: ['A1', 'A2'],
431
+ disabled: false,
432
+ molecularWeight: 300,
433
+ },
434
+ },
435
+ })
436
+ })
437
+
438
+ it('accepts component prop mappings from a nested control model', async () => {
439
+ const model = defineControlModel({
440
+ views: {
441
+ design: {
442
+ label: 'Dose design',
443
+ controls: {
444
+ doseMode: {
445
+ default: 'serial',
446
+ options: ['serial', 'dilution'],
447
+ },
448
+ targetWells: {
449
+ type: 'tags',
450
+ default: ['A1', 'A2'],
451
+ },
452
+ enabled: true,
453
+ },
454
+ },
455
+ },
456
+ componentProps: {
457
+ mode: 'doseMode',
458
+ targetWells: 'targetWells',
459
+ disabled: values => !values.enabled,
460
+ },
461
+ componentPropsById: {
462
+ dose: {
463
+ mode: 'doseMode',
464
+ targetWells: 'targetWells',
465
+ disabled: values => !values.enabled,
466
+ },
467
+ plate: {
468
+ modelValue: 'targetWells',
469
+ disabled: values => !values.enabled,
470
+ },
471
+ },
472
+ componentBindings: {
473
+ dose: {
474
+ component: 'DoseCalculator',
475
+ props: {
476
+ mode: 'doseMode',
477
+ targetWells: 'targetWells',
478
+ disabled: values => !values.enabled,
479
+ },
480
+ },
481
+ plate: {
482
+ component: 'WellPlate',
483
+ props: {
484
+ modelValue: 'targetWells',
485
+ disabled: values => !values.enabled,
486
+ },
487
+ },
488
+ },
489
+ })
490
+
491
+ const wrapper = mount(ControlWorkspaceView, {
492
+ props: {
493
+ ...model,
494
+ },
495
+ slots: {
496
+ default: ({ componentBindingsById, componentProps, componentPropsById }) => h(
497
+ 'pre',
498
+ { class: 'component-props' },
499
+ JSON.stringify({ componentBindingsById, componentProps, componentPropsById }),
500
+ ),
501
+ },
502
+ global: globalOptions,
503
+ })
504
+
505
+ expect(JSON.parse(wrapper.find('.component-props').text())).toEqual({
506
+ componentBindingsById: {
507
+ dose: {
508
+ id: 'dose',
509
+ component: 'DoseCalculator',
510
+ props: {
511
+ mode: 'serial',
512
+ targetWells: ['A1', 'A2'],
513
+ disabled: false,
514
+ },
515
+ },
516
+ plate: {
517
+ id: 'plate',
518
+ component: 'WellPlate',
519
+ props: {
520
+ modelValue: ['A1', 'A2'],
521
+ disabled: false,
522
+ },
523
+ },
524
+ },
525
+ componentProps: {
526
+ mode: 'serial',
527
+ targetWells: ['A1', 'A2'],
528
+ disabled: false,
529
+ },
530
+ componentPropsById: {
531
+ dose: {
532
+ mode: 'serial',
533
+ targetWells: ['A1', 'A2'],
534
+ disabled: false,
535
+ },
536
+ plate: {
537
+ modelValue: ['A1', 'A2'],
538
+ disabled: false,
539
+ },
540
+ },
541
+ })
542
+
543
+ wrapper.findComponent(FormBuilder).vm.$emit('update:modelValue', {
544
+ doseMode: 'dilution',
545
+ targetWells: ['B1'],
546
+ enabled: false,
547
+ })
548
+ await nextTick()
549
+
550
+ expect(JSON.parse(wrapper.find('.component-props').text())).toEqual({
551
+ componentBindingsById: {
552
+ dose: {
553
+ id: 'dose',
554
+ component: 'DoseCalculator',
555
+ props: {
556
+ mode: 'dilution',
557
+ targetWells: ['B1'],
558
+ disabled: true,
559
+ },
560
+ },
561
+ plate: {
562
+ id: 'plate',
563
+ component: 'WellPlate',
564
+ props: {
565
+ modelValue: ['B1'],
566
+ disabled: true,
567
+ },
568
+ },
569
+ },
570
+ componentProps: {
571
+ mode: 'dilution',
572
+ targetWells: ['B1'],
573
+ disabled: true,
574
+ },
575
+ componentPropsById: {
576
+ dose: {
577
+ mode: 'dilution',
578
+ targetWells: ['B1'],
579
+ disabled: true,
580
+ },
581
+ plate: {
582
+ modelValue: ['B1'],
583
+ disabled: true,
584
+ },
585
+ },
586
+ })
587
+ })
588
+
589
+ it('accepts initial values directly for internally generated workspaces', () => {
590
+ const controls = defineControls({
591
+ threshold: {
592
+ type: 'number',
593
+ default: 0.05,
594
+ section: 'parameters',
595
+ },
596
+ chartScale: {
597
+ default: 'linear',
598
+ options: ['linear', 'log'],
599
+ section: 'display',
600
+ },
601
+ sampleNames: {
602
+ type: 'tags',
603
+ default: ['Control'],
604
+ section: 'samples',
605
+ },
606
+ enabled: true,
607
+ })
608
+
609
+ const wrapper = mount(ControlWorkspaceView, {
610
+ props: {
611
+ controls,
612
+ controlOptions: {
613
+ initialValues: {
614
+ threshold: 0.2,
615
+ chartScale: 'linear',
616
+ },
617
+ },
618
+ initialValues: {
619
+ chartScale: 'log',
620
+ },
621
+ },
622
+ global: globalOptions,
623
+ })
624
+
625
+ expect(wrapper.findComponent(FormBuilder).props('modelValue')).toMatchObject({
626
+ threshold: 0.2,
627
+ chartScale: 'log',
628
+ enabled: true,
629
+ })
630
+ })
631
+
632
+ it('syncs internally generated workspace values with v-model:values', async () => {
633
+ const controls = defineControls({
634
+ threshold: {
635
+ type: 'number',
636
+ default: 0.05,
637
+ section: 'parameters',
638
+ },
639
+ chartScale: {
640
+ default: 'linear',
641
+ options: ['linear', 'log'],
642
+ section: 'display',
643
+ },
644
+ enabled: true,
645
+ })
646
+
647
+ const wrapper = mount(ControlWorkspaceView, {
648
+ props: {
649
+ controls,
650
+ values: {
651
+ threshold: 0.2,
652
+ chartScale: 'log',
653
+ sampleNames: ['Vehicle', 'Drug'],
654
+ enabled: true,
655
+ },
656
+ },
657
+ global: globalOptions,
658
+ })
659
+
660
+ expect(wrapper.findComponent(FormBuilder).props('modelValue')).toMatchObject({
661
+ threshold: 0.2,
662
+ chartScale: 'log',
663
+ sampleNames: ['Vehicle', 'Drug'],
664
+ enabled: true,
665
+ })
666
+
667
+ wrapper.findComponent(FormBuilder).vm.$emit('update:modelValue', {
668
+ threshold: 0.5,
669
+ chartScale: 'linear',
670
+ sampleNames: ['Vehicle', 'Dose A'],
671
+ enabled: false,
672
+ })
673
+ await nextTick()
674
+
675
+ expect(wrapper.emitted('update:values')?.at(-1)).toEqual([{
676
+ threshold: 0.5,
677
+ chartScale: 'linear',
678
+ sampleNames: ['Vehicle', 'Dose A'],
679
+ enabled: false,
680
+ }])
681
+
682
+ await wrapper.setProps({
683
+ values: {
684
+ threshold: 0.8,
685
+ chartScale: 'log',
686
+ sampleNames: ['Vehicle', 'Dose A'],
687
+ enabled: true,
688
+ },
689
+ })
690
+
691
+ expect(wrapper.findComponent(FormBuilder).props('modelValue')).toMatchObject({
692
+ threshold: 0.8,
693
+ chartScale: 'log',
694
+ sampleNames: ['Vehicle', 'Dose A'],
695
+ enabled: true,
696
+ })
697
+ })
698
+
699
+ it('syncs internally generated workspace values with default v-model', async () => {
700
+ const controls = defineControls({
701
+ threshold: {
702
+ type: 'number',
703
+ default: 0.05,
704
+ section: 'parameters',
705
+ },
706
+ chartScale: {
707
+ default: 'linear',
708
+ options: ['linear', 'log'],
709
+ section: 'display',
710
+ },
711
+ })
712
+
713
+ const wrapper = mount(ControlWorkspaceView, {
714
+ props: {
715
+ controls,
716
+ modelValue: {
717
+ threshold: 0.2,
718
+ chartScale: 'log',
719
+ },
720
+ },
721
+ global: globalOptions,
722
+ })
723
+
724
+ expect(wrapper.findComponent(FormBuilder).props('modelValue')).toMatchObject({
725
+ threshold: 0.2,
726
+ chartScale: 'log',
727
+ })
728
+
729
+ wrapper.findComponent(FormBuilder).vm.$emit('update:modelValue', {
730
+ threshold: 0.5,
731
+ chartScale: 'linear',
732
+ })
733
+ await nextTick()
734
+
735
+ expect(wrapper.emitted('update:modelValue')?.at(-1)).toEqual([{
736
+ threshold: 0.5,
737
+ chartScale: 'linear',
738
+ }])
739
+ expect(wrapper.emitted('update:values')?.at(-1)).toEqual([{
740
+ threshold: 0.5,
741
+ chartScale: 'linear',
742
+ }])
743
+
744
+ await wrapper.setProps({
745
+ modelValue: {
746
+ threshold: 0.8,
747
+ chartScale: 'log',
748
+ },
749
+ })
750
+
751
+ expect(wrapper.findComponent(FormBuilder).props('modelValue')).toMatchObject({
752
+ threshold: 0.8,
753
+ chartScale: 'log',
754
+ })
755
+ })
756
+
757
+ it('activates direct controls that are loaded after mount', async () => {
758
+ const controls = defineControls({
759
+ threshold: {
760
+ type: 'number',
761
+ default: 0.05,
762
+ section: 'parameters',
763
+ view: 'analysis',
764
+ },
765
+ method: {
766
+ default: 'linear',
767
+ options: ['linear', 'logistic'],
768
+ section: 'parameters',
769
+ view: 'analysis',
770
+ },
771
+ })
772
+
773
+ const wrapper = mount(ControlWorkspaceView, {
774
+ props: {
775
+ title: 'Async Workspace',
776
+ },
777
+ global: globalOptions,
778
+ })
779
+
780
+ expect(wrapper.findComponent(AppSidebar).props('activeView')).toBe('')
781
+
782
+ await wrapper.setProps({
783
+ controls,
784
+ controlOptions: {
785
+ views: {
786
+ analysis: { label: 'Run' },
787
+ },
788
+ },
789
+ })
790
+
791
+ expect(wrapper.findComponent(AppTopBar).props('pillNav')).toEqual([
792
+ { id: 'analysis', label: 'Run' },
793
+ ])
794
+ expect(wrapper.findComponent(AppSidebar).props('activeView')).toBe('analysis')
795
+ expect(wrapper.findComponent(FormBuilder).props('modelValue')).toMatchObject({
796
+ threshold: 0.05,
797
+ method: 'linear',
798
+ })
799
+ })
800
+
801
+ it('preserves generated values and active view when direct controls change', async () => {
802
+ const controls = defineControls({
803
+ threshold: {
804
+ type: 'number',
805
+ default: 0.05,
806
+ section: 'parameters',
807
+ view: 'analysis',
808
+ },
809
+ chartScale: {
810
+ default: 'linear',
811
+ options: ['linear', 'log'],
812
+ section: 'display',
813
+ view: 'results',
814
+ },
815
+ })
816
+ const nextControls = defineControls({
817
+ threshold: {
818
+ type: 'number',
819
+ default: 0.05,
820
+ section: 'parameters',
821
+ view: 'analysis',
822
+ },
823
+ chartScale: {
824
+ default: 'linear',
825
+ options: ['linear', 'log'],
826
+ section: 'display',
827
+ view: 'results',
828
+ },
829
+ normalize: {
830
+ default: true,
831
+ section: 'display',
832
+ view: 'results',
833
+ },
834
+ })
835
+
836
+ const wrapper = mount(ControlWorkspaceView, {
837
+ props: {
838
+ controls,
839
+ controlOptions: {
840
+ views: {
841
+ analysis: { label: 'Run' },
842
+ results: { label: 'Results' },
843
+ },
844
+ },
845
+ },
846
+ global: globalOptions,
847
+ })
848
+
849
+ wrapper.findComponent(FormBuilder).vm.$emit('update:modelValue', {
850
+ threshold: 0.5,
851
+ chartScale: 'log',
852
+ })
853
+ wrapper.findComponent(AppTopBar).vm.$emit('pill-select', { id: 'results', label: 'Results' })
854
+ await nextTick()
855
+
856
+ await wrapper.setProps({ controls: nextControls })
857
+
858
+ expect(wrapper.findComponent(AppTopBar).props('currentPillId')).toBe('results')
859
+ expect(wrapper.findComponent(AppSidebar).props('activeView')).toBe('results')
860
+ expect(wrapper.findComponent(FormBuilder).props('modelValue')).toMatchObject({
861
+ threshold: 0.5,
862
+ chartScale: 'log',
863
+ normalize: true,
864
+ })
865
+ })
866
+
867
+ it('forwards default FormBuilder submit and cancel events', async () => {
868
+ const controls = defineControls({
869
+ threshold: {
870
+ type: 'number',
871
+ default: 0.05,
872
+ section: 'parameters',
873
+ },
874
+ })
875
+ const wrapper = mount(ControlWorkspaceView, {
876
+ props: {
877
+ controls,
878
+ controlOptions: {
879
+ showCancel: true,
880
+ },
881
+ showFormActions: true,
882
+ },
883
+ global: globalOptions,
884
+ })
885
+
886
+ wrapper.findComponent(FormActions).vm.$emit('submit')
887
+ wrapper.findComponent(FormActions).vm.$emit('cancel')
888
+ await nextTick()
889
+
890
+ expect(wrapper.emitted('submit')).toEqual([[{ threshold: 0.05 }]])
891
+ expect(wrapper.emitted('cancel')).toEqual([[]])
892
+ })
893
+
894
+ it('passes generated form state and enhancements to sidebar and content forms', () => {
895
+ const controls = defineControls({
896
+ threshold: {
897
+ type: 'number',
898
+ default: 0.05,
899
+ section: 'parameters',
900
+ view: 'analysis',
901
+ },
902
+ })
903
+ const formEnhancements = {
904
+ onFieldChange: vi.fn(),
905
+ }
906
+ const wrapper = mount(ControlWorkspaceView, {
907
+ props: {
908
+ controls,
909
+ formEnhancements,
910
+ formLoading: true,
911
+ formDisabled: true,
912
+ formReadonly: true,
913
+ },
914
+ global: globalOptions,
915
+ })
916
+
917
+ const sidebar = wrapper.findComponent(AppSidebar)
918
+ expect(sidebar.props('formEnhancements')).toMatchObject(formEnhancements)
919
+ expect(sidebar.props('formLoading')).toBe(true)
920
+ expect(sidebar.props('formDisabled')).toBe(true)
921
+ expect(sidebar.props('formReadonly')).toBe(true)
922
+
923
+ for (const form of wrapper.findAllComponents(FormBuilder)) {
924
+ expect(form.props('enhancements')).toMatchObject(formEnhancements)
925
+ expect(form.props('loading')).toBe(true)
926
+ expect(form.props('disabled')).toBe(true)
927
+ expect(form.props('readonly')).toBe(true)
928
+ }
929
+ })
930
+
931
+ it('keeps topbar pill navigation and sidebar view synchronized', async () => {
932
+ const workspace = createWorkspace()
933
+ const wrapper = mountView(workspace)
934
+
935
+ wrapper.findComponent(AppTopBar).vm.$emit('pill-select', { id: 'results', label: 'Results' })
936
+ await nextTick()
937
+
938
+ expect(workspace.activeView.value).toBe('results')
939
+ expect(wrapper.findComponent(AppTopBar).props('currentPillId')).toBe('results')
940
+ expect(wrapper.findComponent(AppSidebar).props('activeView')).toBe('results')
941
+ })
942
+
943
+ it('exposes workspace data to the default slot', () => {
944
+ const workspace = createWorkspace()
945
+ const wrapper = mount(ControlWorkspaceView, {
946
+ props: { workspace },
947
+ slots: {
948
+ default: '<p class="custom-content">Custom analysis content</p>',
949
+ },
950
+ global: {
951
+ plugins: [createPinia()],
952
+ stubs: {
953
+ 'router-link': {
954
+ template: '<a><slot /></a>',
955
+ },
956
+ BaseModal: {
957
+ template: '<div><slot /></div>',
958
+ },
959
+ },
960
+ },
961
+ })
962
+
963
+ expect(wrapper.find('.custom-content').text()).toBe('Custom analysis content')
964
+ expect(wrapper.findAllComponents(FormBuilder)).toHaveLength(1)
965
+ })
966
+
967
+ it('exposes mapped component props to the default slot', async () => {
968
+ const controls = defineControls({
969
+ sampleNames: {
970
+ type: 'tags',
971
+ default: ['Control', 'Treatment'],
972
+ section: 'samples',
973
+ },
974
+ enabled: true,
975
+ })
976
+ const wrapper = mount(ControlWorkspaceView, {
977
+ props: {
978
+ controls,
979
+ componentProps: {
980
+ samples: 'sampleNames',
981
+ disabled: values => !values.enabled,
982
+ },
983
+ },
984
+ slots: {
985
+ default: ({ componentProps }) => h(
986
+ 'pre',
987
+ { class: 'component-props' },
988
+ JSON.stringify(componentProps),
989
+ ),
990
+ },
991
+ global: globalOptions,
992
+ })
993
+
994
+ expect(JSON.parse(wrapper.find('.component-props').text())).toEqual({
995
+ samples: ['Control', 'Treatment'],
996
+ disabled: false,
997
+ })
998
+
999
+ wrapper.findComponent(FormBuilder).vm.$emit('update:modelValue', {
1000
+ sampleNames: ['Vehicle'],
1001
+ enabled: false,
1002
+ })
1003
+ await nextTick()
1004
+
1005
+ expect(JSON.parse(wrapper.find('.component-props').text())).toEqual({
1006
+ samples: ['Vehicle'],
1007
+ disabled: true,
1008
+ })
1009
+ })
1010
+
1011
+ it('exposes resolved workspace bindings to custom slots', () => {
1012
+ const model = defineControlModel({
1013
+ views: {
1014
+ analysis: {
1015
+ label: 'Run',
1016
+ controls: {
1017
+ threshold: {
1018
+ type: 'number',
1019
+ default: 0.05,
1020
+ },
1021
+ enabled: true,
1022
+ },
1023
+ },
1024
+ },
1025
+ componentProps: {
1026
+ score: 'threshold',
1027
+ disabled: values => !values.enabled,
1028
+ },
1029
+ componentPropsById: {
1030
+ chart: {
1031
+ score: 'threshold',
1032
+ disabled: values => !values.enabled,
1033
+ },
1034
+ },
1035
+ componentBindings: {
1036
+ chart: {
1037
+ component: 'ResultChart',
1038
+ props: {
1039
+ score: 'threshold',
1040
+ disabled: values => !values.enabled,
1041
+ },
1042
+ },
1043
+ },
1044
+ })
1045
+
1046
+ const wrapper = mount(ControlWorkspaceView, {
1047
+ props: {
1048
+ model,
1049
+ },
1050
+ slots: {
1051
+ topbar: ({ bindings, topBar }) => h(
1052
+ 'pre',
1053
+ { class: 'topbar-bindings' },
1054
+ JSON.stringify({
1055
+ slotPillId: topBar.currentPillId,
1056
+ pillId: bindings.topBar.value.currentPillId,
1057
+ }),
1058
+ ),
1059
+ sidebar: ({ bindings, sidebar }) => h(
1060
+ 'pre',
1061
+ { class: 'sidebar-bindings' },
1062
+ JSON.stringify({
1063
+ activeView: sidebar.activeView,
1064
+ bindingActiveView: bindings.sidebar.activeView,
1065
+ }),
1066
+ ),
1067
+ default: ({ bindings, componentBindingsById, componentPropsById }) => h(
1068
+ 'pre',
1069
+ { class: 'default-bindings' },
1070
+ JSON.stringify({
1071
+ formThreshold: bindings.form.modelValue.threshold,
1072
+ component: componentBindingsById.chart.component,
1073
+ bindingComponent: bindings.componentBindingsById.value.chart.component,
1074
+ bindingScore: bindings.componentProps.value.score,
1075
+ bindingComponentScore: bindings.componentBindings.value[0].props.score,
1076
+ slotScore: componentPropsById.chart.score,
1077
+ bindingSlotScore: bindings.componentPropsById.value.chart.score,
1078
+ }),
1079
+ ),
1080
+ },
1081
+ global: globalOptions,
1082
+ })
1083
+
1084
+ expect(JSON.parse(wrapper.find('.topbar-bindings').text())).toEqual({
1085
+ slotPillId: 'analysis',
1086
+ pillId: 'analysis',
1087
+ })
1088
+ expect(JSON.parse(wrapper.find('.sidebar-bindings').text())).toEqual({
1089
+ activeView: 'analysis',
1090
+ bindingActiveView: 'analysis',
1091
+ })
1092
+ expect(JSON.parse(wrapper.find('.default-bindings').text())).toEqual({
1093
+ formThreshold: 0.05,
1094
+ component: 'ResultChart',
1095
+ bindingComponent: 'ResultChart',
1096
+ bindingScore: 0.05,
1097
+ bindingComponentScore: 0.05,
1098
+ slotScore: 0.05,
1099
+ bindingSlotScore: 0.05,
1100
+ })
1101
+ })
1102
+ })