@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,1033 @@
1
+ import { describe, expect, it } from 'vitest'
2
+ import { nextTick } from 'vue'
3
+ import {
4
+ controlsToFormSchema,
5
+ controlsToSectionFormSchema,
6
+ controlsToSectionFormSchemas,
7
+ controlsToSidebarPanels,
8
+ controlsToSettingsSchema,
9
+ controlsToTopBarSettingsConfig,
10
+ controlsToViewIds,
11
+ controlsToViewItems,
12
+ controlValuesToComponentBindings,
13
+ controlValuesToComponentBindingsById,
14
+ controlValuesToComponentProps,
15
+ defineControlComponentBindings,
16
+ defineControlModel,
17
+ defineDoseDesignControlModel,
18
+ defineDoseCalculatorControlProps,
19
+ defineControls,
20
+ defineWellPlateControlProps,
21
+ defineWellPlateDoseComponentBindings,
22
+ defineWellPlateDoseControlProps,
23
+ getDefaultControlView,
24
+ getControlDefaults,
25
+ useControlSchema,
26
+ useControlWorkspace,
27
+ } from '../../composables/useControlSchema'
28
+ import {
29
+ useBioTemplateComponents,
30
+ } from '../../composables/useBioTemplateComponents'
31
+ import {
32
+ useBioTemplateControls,
33
+ } from '../../composables/useBioTemplateControls'
34
+ import {
35
+ createWellPlateScreenCollection,
36
+ } from '../../templates'
37
+
38
+ describe('useControlSchema', () => {
39
+ const shorthandControls = defineControls({
40
+ threshold: 0.05,
41
+ model: ['linear', 'logistic'],
42
+ enabled: true,
43
+ title: 'Drug screen',
44
+ mode: [
45
+ { value: 'fast', label: 'Fast' },
46
+ { value: 'careful', label: 'Careful' },
47
+ ],
48
+ })
49
+
50
+ const controls = defineControls({
51
+ threshold: {
52
+ type: 'slider',
53
+ label: 'Threshold',
54
+ default: 0.05,
55
+ min: 0,
56
+ max: 1,
57
+ props: { step: 0.01 },
58
+ section: 'parameters',
59
+ sectionLabel: 'Parameters',
60
+ sectionSubtitle: 'Analysis controls',
61
+ view: 'analysis',
62
+ sidebar: {
63
+ label: 'Analysis Parameters',
64
+ icon: 'settings',
65
+ iconColor: '#6366f1',
66
+ iconBg: '#e0e7ff',
67
+ },
68
+ },
69
+ method: {
70
+ label: 'Model',
71
+ default: 'linear',
72
+ options: ['linear', 'logistic'],
73
+ required: true,
74
+ section: 'parameters',
75
+ view: 'analysis',
76
+ },
77
+ saveResults: {
78
+ label: 'Save results',
79
+ default: true,
80
+ section: 'output',
81
+ sectionLabel: 'Output',
82
+ view: 'analysis',
83
+ sidebar: { defaultOpen: false, showToggle: true },
84
+ },
85
+ notes: {
86
+ label: 'Notes',
87
+ placeholder: 'Optional notes',
88
+ section: 'output',
89
+ view: 'analysis',
90
+ sidebar: false,
91
+ },
92
+ diagnostics: {
93
+ label: 'Diagnostics',
94
+ default: false,
95
+ section: 'diagnostics',
96
+ sectionLabel: 'Diagnostics',
97
+ view: 'analysis',
98
+ sidebar: { enabled: false },
99
+ },
100
+ })
101
+
102
+ it('builds defaults from compact controls', () => {
103
+ expect(getControlDefaults(controls)).toEqual({
104
+ threshold: 0.05,
105
+ method: 'linear',
106
+ saveResults: true,
107
+ notes: '',
108
+ diagnostics: false,
109
+ })
110
+ })
111
+
112
+ it('accepts primitive and option-array shorthand controls', () => {
113
+ const schema = controlsToFormSchema(shorthandControls)
114
+
115
+ expect(getControlDefaults(shorthandControls)).toEqual({
116
+ threshold: 0.05,
117
+ model: 'linear',
118
+ enabled: true,
119
+ title: 'Drug screen',
120
+ mode: 'fast',
121
+ })
122
+ expect(schema.sections).toHaveLength(1)
123
+ expect(schema.sections[0].fields.map(field => [field.name, field.type, field.defaultValue])).toEqual([
124
+ ['threshold', 'number', 0.05],
125
+ ['model', 'select', 'linear'],
126
+ ['enabled', 'toggle', true],
127
+ ['title', 'text', 'Drug screen'],
128
+ ['mode', 'select', 'fast'],
129
+ ])
130
+ expect(schema.sections[0].fields[1].props?.options).toEqual([
131
+ { value: 'linear', label: 'Linear' },
132
+ { value: 'logistic', label: 'Logistic' },
133
+ ])
134
+ expect(schema.sections[0].fields[4].props?.options).toEqual([
135
+ { value: 'fast', label: 'Fast' },
136
+ { value: 'careful', label: 'Careful' },
137
+ ])
138
+ })
139
+
140
+ it('converts controls to a FormBuilder schema', () => {
141
+ const schema = controlsToFormSchema(controls, {
142
+ submitLabel: 'Run analysis',
143
+ sections: {
144
+ parameters: { columns: 2 },
145
+ },
146
+ })
147
+
148
+ expect(schema.submitLabel).toBe('Run analysis')
149
+ expect(schema.sections.map(section => section.id)).toEqual(['parameters', 'output', 'diagnostics'])
150
+ expect(schema.sections[0].title).toBe('Parameters')
151
+ expect(schema.sections[0].description).toBeUndefined()
152
+ expect(schema.sections[0].columns).toBe(2)
153
+ expect(schema.sections[0].fields.map(field => field.name)).toEqual(['threshold', 'method'])
154
+
155
+ const threshold = schema.sections[0].fields[0]
156
+ expect(threshold.type).toBe('slider')
157
+ expect(threshold.defaultValue).toBe(0.05)
158
+ expect(threshold.props).toMatchObject({ min: 0, max: 1, step: 0.01 })
159
+
160
+ const method = schema.sections[0].fields[1]
161
+ expect(method.type).toBe('select')
162
+ expect(method.validation?.required).toBe(true)
163
+ expect(method.props?.options).toEqual([
164
+ { label: 'Linear', value: 'linear' },
165
+ { label: 'Logistic', value: 'logistic' },
166
+ ])
167
+ })
168
+
169
+ it('converts controls to AppSidebar panels', () => {
170
+ const panels = controlsToSidebarPanels(controls)
171
+
172
+ expect(Object.keys(panels)).toEqual(['analysis'])
173
+ expect(panels.analysis).toEqual([
174
+ {
175
+ id: 'parameters',
176
+ label: 'Analysis Parameters',
177
+ subtitle: 'Analysis controls',
178
+ icon: 'settings',
179
+ iconColor: '#6366f1',
180
+ iconBg: '#e0e7ff',
181
+ defaultOpen: undefined,
182
+ showToggle: undefined,
183
+ },
184
+ {
185
+ id: 'output',
186
+ label: 'Output',
187
+ subtitle: undefined,
188
+ icon: undefined,
189
+ iconColor: undefined,
190
+ iconBg: undefined,
191
+ defaultOpen: false,
192
+ showToggle: true,
193
+ },
194
+ ])
195
+ })
196
+
197
+ it('returns view helpers that can drive AppPillNav and AppSidebar together', () => {
198
+ expect(controlsToViewIds(controls)).toEqual(['analysis'])
199
+ expect(controlsToViewItems(controls)).toEqual([{ id: 'analysis', label: 'Analysis' }])
200
+ expect(getDefaultControlView(controls)).toBe('analysis')
201
+ })
202
+
203
+ it('uses shared view metadata for AppTopBar pill navigation', () => {
204
+ const runIcon = 'M8 5v14l11-7z'
205
+ const options = {
206
+ views: {
207
+ analysis: {
208
+ label: 'Run',
209
+ icon: runIcon,
210
+ disabled: false,
211
+ },
212
+ },
213
+ }
214
+
215
+ expect(controlsToViewItems(controls, options)).toEqual([
216
+ {
217
+ id: 'analysis',
218
+ label: 'Run',
219
+ icon: runIcon,
220
+ disabled: false,
221
+ },
222
+ ])
223
+ })
224
+
225
+ it('flattens nested control models into workspace bindings', () => {
226
+ const model = defineControlModel({
227
+ initialValues: { replicates: 4 },
228
+ topBarSettings: { title: 'Design settings', showAppearance: false },
229
+ views: {
230
+ design: {
231
+ label: 'Design',
232
+ icon: 'flask',
233
+ sections: {
234
+ dose: {
235
+ label: 'Dose design',
236
+ description: 'Dose and layout controls',
237
+ icon: 'settings',
238
+ columns: 2,
239
+ defaultOpen: false,
240
+ controls: {
241
+ concentrationUnit: ['uM', 'nM'],
242
+ replicates: { type: 'number', default: 3, min: 1 },
243
+ },
244
+ },
245
+ display: {
246
+ label: 'Display',
247
+ sidebar: false,
248
+ controls: {
249
+ showHeatmap: true,
250
+ },
251
+ },
252
+ },
253
+ },
254
+ results: {
255
+ label: 'Results',
256
+ section: 'plots',
257
+ sectionLabel: 'Plots',
258
+ controls: {
259
+ chartScale: ['linear', 'log'],
260
+ },
261
+ },
262
+ },
263
+ componentProps: {
264
+ mode: 'concentrationUnit',
265
+ targetWells: values => values.showHeatmap ? ['A1', 'A2'] : [],
266
+ disabled: values => !values.showHeatmap,
267
+ },
268
+ componentPropsById: {
269
+ dose: {
270
+ mode: 'concentrationUnit',
271
+ targetWells: values => values.showHeatmap ? ['A1', 'A2'] : [],
272
+ disabled: values => !values.showHeatmap,
273
+ },
274
+ plate: {
275
+ selectedWells: values => values.showHeatmap ? ['A1', 'A2'] : [],
276
+ disabled: values => !values.showHeatmap,
277
+ },
278
+ },
279
+ components: defineControlComponentBindings({
280
+ dose: {
281
+ component: 'DoseCalculator',
282
+ props: {
283
+ mode: 'concentrationUnit',
284
+ targetWells: values => values.showHeatmap ? ['A1', 'A2'] : [],
285
+ disabled: values => !values.showHeatmap,
286
+ },
287
+ },
288
+ plate: {
289
+ component: 'WellPlate',
290
+ props: {
291
+ modelValue: values => values.showHeatmap ? ['A1', 'A2'] : [],
292
+ disabled: values => !values.showHeatmap,
293
+ },
294
+ },
295
+ }),
296
+ })
297
+
298
+ expect(model.controls.concentrationUnit).toMatchObject({
299
+ default: 'uM',
300
+ section: 'design-dose',
301
+ view: 'design',
302
+ })
303
+ expect(model.controls.showHeatmap).toMatchObject({
304
+ default: true,
305
+ section: 'design-display',
306
+ view: 'design',
307
+ sidebar: false,
308
+ })
309
+ expect(model.controlOptions.sections?.['design-dose']).toMatchObject({
310
+ id: 'design-dose',
311
+ label: 'Dose design',
312
+ description: 'Dose and layout controls',
313
+ icon: 'settings',
314
+ columns: 2,
315
+ defaultOpen: false,
316
+ })
317
+ expect(model.controlOptions.views?.design).toMatchObject({
318
+ label: 'Design',
319
+ icon: 'flask',
320
+ })
321
+ expect(model.componentProps).toBeDefined()
322
+ expect(model.componentPropsById).toBeDefined()
323
+ expect(model.componentBindings).toBeDefined()
324
+
325
+ const workspace = useControlWorkspace(model.controls, model.controlOptions)
326
+
327
+ expect(workspace.values.replicates).toBe(4)
328
+ expect(workspace.topBar.value.pillNav.map(item => [item.id, item.label])).toEqual([
329
+ ['design', 'Design'],
330
+ ['results', 'Results'],
331
+ ])
332
+ expect(workspace.sidebarPanels.design.map(panel => panel.id)).toEqual(['design-dose'])
333
+ expect(workspace.sidebarPanels.results.map(panel => panel.id)).toEqual(['plots'])
334
+ expect(workspace.formSchema.sections.map(section => section.id)).toEqual([
335
+ 'design-dose',
336
+ 'design-display',
337
+ 'plots',
338
+ ])
339
+ expect(workspace.topBarSettings.settingsConfig.title).toBe('Design settings')
340
+ expect(workspace.topBarSettings.settingsConfig.showAppearance).toBe(false)
341
+ expect(workspace.getComponentProps(model.componentProps)).toEqual({
342
+ mode: 'uM',
343
+ targetWells: ['A1', 'A2'],
344
+ disabled: false,
345
+ })
346
+ expect(workspace.getComponentPropsById(model.componentPropsById)).toEqual({
347
+ dose: {
348
+ mode: 'uM',
349
+ targetWells: ['A1', 'A2'],
350
+ disabled: false,
351
+ },
352
+ plate: {
353
+ selectedWells: ['A1', 'A2'],
354
+ disabled: false,
355
+ },
356
+ })
357
+ expect(workspace.getComponentBindings(model.componentBindings)).toEqual([
358
+ {
359
+ id: 'dose',
360
+ component: 'DoseCalculator',
361
+ props: {
362
+ mode: 'uM',
363
+ targetWells: ['A1', 'A2'],
364
+ disabled: false,
365
+ },
366
+ },
367
+ {
368
+ id: 'plate',
369
+ component: 'WellPlate',
370
+ props: {
371
+ modelValue: ['A1', 'A2'],
372
+ disabled: false,
373
+ },
374
+ },
375
+ ])
376
+ expect(workspace.getComponentBindingsById(model.componentBindings).plate.props.modelValue).toEqual(['A1', 'A2'])
377
+ })
378
+
379
+ it('maps control values into named SDK component bindings', () => {
380
+ const values = {
381
+ threshold: 0.05,
382
+ selectedWells: ['A1'],
383
+ enabled: true,
384
+ }
385
+ const bindings = defineControlComponentBindings([
386
+ {
387
+ component: 'ResultChart',
388
+ props: {
389
+ score: 'threshold',
390
+ disabled: values => !values.enabled,
391
+ },
392
+ },
393
+ {
394
+ component: 'ResultChart',
395
+ props: ['selectedWells'],
396
+ },
397
+ {
398
+ id: 'plate',
399
+ component: 'WellPlate',
400
+ props: {
401
+ modelValue: 'selectedWells',
402
+ },
403
+ },
404
+ ])
405
+
406
+ expect(controlValuesToComponentBindings(values, bindings)).toEqual([
407
+ {
408
+ id: 'ResultChart',
409
+ component: 'ResultChart',
410
+ props: {
411
+ score: 0.05,
412
+ disabled: false,
413
+ },
414
+ },
415
+ {
416
+ id: 'ResultChart-2',
417
+ component: 'ResultChart',
418
+ props: {
419
+ selectedWells: ['A1'],
420
+ },
421
+ },
422
+ {
423
+ id: 'plate',
424
+ component: 'WellPlate',
425
+ props: {
426
+ modelValue: ['A1'],
427
+ },
428
+ },
429
+ ])
430
+ expect(controlValuesToComponentBindingsById(values, bindings).plate.component).toBe('WellPlate')
431
+ })
432
+
433
+ it('keeps direct view controls in separate generated sections', () => {
434
+ const model = defineControlModel({
435
+ views: {
436
+ run: {
437
+ label: 'Run',
438
+ controls: { threshold: 0.05 },
439
+ },
440
+ results: {
441
+ label: 'Results',
442
+ controls: { chartScale: ['linear', 'log'] },
443
+ },
444
+ },
445
+ })
446
+ const workspace = useControlWorkspace(model.controls, model.controlOptions)
447
+
448
+ expect(model.controls.threshold).toMatchObject({ section: 'run-controls', view: 'run' })
449
+ expect(model.controls.chartScale).toMatchObject({ section: 'results-controls', view: 'results' })
450
+ expect(workspace.sectionSchemas['run-controls'].sections[0].fields.map(field => field.name)).toEqual([
451
+ 'threshold',
452
+ ])
453
+ expect(workspace.sectionSchemas['results-controls'].sections[0].fields.map(field => field.name)).toEqual([
454
+ 'chartScale',
455
+ ])
456
+ expect(workspace.sidebarPanels.run.map(panel => panel.id)).toEqual(['run-controls'])
457
+ expect(workspace.sidebarPanels.results.map(panel => panel.id)).toEqual(['results-controls'])
458
+ })
459
+
460
+ it('rejects duplicate control names in nested control models', () => {
461
+ expect(() =>
462
+ defineControlModel({
463
+ sections: {
464
+ first: { controls: { threshold: 0.05 } },
465
+ second: { controls: { threshold: 0.1 } },
466
+ },
467
+ }),
468
+ ).toThrow('Duplicate control "threshold"')
469
+ })
470
+
471
+ it('converts controls to a SettingsModal schema', () => {
472
+ const condition = { field: 'method', eq: 'linear' } as const
473
+ const schema = controlsToSettingsSchema(controls, {
474
+ sections: {
475
+ parameters: {
476
+ label: 'Analysis Settings',
477
+ description: 'Controls used by the analysis run',
478
+ icon: '<svg viewBox="0 0 24 24"></svg>',
479
+ columns: 2,
480
+ condition,
481
+ },
482
+ },
483
+ })
484
+
485
+ expect(schema.groups.map(group => group.id)).toEqual(['parameters', 'output', 'diagnostics'])
486
+ expect(schema.groups[0]).toMatchObject({
487
+ id: 'parameters',
488
+ label: 'Analysis Settings',
489
+ description: 'Controls used by the analysis run',
490
+ icon: '<svg viewBox="0 0 24 24"></svg>',
491
+ columns: 2,
492
+ condition,
493
+ })
494
+ expect(schema.groups[0].fields.map(field => field.name)).toEqual(['threshold', 'method'])
495
+ expect(schema.groups[1].label).toBe('Output')
496
+ expect(schema.groups[1].description).toBeUndefined()
497
+ })
498
+
499
+ it('converts controls to an AppTopBar settings config', () => {
500
+ const options = {
501
+ sections: {
502
+ parameters: { label: 'Analysis Settings' },
503
+ },
504
+ }
505
+ const config = controlsToTopBarSettingsConfig(controls, options, {
506
+ title: 'Analysis controls',
507
+ showAppearance: false,
508
+ layout: 'vertical',
509
+ })
510
+
511
+ expect(config.title).toBe('Analysis controls')
512
+ expect(config.showAppearance).toBe(false)
513
+ expect(config.layout).toBe('vertical')
514
+ expect(config.controls).toBe(controls)
515
+ expect(config.controlOptions).toBe(options)
516
+ expect(config.schema).toBeUndefined()
517
+ })
518
+
519
+ it('returns section-only schemas for sidebar slots', () => {
520
+ const schema = controlsToSectionFormSchema(controls, 'output')
521
+
522
+ expect(schema.sections).toHaveLength(1)
523
+ expect(schema.sections[0].id).toBe('output')
524
+ expect(schema.sections[0].title).toBe('')
525
+ expect(schema.sections[0].fields.map(field => field.name)).toEqual(['saveResults', 'notes'])
526
+ })
527
+
528
+ it('returns section-only schemas keyed by section for AppSidebar forms', () => {
529
+ const schemas = controlsToSectionFormSchemas(controls)
530
+
531
+ expect(Object.keys(schemas)).toEqual(['parameters', 'output', 'diagnostics'])
532
+ expect(schemas.parameters.sections[0].title).toBe('')
533
+ expect(schemas.parameters.sections[0].fields.map(field => field.name)).toEqual(['threshold', 'method'])
534
+ expect(schemas.output.sections[0].fields.map(field => field.name)).toEqual(['saveResults', 'notes'])
535
+ })
536
+
537
+ it('packages form schema, sidebar panels, defaults, and section helpers', () => {
538
+ const toolkit = useControlSchema(controls)
539
+
540
+ expect(toolkit.initialValues.saveResults).toBe(true)
541
+ expect(toolkit.formSchema.sections[0].id).toBe('parameters')
542
+ expect(toolkit.form.schema).toBe(toolkit.formSchema)
543
+ expect(toolkit.settingsSchema.groups[0].id).toBe('parameters')
544
+ expect(toolkit.settings.schema).toBe(toolkit.settingsSchema)
545
+ expect(toolkit.topBarSettingsConfig.controls).toBe(toolkit.controls)
546
+ expect(toolkit.topBarSettingsConfig.controlOptions).toEqual({})
547
+ expect(toolkit.topBarSettingsConfig.schema).toBeUndefined()
548
+ expect(toolkit.topBarSettings).toEqual({
549
+ showSettings: true,
550
+ settingsConfig: toolkit.topBarSettingsConfig,
551
+ })
552
+ expect(toolkit.sidebarPanels.analysis).toHaveLength(2)
553
+ expect(toolkit.viewIds).toEqual(['analysis'])
554
+ expect(toolkit.viewItems).toEqual([{ id: 'analysis', label: 'Analysis' }])
555
+ expect(toolkit.defaultView).toBe('analysis')
556
+ expect(toolkit.sidebar.panels).toBe(toolkit.sidebarPanels)
557
+ expect(toolkit.sidebar.forms).toBe(toolkit.sectionSchemas)
558
+ expect(toolkit.sidebar.viewIds).toBe(toolkit.viewIds)
559
+ expect(toolkit.sidebar.viewItems).toBe(toolkit.viewItems)
560
+ expect(toolkit.sidebar.defaultView).toBe('analysis')
561
+ expect(toolkit.fields.map(field => field.name)).toEqual([
562
+ 'threshold',
563
+ 'method',
564
+ 'saveResults',
565
+ 'notes',
566
+ 'diagnostics',
567
+ ])
568
+ expect(toolkit.sectionSchema('parameters').sections[0].fields).toHaveLength(2)
569
+ expect(toolkit.sectionSchemas.parameters.sections[0].fields).toHaveLength(2)
570
+ })
571
+
572
+ it('packages shared values and direct component bindings for control workspaces', () => {
573
+ const workspace = useControlWorkspace(shorthandControls, {
574
+ initialValues: { threshold: 0.2 },
575
+ topBarSettings: { title: 'Controls', showAppearance: false },
576
+ })
577
+
578
+ expect(workspace.schema.controls).toBe(workspace.controls)
579
+ expect(workspace.values).toMatchObject({
580
+ threshold: 0.2,
581
+ model: 'linear',
582
+ enabled: true,
583
+ title: 'Drug screen',
584
+ mode: 'fast',
585
+ })
586
+ expect(workspace.form.modelValue).toBe(workspace.values)
587
+ expect(workspace.sidebar.modelValue).toBe(workspace.values)
588
+ expect(workspace.sidebar.values).toBe(workspace.values)
589
+ expect(workspace.sidebar.activeView).toBe('default')
590
+ expect(workspace.topBar.value.pillNav).toEqual([{ id: 'default', label: 'Default' }])
591
+ expect(workspace.topBar.value.currentPillId).toBe('default')
592
+ expect(workspace.pillNav.items).toEqual([{ id: 'default', label: 'Default' }])
593
+ expect(workspace.pillNav.currentItemId).toBe('default')
594
+ expect(workspace.topBarSettings.settingsConfig.values).toBe(workspace.values)
595
+ expect(workspace.topBarSettings.settingsConfig.title).toBe('Controls')
596
+ expect(workspace.topBarSettings.settingsConfig.showAppearance).toBe(false)
597
+ expect(workspace.topBarSettings.settingsConfig.controls).toBe(workspace.controls)
598
+ expect(workspace.topBarSettings.settingsConfig.controlOptions).toEqual({})
599
+ expect(workspace.topBarSettings.settingsConfig.schema).toBeUndefined()
600
+ expect(workspace.bindings.form).toBe(workspace.form)
601
+ expect(workspace.bindings.sidebar).toBe(workspace.sidebar)
602
+ expect(workspace.bindings.topBar).toBe(workspace.topBar)
603
+ expect(workspace.bindings.topBarSettings).toBe(workspace.topBarSettings)
604
+ expect(workspace.bindings.pillNav).toBe(workspace.pillNav)
605
+ expect(workspace.bindings.componentBindings).toBe(workspace.componentBindings)
606
+ expect(workspace.bindings.componentBindingsById).toBe(workspace.componentBindingsById)
607
+ expect(workspace.bindings.componentProps).toBe(workspace.componentProps)
608
+ expect(workspace.bindings.componentPropsById).toBe(workspace.componentPropsById)
609
+ expect(workspace.bindings.topBar.value).toMatchObject({
610
+ pillNav: workspace.pillNav.items,
611
+ currentPillId: 'default',
612
+ showSettings: true,
613
+ settingsConfig: workspace.topBarSettingsConfig,
614
+ })
615
+
616
+ workspace.form['onUpdate:modelValue']({ threshold: 0.5, model: 'logistic' })
617
+ expect(workspace.values.threshold).toBe(0.5)
618
+ expect(workspace.values.model).toBe('logistic')
619
+
620
+ workspace.sidebar['onUpdate:values']({ enabled: false })
621
+ expect(workspace.values.enabled).toBe(false)
622
+
623
+ workspace.sidebar['onUpdate:modelValue']({ enabled: true, title: 'Updated screen' })
624
+ expect(workspace.values.enabled).toBe(true)
625
+ expect(workspace.values.title).toBe('Updated screen')
626
+
627
+ workspace.topBarSettings.onSettingsValuesChange({ mode: 'careful' })
628
+ expect(workspace.values.mode).toBe('careful')
629
+
630
+ workspace.topBar.value.onPillSelect({ id: 'missing', label: 'Missing' })
631
+ expect(workspace.activeView.value).toBe('default')
632
+
633
+ workspace.resetValues()
634
+ expect(workspace.values).toMatchObject({
635
+ threshold: 0.05,
636
+ model: 'linear',
637
+ enabled: true,
638
+ title: 'Drug screen',
639
+ mode: 'fast',
640
+ })
641
+ })
642
+
643
+ it('maps workspace values into direct component props', () => {
644
+ const workspace = useControlWorkspace(defineControls({
645
+ sampleNames: {
646
+ default: ['Control', 'Treatment'],
647
+ type: 'tags',
648
+ },
649
+ unit: 'uM',
650
+ enabled: true,
651
+ }))
652
+
653
+ expect(workspace.getComponentProps(['sampleNames', 'unit'])).toEqual({
654
+ sampleNames: ['Control', 'Treatment'],
655
+ unit: 'uM',
656
+ })
657
+ expect(workspace.getComponentProps({
658
+ samples: 'sampleNames',
659
+ concentrationUnit: 'unit',
660
+ disabled: values => !values.enabled,
661
+ })).toEqual({
662
+ samples: ['Control', 'Treatment'],
663
+ concentrationUnit: 'uM',
664
+ disabled: false,
665
+ })
666
+ expect(workspace.getComponentPropsById({
667
+ plate: {
668
+ modelValue: 'sampleNames',
669
+ disabled: values => !values.enabled,
670
+ },
671
+ dose: {
672
+ unit: 'unit',
673
+ readonly: values => !values.enabled,
674
+ },
675
+ })).toEqual({
676
+ plate: {
677
+ modelValue: ['Control', 'Treatment'],
678
+ disabled: false,
679
+ },
680
+ dose: {
681
+ unit: 'uM',
682
+ readonly: false,
683
+ },
684
+ })
685
+ expect(workspace.getComponentPropsById()).toEqual({})
686
+
687
+ workspace.setValues({ enabled: false, unit: 'nM' })
688
+
689
+ expect(controlValuesToComponentProps(workspace.values, {
690
+ unit: 'unit',
691
+ disabled: values => !values.enabled,
692
+ })).toEqual({
693
+ unit: 'nM',
694
+ disabled: true,
695
+ })
696
+ })
697
+
698
+ it('provides built-in WellPlate and DoseCalculator component prop mappings', () => {
699
+ const model = defineControlModel({
700
+ controls: {
701
+ selectedWells: { type: 'tags', default: ['A1', 'A2'] },
702
+ plateFormat: [96, 384],
703
+ doseMode: ['serial', 'dilution'],
704
+ disabled: false,
705
+ },
706
+ componentPropsById: defineWellPlateDoseControlProps(),
707
+ })
708
+ const workspace = useControlWorkspace(model.controls, model.controlOptions)
709
+ const componentPropsById = workspace.getComponentPropsById(model.componentPropsById)
710
+
711
+ expect(componentPropsById.plate.modelValue).toEqual(['A1', 'A2'])
712
+ expect(componentPropsById.plate.format).toBe(96)
713
+ expect(componentPropsById.plate.wells).toBeUndefined()
714
+ expect(componentPropsById.plate.disabled).toBe(false)
715
+ expect(componentPropsById.plate.readonly).toBeUndefined()
716
+ expect(componentPropsById.dose.mode).toBe('serial')
717
+ expect(componentPropsById.dose.targetWells).toEqual(['A1', 'A2'])
718
+ expect(componentPropsById.dose.disabled).toBe(false)
719
+
720
+ const updatePlateSelection = componentPropsById.plate['onUpdate:modelValue']
721
+ expect(typeof updatePlateSelection).toBe('function')
722
+ if (typeof updatePlateSelection !== 'function') throw new Error('Expected WellPlate update handler')
723
+ updatePlateSelection(['B1', 'B2'])
724
+
725
+ expect(workspace.values.selectedWells).toEqual(['B1', 'B2'])
726
+ const syncedPropsById = workspace.getComponentPropsById(model.componentPropsById)
727
+ expect(syncedPropsById.dose.targetWells).toEqual(['B1', 'B2'])
728
+
729
+ const applyToWells = syncedPropsById.dose.onApplyToWells
730
+ expect(typeof applyToWells).toBe('function')
731
+ if (typeof applyToWells !== 'function') throw new Error('Expected DoseCalculator apply handler')
732
+ applyToWells([
733
+ {
734
+ wellId: 'B1',
735
+ concentration: { value: 100, unit: 'µM' },
736
+ volume: { value: 50, unit: 'µL' },
737
+ },
738
+ {
739
+ wellId: 'B2',
740
+ concentration: { value: 50, unit: 'µM' },
741
+ },
742
+ ])
743
+
744
+ expect(workspace.values.wells).toMatchObject({
745
+ B1: {
746
+ state: 'filled',
747
+ value: 100,
748
+ metadata: {
749
+ concentration: { value: 100, unit: 'µM' },
750
+ volume: { value: 50, unit: 'µL' },
751
+ },
752
+ },
753
+ B2: {
754
+ state: 'filled',
755
+ value: 50,
756
+ metadata: {
757
+ concentration: { value: 50, unit: 'µM' },
758
+ },
759
+ },
760
+ })
761
+ expect(workspace.getComponentPropsById(model.componentPropsById).plate.wells).toBe(workspace.values.wells)
762
+ })
763
+
764
+ it('accepts a complete control model binding directly', () => {
765
+ const model = defineDoseDesignControlModel({
766
+ selectedWells: ['E1'],
767
+ plateFormat: 384,
768
+ })
769
+ const workspace = useControlWorkspace(model, {
770
+ initialValues: {
771
+ disabled: true,
772
+ },
773
+ })
774
+
775
+ expect(workspace.values.selectedWells).toEqual(['E1'])
776
+ expect(workspace.values.plateFormat).toBe(384)
777
+ expect(workspace.values.doseMode).toBe('serial')
778
+ expect(workspace.values.disabled).toBe(true)
779
+ expect(workspace.sidebar.panels.design.map(panel => panel.id)).toEqual(['design-dose'])
780
+ expect(workspace.componentProps.value).toEqual({})
781
+ expect(workspace.componentPropsById.value.plate.modelValue).toEqual(['E1'])
782
+ expect(workspace.componentPropsById.value.plate.format).toBe(384)
783
+ expect(workspace.componentPropsById.value.plate.disabled).toBe(true)
784
+ expect(workspace.componentPropsById.value.dose.mode).toBe('serial')
785
+ expect(workspace.componentPropsById.value.dose.targetWells).toEqual(['E1'])
786
+ expect(workspace.componentPropsById.value.dose.disabled).toBe(true)
787
+ expect(workspace.componentBindings.value.map(binding => [binding.id, binding.component])).toEqual([
788
+ ['plate', 'WellPlate'],
789
+ ['dose', 'DoseCalculator'],
790
+ ])
791
+ expect(workspace.componentBindingsById.value.plate.props.modelValue).toEqual(['E1'])
792
+ expect(workspace.componentBindingsById.value.dose.props.disabled).toBe(true)
793
+
794
+ const updatePlateSelection = workspace.componentPropsById.value.plate['onUpdate:modelValue']
795
+ expect(typeof updatePlateSelection).toBe('function')
796
+ if (typeof updatePlateSelection !== 'function') throw new Error('Expected WellPlate update handler')
797
+ updatePlateSelection(['F1', 'F2'])
798
+
799
+ expect(workspace.values.selectedWells).toEqual(['F1', 'F2'])
800
+ expect(workspace.componentPropsById.value.dose.targetWells).toEqual(['F1', 'F2'])
801
+ })
802
+
803
+ it('allows biology component prop mapping field overrides', () => {
804
+ const workspace = useControlWorkspace(defineControls({
805
+ wells: { type: 'tags', default: ['B1'] },
806
+ format: [384, 96],
807
+ mode: ['dilution', 'serial'],
808
+ locked: true,
809
+ molecularWeight: 300.44,
810
+ }))
811
+ const plateProps = workspace.getComponentProps(defineWellPlateControlProps({
812
+ selectedWells: 'wells',
813
+ format: 'format',
814
+ wells: () => undefined,
815
+ disabled: 'locked',
816
+ showSampleTypeIndicator: () => true,
817
+ }))
818
+
819
+ expect(plateProps.modelValue).toEqual(['B1'])
820
+ expect(plateProps.format).toBe(384)
821
+ expect(plateProps.wells).toBeUndefined()
822
+ expect(plateProps.disabled).toBe(true)
823
+ expect(plateProps.readonly).toBeUndefined()
824
+ expect(plateProps.showSampleTypeIndicator).toBe(true)
825
+ const updatePlateSelection = plateProps['onUpdate:modelValue']
826
+ expect(typeof updatePlateSelection).toBe('function')
827
+ if (typeof updatePlateSelection !== 'function') throw new Error('Expected WellPlate update handler')
828
+ updatePlateSelection(['C1', 'C2'])
829
+ expect(workspace.values.wells).toEqual(['C1', 'C2'])
830
+ const doseProps = workspace.getComponentProps(defineDoseCalculatorControlProps({
831
+ mode: 'mode',
832
+ targetWells: 'wells',
833
+ disabled: 'locked',
834
+ molecularWeight: 'molecularWeight',
835
+ }))
836
+
837
+ expect(doseProps.mode).toBe('dilution')
838
+ expect(doseProps.targetWells).toEqual(['C1', 'C2'])
839
+ expect(doseProps.disabled).toBe(true)
840
+ expect(doseProps.molecularWeight).toBe(300.44)
841
+ expect(doseProps.onApplyToWells).toBeUndefined()
842
+ })
843
+
844
+ it('creates WellPlate and DoseCalculator component bindings for dose controls', () => {
845
+ const workspace = useControlWorkspace(defineControls({
846
+ selectedWells: { type: 'tags', default: ['B1'] },
847
+ plateFormat: 384,
848
+ doseMode: ['serial', 'dilution'],
849
+ disabled: false,
850
+ }))
851
+ const bindings = workspace.getComponentBindings(defineWellPlateDoseComponentBindings())
852
+
853
+ expect(bindings.map(binding => [binding.id, binding.component])).toEqual([
854
+ ['plate', 'WellPlate'],
855
+ ['dose', 'DoseCalculator'],
856
+ ])
857
+ expect(bindings[0].props.modelValue).toEqual(['B1'])
858
+ expect(bindings[0].props.format).toBe(384)
859
+ expect(bindings[1].props.mode).toBe('serial')
860
+ expect(bindings[1].props.targetWells).toEqual(['B1'])
861
+ })
862
+
863
+ it('creates a complete dose design control model from one helper', () => {
864
+ const model = defineDoseDesignControlModel({
865
+ selectedWells: ['C1', 'C2'],
866
+ plateFormat: 384,
867
+ includeMolecularWeight: true,
868
+ molecularWeight: 300.44,
869
+ })
870
+ const workspace = useControlWorkspace(model.controls, model.controlOptions)
871
+
872
+ expect(model.controlOptions.views?.design).toMatchObject({ label: 'Design' })
873
+ expect(model.controlOptions.sections?.['design-dose']).toMatchObject({
874
+ label: 'Dose design',
875
+ description: 'Well selection and dose calculator controls',
876
+ })
877
+ expect(workspace.values.selectedWells).toEqual(['C1', 'C2'])
878
+ expect(workspace.values.plateFormat).toBe(384)
879
+ expect(workspace.values.doseMode).toBe('serial')
880
+ expect(workspace.values.disabled).toBe(false)
881
+ expect(workspace.values.molecularWeight).toBe(300.44)
882
+ expect(workspace.sidebar.panels.design.map(panel => panel.id)).toEqual(['design-dose'])
883
+ const componentPropsById = workspace.getComponentPropsById(model.componentPropsById)
884
+ expect(componentPropsById.plate.modelValue).toEqual(['C1', 'C2'])
885
+ expect(componentPropsById.plate.format).toBe(384)
886
+ expect(componentPropsById.plate.wells).toBeUndefined()
887
+ expect(componentPropsById.plate.disabled).toBe(false)
888
+ expect(componentPropsById.plate.readonly).toBeUndefined()
889
+ expect(componentPropsById.dose.mode).toBe('serial')
890
+ expect(componentPropsById.dose.targetWells).toEqual(['C1', 'C2'])
891
+ expect(componentPropsById.dose.disabled).toBe(false)
892
+ expect(componentPropsById.dose.molecularWeight).toBe(300.44)
893
+ expect(workspace.getComponentBindings(model.componentBindings).map(binding => [binding.id, binding.component])).toEqual([
894
+ ['plate', 'WellPlate'],
895
+ ['dose', 'DoseCalculator'],
896
+ ])
897
+
898
+ const updatePlateSelection = componentPropsById.plate['onUpdate:modelValue']
899
+ expect(typeof updatePlateSelection).toBe('function')
900
+ if (typeof updatePlateSelection !== 'function') throw new Error('Expected WellPlate update handler')
901
+ updatePlateSelection(['D1'])
902
+
903
+ expect(workspace.values.selectedWells).toEqual(['D1'])
904
+ expect(workspace.getComponentPropsById(model.componentPropsById).dose.targetWells).toEqual(['D1'])
905
+ })
906
+
907
+ it('customizes the generated dose design control model ids and component mappings', () => {
908
+ const model = defineDoseDesignControlModel({
909
+ viewId: 'setup',
910
+ viewLabel: 'Setup',
911
+ sectionId: 'plate',
912
+ sectionLabel: 'Plate setup',
913
+ componentProps: {
914
+ plateId: 'wellPlate',
915
+ doseId: 'doseCalculator',
916
+ plate: { showLegend: () => true },
917
+ },
918
+ })
919
+ const workspace = useControlWorkspace(model.controls, model.controlOptions)
920
+
921
+ expect(model.controlOptions.views?.setup).toMatchObject({ label: 'Setup' })
922
+ expect(model.controlOptions.sections?.['setup-plate']).toMatchObject({ label: 'Plate setup' })
923
+ const componentPropsById = workspace.getComponentPropsById(model.componentPropsById)
924
+ expect(componentPropsById.wellPlate.modelValue).toEqual(['A1', 'A2'])
925
+ expect(componentPropsById.wellPlate.showLegend).toBe(true)
926
+ expect(componentPropsById.doseCalculator.mode).toBe('serial')
927
+ expect(componentPropsById.doseCalculator.targetWells).toEqual(['A1', 'A2'])
928
+ expect(workspace.getComponentBindings(model.componentBindings).map(binding => [binding.id, binding.component])).toEqual([
929
+ ['wellPlate', 'WellPlate'],
930
+ ['doseCalculator', 'DoseCalculator'],
931
+ ])
932
+ })
933
+
934
+ it('keeps AppTopBar pill navigation and AppSidebar active view on the same workspace state', async () => {
935
+ const workspace = useControlWorkspace({
936
+ threshold: {
937
+ default: 0.05,
938
+ section: 'parameters',
939
+ view: 'analysis',
940
+ },
941
+ chartScale: {
942
+ default: 'linear',
943
+ section: 'display',
944
+ view: 'results',
945
+ },
946
+ }, {
947
+ views: {
948
+ analysis: { label: 'Run' },
949
+ results: { label: 'Results' },
950
+ },
951
+ })
952
+
953
+ expect(workspace.activeView.value).toBe('analysis')
954
+ expect(workspace.sidebar.activeView).toBe('analysis')
955
+ expect(workspace.topBar.value.currentPillId).toBe('analysis')
956
+ expect(workspace.pillNav.currentItemId).toBe('analysis')
957
+ expect(workspace.bindings.topBar.value.currentPillId).toBe('analysis')
958
+ expect(workspace.topBar.value.pillNav.map(item => item.label)).toEqual(['Run', 'Results'])
959
+ expect(workspace.pillNav.items.map(item => item.label)).toEqual(['Run', 'Results'])
960
+
961
+ workspace.topBar.value.onPillSelect({ id: 'results', label: 'Results' })
962
+ expect(workspace.activeView.value).toBe('results')
963
+ expect(workspace.sidebar.activeView).toBe('results')
964
+ expect(workspace.topBar.value.currentPillId).toBe('results')
965
+ expect(workspace.pillNav.currentItemId).toBe('results')
966
+ expect(workspace.bindings.topBar.value.currentPillId).toBe('results')
967
+
968
+ workspace.setActiveView('analysis')
969
+ expect(workspace.activeView.value).toBe('analysis')
970
+ expect(workspace.sidebar.activeView).toBe('analysis')
971
+ expect(workspace.topBar.value.currentPillId).toBe('analysis')
972
+ expect(workspace.pillNav.currentItemId).toBe('analysis')
973
+ workspace.bindings.topBar.value.onPillSelect({ id: 'results', label: 'Results' })
974
+ expect(workspace.activeView.value).toBe('results')
975
+ expect(workspace.bindings.topBar.value.currentPillId).toBe('results')
976
+
977
+ workspace.pillNav.onSelect({ id: 'analysis', label: 'Run' })
978
+ expect(workspace.activeView.value).toBe('analysis')
979
+ expect(workspace.bindings.topBar.value.currentPillId).toBe('analysis')
980
+
981
+ workspace.pillNav.onSelect({ id: 'results', label: 'Results' })
982
+ expect(workspace.activeView.value).toBe('results')
983
+ expect(workspace.sidebar.activeView).toBe('results')
984
+ expect(workspace.topBar.value.currentPillId).toBe('results')
985
+ expect(workspace.pillNav.currentItemId).toBe('results')
986
+
987
+ workspace.activeView.value = 'analysis'
988
+ await nextTick()
989
+
990
+ expect(workspace.sidebar.activeView).toBe('analysis')
991
+ expect(workspace.topBar.value.currentPillId).toBe('analysis')
992
+ expect(workspace.pillNav.currentItemId).toBe('analysis')
993
+
994
+ workspace.pillNav.currentItemId = 'results'
995
+ await nextTick()
996
+
997
+ expect(workspace.activeView.value).toBe('results')
998
+ expect(workspace.sidebar.activeView).toBe('results')
999
+ expect(workspace.topBar.value.currentPillId).toBe('results')
1000
+ })
1001
+
1002
+ it('builds controls and component bindings for biology templates', () => {
1003
+ const controlsToolkit = useBioTemplateControls('wellplate-screen')
1004
+ const collection = createWellPlateScreenCollection({
1005
+ samples: ['Control', 'Treatment'],
1006
+ compounds: { 'Drug A': [10, 1] },
1007
+ })
1008
+ const componentsToolkit = useBioTemplateComponents(collection)
1009
+
1010
+ expect(controlsToolkit.initialValues.plateFormat).toBe(96)
1011
+ expect(controlsToolkit.sidebarPanels.design.map(panel => panel.id)).toContain('dose')
1012
+ expect(componentsToolkit.bindings.map(binding => binding.component)).toContain('WellPlate')
1013
+ expect(componentsToolkit.imports[0].statement).toContain('DoseCalculator')
1014
+ expect(componentsToolkit.componentBindingsById['dose-response:DoseCalculator'].component).toBe('DoseCalculator')
1015
+ expect(componentsToolkit.componentBindingsById['dose-response:DoseCalculator'].props.mode).toBe('serial')
1016
+ expect(componentsToolkit.componentProps.map(binding => binding.component)).toContain('DoseCalculator')
1017
+ expect(componentsToolkit.componentPropsById['dose-response:DoseCalculator'].mode).toBe('serial')
1018
+ expect(componentsToolkit.componentPropsByComponent.WellPlate).toHaveLength(2)
1019
+ expect(componentsToolkit.getComponentProps('DoseCalculator')?.mode).toBe('serial')
1020
+ expect(componentsToolkit.snippets.map(snippet => snippet.component)).toContain('DoseCalculator')
1021
+ expect(componentsToolkit.usage?.template).toContain('<DoseCalculator')
1022
+
1023
+ const presetOnlyToolkit = useBioTemplateComponents('wellplate-screen')
1024
+ expect(presetOnlyToolkit.imports[0].statement).toContain('PlateMapEditor')
1025
+ expect(presetOnlyToolkit.componentProps).toEqual([])
1026
+ expect(presetOnlyToolkit.componentBindingsById).toEqual({})
1027
+ expect(presetOnlyToolkit.componentPropsById).toEqual({})
1028
+ expect(presetOnlyToolkit.componentPropsByComponent).toEqual({})
1029
+ expect(presetOnlyToolkit.getComponentProps('WellPlate')).toBeUndefined()
1030
+ expect(presetOnlyToolkit.snippets).toEqual([])
1031
+ expect(presetOnlyToolkit.usage).toBeNull()
1032
+ })
1033
+ })