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

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (421) hide show
  1. package/README.md +218 -6
  2. package/dist/__tests__/components/ActionItem.test.d.ts +1 -0
  3. package/dist/__tests__/components/AppAvatarMenu.test.d.ts +1 -0
  4. package/dist/__tests__/components/AppPageSelector.test.d.ts +1 -0
  5. package/dist/__tests__/components/AppPillNav.test.d.ts +1 -0
  6. package/dist/__tests__/components/AppPluginSwitcher.test.d.ts +1 -0
  7. package/dist/__tests__/components/AppToastContainer.test.d.ts +1 -0
  8. package/dist/__tests__/components/BaseRadioGroup.test.d.ts +1 -0
  9. package/dist/__tests__/components/BaseSelect.test.d.ts +1 -0
  10. package/dist/__tests__/components/BaseTabs.test.d.ts +1 -0
  11. package/dist/__tests__/components/BatchProgressList.test.d.ts +1 -0
  12. package/dist/__tests__/components/BioTemplateExperimentWorkspaceView.test.d.ts +1 -0
  13. package/dist/__tests__/components/BioTemplatePackWorkspaceView.test.d.ts +1 -0
  14. package/dist/__tests__/components/BioTemplatePresetWorkspaceView.test.d.ts +1 -0
  15. package/dist/__tests__/components/BioTemplateRenderer.test.d.ts +1 -0
  16. package/dist/__tests__/components/Breadcrumb.test.d.ts +1 -0
  17. package/dist/__tests__/components/CalendarGridPanel.test.d.ts +1 -0
  18. package/dist/__tests__/components/ConcentrationInput.test.d.ts +1 -0
  19. package/dist/__tests__/components/ControlWorkspaceView.test.d.ts +1 -0
  20. package/dist/__tests__/components/DatePicker.test.d.ts +1 -0
  21. package/dist/__tests__/components/DateTimePicker.test.d.ts +1 -0
  22. package/dist/__tests__/components/EmptyState.test.d.ts +1 -0
  23. package/dist/__tests__/components/ExperimentPopover.test.d.ts +1 -0
  24. package/dist/__tests__/components/FormBuilder.test.d.ts +1 -0
  25. package/dist/__tests__/components/FormCompatibility.test.d.ts +1 -0
  26. package/dist/__tests__/components/GroupAssigner.test.d.ts +1 -0
  27. package/dist/__tests__/components/GroupingModal.test.d.ts +1 -0
  28. package/dist/__tests__/components/MultiSelect.test.d.ts +1 -0
  29. package/dist/__tests__/components/PluginIcon.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/SettingsButton.test.d.ts +1 -0
  36. package/dist/__tests__/components/SettingsModal.test.d.ts +1 -0
  37. package/dist/__tests__/components/TagsInput.test.d.ts +1 -0
  38. package/dist/__tests__/components/ThemeToggle.test.d.ts +1 -0
  39. package/dist/__tests__/components/TimePicker.test.d.ts +1 -0
  40. package/dist/__tests__/composables/useBioTemplatePackWorkspace.test.d.ts +1 -0
  41. package/dist/__tests__/composables/useBioTemplatePresetWorkspace.test.d.ts +1 -0
  42. package/dist/__tests__/composables/useBioTemplateWorkspace.test.d.ts +1 -0
  43. package/dist/__tests__/composables/useCalendarGrid.test.d.ts +1 -0
  44. package/dist/__tests__/composables/useControlSchema.test.d.ts +1 -0
  45. package/dist/__tests__/composables/useDebouncedWatch.test.d.ts +1 -0
  46. package/dist/__tests__/composables/useDropdownState.test.d.ts +1 -0
  47. package/dist/__tests__/composables/useEventListener.test.d.ts +1 -0
  48. package/dist/__tests__/composables/useExpansionSet.test.d.ts +1 -0
  49. package/dist/__tests__/composables/useExperimentData.test.d.ts +1 -0
  50. package/dist/__tests__/composables/useExperimentSelector.test.d.ts +1 -0
  51. package/dist/__tests__/composables/useGroupAssignment.test.d.ts +1 -0
  52. package/dist/__tests__/composables/useListSelection.test.d.ts +1 -0
  53. package/dist/__tests__/composables/usePluginClient.test.d.ts +1 -0
  54. package/dist/__tests__/composables/usePluginConfig.test.d.ts +1 -0
  55. package/dist/__tests__/composables/useRequestSyncState.test.d.ts +1 -0
  56. package/dist/__tests__/composables/useSampleGroups.test.d.ts +1 -0
  57. package/dist/__tests__/composables/useSelectionLimit.test.d.ts +1 -0
  58. package/dist/__tests__/composables/useSortedItems.test.d.ts +1 -0
  59. package/dist/__tests__/composables/useTemplateCollection.test.d.ts +1 -0
  60. package/dist/__tests__/composables/useTextSearch.test.d.ts +1 -0
  61. package/dist/__tests__/composables/useTheme.test.d.ts +1 -0
  62. package/dist/__tests__/composables/useTimeUtils.test.d.ts +1 -0
  63. package/dist/__tests__/docs/frontendDocsCatalog.test.d.ts +1 -0
  64. package/dist/__tests__/templates/templates.test.d.ts +1 -0
  65. package/dist/{auth-DsI0rQ7_.js → auth-QQj2kkze.js} +12 -5
  66. package/dist/auth-QQj2kkze.js.map +1 -0
  67. package/dist/components/ActionItem.vue.d.ts +32 -0
  68. package/dist/components/AppAvatarMenu.vue.d.ts +2 -7
  69. package/dist/components/AppPageSelector.vue.d.ts +3 -6
  70. package/dist/components/AppPillNav.vue.d.ts +2 -2
  71. package/dist/components/AppSidebar.vue.d.ts +56 -3
  72. package/dist/components/AppToastContainer.vue.d.ts +2 -0
  73. package/dist/components/AppTopBar.vue.d.ts +43 -10
  74. package/dist/components/BaseButton.vue.d.ts +2 -2
  75. package/dist/components/BaseCheckbox.vue.d.ts +1 -1
  76. package/dist/components/BaseInput.vue.d.ts +2 -2
  77. package/dist/components/BasePill.vue.d.ts +3 -3
  78. package/dist/components/BaseRadioGroup.vue.d.ts +4 -4
  79. package/dist/components/BaseSelect.vue.d.ts +4 -4
  80. package/dist/components/BaseSlider.vue.d.ts +1 -1
  81. package/dist/components/BaseTabs.vue.d.ts +2 -2
  82. package/dist/components/BaseTextarea.vue.d.ts +2 -2
  83. package/dist/components/BaseToggle.vue.d.ts +1 -1
  84. package/dist/components/BioTemplateExperimentWorkspaceView.vue.d.ts +117 -0
  85. package/dist/components/BioTemplatePackWorkspaceView.vue.d.ts +92 -0
  86. package/dist/components/BioTemplatePresetWorkspaceView.vue.d.ts +82 -0
  87. package/dist/components/BioTemplateRenderer.vue.d.ts +29 -0
  88. package/dist/components/Breadcrumb.vue.d.ts +2 -2
  89. package/dist/components/Calendar.vue.d.ts +1 -1
  90. package/dist/components/CalendarGridPanel.vue.d.ts +25 -0
  91. package/dist/components/CollapsibleCard.vue.d.ts +1 -1
  92. package/dist/components/ColorSlider.vue.d.ts +1 -1
  93. package/dist/components/ConcentrationInput.vue.d.ts +2 -2
  94. package/dist/components/ConfirmDialog.vue.d.ts +2 -2
  95. package/dist/components/ControlWorkspaceView.vue.d.ts +130 -0
  96. package/dist/components/DatePicker.vue.d.ts +2 -2
  97. package/dist/components/DateTimePicker.vue.d.ts +3 -3
  98. package/dist/components/DropdownButton.vue.d.ts +4 -4
  99. package/dist/components/EmptyState.vue.d.ts +1 -2
  100. package/dist/components/ExperimentDataViewer.vue.d.ts +1 -1
  101. package/dist/components/ExperimentTimeline.vue.d.ts +2 -2
  102. package/dist/components/FileUploader.vue.d.ts +2 -2
  103. package/dist/components/FitPanel.vue.d.ts +1 -1
  104. package/dist/components/FormActions.vue.d.ts +4 -4
  105. package/dist/components/FormBuilder.vue.d.ts +22 -8
  106. package/dist/components/FormFieldRenderer.vue.d.ts +7 -10
  107. package/dist/components/FormSection.vue.d.ts +11 -24
  108. package/dist/components/FormulaInput.vue.d.ts +2 -2
  109. package/dist/components/IconButton.vue.d.ts +1 -1
  110. package/dist/components/LoadingSpinner.vue.d.ts +1 -1
  111. package/dist/components/MoleculeInput.vue.d.ts +2 -2
  112. package/dist/components/MultiSelect.vue.d.ts +3 -3
  113. package/dist/components/NumberInput.vue.d.ts +2 -2
  114. package/dist/components/PluginIcon.vue.d.ts +11 -0
  115. package/dist/components/ProgressBar.vue.d.ts +2 -2
  116. package/dist/components/ProtocolStepEditor.vue.d.ts +3 -1
  117. package/dist/components/RackEditor.vue.d.ts +2 -2
  118. package/dist/components/ReagentEditor.vue.d.ts +1 -1
  119. package/dist/components/ResourceCard.vue.d.ts +1 -1
  120. package/dist/components/SampleLegend.vue.d.ts +2 -2
  121. package/dist/components/SampleSelector.vue.d.ts +1 -1
  122. package/dist/components/ScheduleCalendar.vue.d.ts +2 -2
  123. package/dist/components/ScientificNumber.vue.d.ts +1 -1
  124. package/dist/components/SegmentedControl.vue.d.ts +3 -3
  125. package/dist/components/SequenceInput.vue.d.ts +3 -3
  126. package/dist/components/SettingsButton.vue.d.ts +2 -2
  127. package/dist/components/SettingsModal.vue.d.ts +32 -4
  128. package/dist/components/StatusIndicator.vue.d.ts +1 -1
  129. package/dist/components/TagsInput.vue.d.ts +3 -2
  130. package/dist/components/TimePicker.vue.d.ts +3 -3
  131. package/dist/components/TimeRangeInput.vue.d.ts +2 -2
  132. package/dist/components/UnitInput.vue.d.ts +2 -2
  133. package/dist/components/WellPlate.vue.d.ts +8 -8
  134. package/dist/components/index.d.ts +12 -1
  135. package/dist/components/index.js +3 -3
  136. package/dist/components/internal/FormFieldRendererInternal.vue.d.ts +31 -0
  137. package/dist/components/internal/FormSectionRenderer.vue.d.ts +43 -0
  138. package/dist/{components-CzbQQPCb.js → components-D_Sr0adg.js} +9629 -8647
  139. package/dist/components-D_Sr0adg.js.map +1 -0
  140. package/dist/composables/index.d.ts +21 -2
  141. package/dist/composables/index.js +4 -3
  142. package/dist/composables/platformContextHelpers.d.ts +14 -0
  143. package/dist/composables/useBioTemplateComponents.d.ts +20 -0
  144. package/dist/composables/useBioTemplateControls.d.ts +6 -0
  145. package/dist/composables/useBioTemplatePackWorkspace.d.ts +45 -0
  146. package/dist/composables/useBioTemplatePresetWorkspace.d.ts +74 -0
  147. package/dist/composables/useBioTemplateWorkspace.d.ts +50 -0
  148. package/dist/composables/useCalendarGrid.d.ts +26 -0
  149. package/dist/composables/useControlSchema.d.ts +321 -0
  150. package/dist/composables/useDebouncedWatch.d.ts +20 -0
  151. package/dist/composables/useDropdownState.d.ts +19 -0
  152. package/dist/composables/useEventListener.d.ts +13 -0
  153. package/dist/composables/useExpansionSet.d.ts +21 -0
  154. package/dist/composables/useExperimentData.d.ts +10 -0
  155. package/dist/composables/useExperimentSave.d.ts +31 -2
  156. package/dist/composables/useExperimentSelector.d.ts +20 -0
  157. package/dist/composables/useForm.d.ts +2 -0
  158. package/dist/composables/useGroupAssignment.d.ts +31 -0
  159. package/dist/composables/useListSelection.d.ts +35 -0
  160. package/dist/composables/usePlatformContext.d.ts +24 -3
  161. package/dist/composables/usePluginApi.d.ts +7 -14
  162. package/dist/composables/usePluginClient.d.ts +109 -0
  163. package/dist/composables/usePluginConfig.d.ts +12 -0
  164. package/dist/composables/useRequestSyncState.d.ts +34 -0
  165. package/dist/composables/useSampleGroups.d.ts +32 -0
  166. package/dist/composables/useSelectionLimit.d.ts +17 -0
  167. package/dist/composables/useSortedItems.d.ts +32 -0
  168. package/dist/composables/useTemplateCollection.d.ts +58 -0
  169. package/dist/composables/useTextSearch.d.ts +18 -0
  170. package/dist/composables/useTimeUtils.d.ts +8 -0
  171. package/dist/{composables-BXklV5ii.js → composables-C3dpXQN5.js} +228 -146
  172. package/dist/composables-C3dpXQN5.js.map +1 -0
  173. package/dist/index.d.ts +12 -3
  174. package/dist/index.js +6 -5
  175. package/dist/install.d.ts +7 -2
  176. package/dist/install.js +2 -2
  177. package/dist/install.js.map +1 -1
  178. package/dist/stores/auth.d.ts +1 -1
  179. package/dist/stores/index.js +1 -1
  180. package/dist/stores/settings.d.ts +4 -1
  181. package/dist/styles.css +5255 -5654
  182. package/dist/templates/adapters.d.ts +43 -0
  183. package/dist/templates/builders.d.ts +63 -0
  184. package/dist/templates/catalog.d.ts +188 -0
  185. package/dist/templates/componentBindings.d.ts +58 -0
  186. package/dist/templates/controlSchemas.d.ts +25 -0
  187. package/dist/templates/index.d.ts +15 -0
  188. package/dist/templates/index.js +2 -0
  189. package/dist/templates/lookup.d.ts +4 -0
  190. package/dist/templates/packs.d.ts +18 -0
  191. package/dist/templates/presets.d.ts +90 -0
  192. package/dist/templates/types.d.ts +531 -0
  193. package/dist/templates-50NPjaxL.js +9333 -0
  194. package/dist/templates-50NPjaxL.js.map +1 -0
  195. package/dist/types/components.d.ts +62 -1
  196. package/dist/types/form-builder.d.ts +6 -8
  197. package/dist/types/index.d.ts +2 -2
  198. package/dist/types/platform.d.ts +8 -1
  199. package/dist/useScheduleDrag-D4oWdh41.js +4371 -0
  200. package/dist/useScheduleDrag-D4oWdh41.js.map +1 -0
  201. package/dist/utils/formModelSync.d.ts +5 -0
  202. package/dist/utils/items.d.ts +8 -0
  203. package/dist/utils/options.d.ts +6 -0
  204. package/dist/utils/pluginIcon.d.ts +9 -0
  205. package/package.json +7 -2
  206. package/src/__tests__/components/ActionItem.test.ts +99 -0
  207. package/src/__tests__/components/AppAvatarMenu.test.ts +27 -0
  208. package/src/__tests__/components/AppPageSelector.test.ts +134 -0
  209. package/src/__tests__/components/AppPillNav.test.ts +78 -0
  210. package/src/__tests__/components/AppPluginSwitcher.test.ts +44 -0
  211. package/src/__tests__/components/AppSidebar.test.ts +370 -0
  212. package/src/__tests__/components/AppToastContainer.test.ts +48 -0
  213. package/src/__tests__/components/AppTopBar.test.ts +414 -13
  214. package/src/__tests__/components/BaseRadioGroup.test.ts +25 -0
  215. package/src/__tests__/components/BaseSelect.test.ts +21 -0
  216. package/src/__tests__/components/BaseTabs.test.ts +25 -0
  217. package/src/__tests__/components/BatchProgressList.test.ts +52 -0
  218. package/src/__tests__/components/BioTemplateExperimentWorkspaceView.test.ts +153 -0
  219. package/src/__tests__/components/BioTemplatePackWorkspaceView.test.ts +161 -0
  220. package/src/__tests__/components/BioTemplatePresetWorkspaceView.test.ts +281 -0
  221. package/src/__tests__/components/BioTemplateRenderer.test.ts +71 -0
  222. package/src/__tests__/components/Breadcrumb.test.ts +23 -0
  223. package/src/__tests__/components/CalendarGridPanel.test.ts +36 -0
  224. package/src/__tests__/components/ConcentrationInput.test.ts +45 -0
  225. package/src/__tests__/components/ControlWorkspaceView.test.ts +1031 -0
  226. package/src/__tests__/components/DataFrame.test.ts +11 -0
  227. package/src/__tests__/components/DatePicker.test.ts +45 -0
  228. package/src/__tests__/components/DateTimePicker.test.ts +48 -0
  229. package/src/__tests__/components/DropdownButton.test.ts +23 -0
  230. package/src/__tests__/components/EmptyState.test.ts +23 -0
  231. package/src/__tests__/components/ExperimentPopover.test.ts +56 -0
  232. package/src/__tests__/components/FormBuilder.test.ts +296 -0
  233. package/src/__tests__/components/FormCompatibility.test.ts +94 -0
  234. package/src/__tests__/components/GroupAssigner.test.ts +30 -0
  235. package/src/__tests__/components/GroupingModal.test.ts +73 -0
  236. package/src/__tests__/components/MultiSelect.test.ts +48 -0
  237. package/src/__tests__/components/PluginIcon.test.ts +119 -0
  238. package/src/__tests__/components/ProtocolStepEditor.test.ts +33 -0
  239. package/src/__tests__/components/ReagentList.test.ts +82 -0
  240. package/src/__tests__/components/SampleHierarchyTree.test.ts +53 -0
  241. package/src/__tests__/components/SampleSelector.test.ts +60 -0
  242. package/src/__tests__/components/SegmentedControl.test.ts +24 -0
  243. package/src/__tests__/components/SettingsButton.test.ts +44 -0
  244. package/src/__tests__/components/SettingsModal.test.ts +296 -0
  245. package/src/__tests__/components/TagsInput.test.ts +75 -0
  246. package/src/__tests__/components/ThemeToggle.test.ts +47 -0
  247. package/src/__tests__/components/TimePicker.test.ts +38 -0
  248. package/src/__tests__/composables/useBioTemplatePackWorkspace.test.ts +122 -0
  249. package/src/__tests__/composables/useBioTemplatePresetWorkspace.test.ts +199 -0
  250. package/src/__tests__/composables/useBioTemplateWorkspace.test.ts +99 -0
  251. package/src/__tests__/composables/useCalendarGrid.test.ts +38 -0
  252. package/src/__tests__/composables/useControlSchema.test.ts +919 -0
  253. package/src/__tests__/composables/useDebouncedWatch.test.ts +93 -0
  254. package/src/__tests__/composables/useDropdownState.test.ts +95 -0
  255. package/src/__tests__/composables/useEventListener.test.ts +116 -0
  256. package/src/__tests__/composables/useExpansionSet.test.ts +62 -0
  257. package/src/__tests__/composables/useExperimentData.test.ts +4 -0
  258. package/src/__tests__/composables/useExperimentSave.test.ts +203 -8
  259. package/src/__tests__/composables/useExperimentSelector.test.ts +164 -0
  260. package/src/__tests__/composables/useForm.test.ts +58 -0
  261. package/src/__tests__/composables/useFormBuilder.test.ts +77 -0
  262. package/src/__tests__/composables/useGroupAssignment.test.ts +73 -0
  263. package/src/__tests__/composables/useListSelection.test.ts +66 -0
  264. package/src/__tests__/composables/usePluginClient.test.ts +444 -0
  265. package/src/__tests__/composables/usePluginConfig.test.ts +5 -0
  266. package/src/__tests__/composables/useRequestSyncState.test.ts +92 -0
  267. package/src/__tests__/composables/useSampleGroups.test.ts +66 -0
  268. package/src/__tests__/composables/useSelectionLimit.test.ts +41 -0
  269. package/src/__tests__/composables/useSortedItems.test.ts +87 -0
  270. package/src/__tests__/composables/useTemplateCollection.test.ts +147 -0
  271. package/src/__tests__/composables/useTextSearch.test.ts +55 -0
  272. package/src/__tests__/composables/useTheme.test.ts +91 -0
  273. package/src/__tests__/composables/useTimeUtils.test.ts +35 -0
  274. package/src/__tests__/docs/frontendDocsCatalog.test.ts +229 -0
  275. package/src/__tests__/fixtures/templates/dose-response.json +81 -0
  276. package/src/__tests__/fixtures/templates/plate-map.json +54 -0
  277. package/src/__tests__/fixtures/templates/qpcr-plate.json +96 -0
  278. package/src/__tests__/fixtures/templates/sample-sheet.json +71 -0
  279. package/src/__tests__/templates/templates.test.ts +1043 -0
  280. package/src/components/ActionItem.vue +82 -0
  281. package/src/components/AppAvatarMenu.vue +15 -69
  282. package/src/components/AppLayout.story.vue +25 -25
  283. package/src/components/AppPageSelector.vue +63 -94
  284. package/src/components/AppPillNav.vue +44 -39
  285. package/src/components/AppPluginSwitcher.vue +41 -145
  286. package/src/components/AppSidebar.story.vue +94 -0
  287. package/src/components/AppSidebar.vue +187 -12
  288. package/src/components/{ToastNotification.story.vue → AppToastContainer.story.vue} +6 -6
  289. package/src/components/AppToastContainer.vue +62 -0
  290. package/src/components/AppTopBar.story.vue +7 -30
  291. package/src/components/AppTopBar.vue +283 -84
  292. package/src/components/BaseModal.vue +3 -5
  293. package/src/components/BaseRadioGroup.vue +7 -3
  294. package/src/components/BaseSelect.vue +11 -7
  295. package/src/components/BaseTabs.vue +6 -4
  296. package/src/components/BatchProgressList.vue +5 -8
  297. package/src/components/BioTemplateExperimentWorkspaceView.story.vue +123 -0
  298. package/src/components/BioTemplateExperimentWorkspaceView.vue +337 -0
  299. package/src/components/BioTemplatePackWorkspaceView.story.vue +107 -0
  300. package/src/components/BioTemplatePackWorkspaceView.vue +176 -0
  301. package/src/components/BioTemplatePresetWorkspaceView.story.vue +151 -0
  302. package/src/components/BioTemplatePresetWorkspaceView.vue +392 -0
  303. package/src/components/BioTemplateRenderer.story.vue +57 -0
  304. package/src/components/BioTemplateRenderer.vue +269 -0
  305. package/src/components/Breadcrumb.vue +14 -8
  306. package/src/components/CalendarGridPanel.vue +120 -0
  307. package/src/components/ConcentrationInput.vue +27 -64
  308. package/src/components/ControlWorkspaceView.story.vue +336 -0
  309. package/src/components/ControlWorkspaceView.vue +347 -0
  310. package/src/components/DataFrame.vue +34 -50
  311. package/src/components/DatePicker.vue +59 -192
  312. package/src/components/DateTimePicker.vue +50 -171
  313. package/src/components/DropdownButton.vue +14 -32
  314. package/src/components/EmptyState.vue +4 -2
  315. package/src/components/ExperimentPopover.vue +5 -22
  316. package/src/components/FormBuilder.vue +124 -27
  317. package/src/components/FormFieldRenderer.vue +15 -38
  318. package/src/components/FormSection.vue +20 -73
  319. package/src/components/GroupAssigner.vue +24 -56
  320. package/src/components/GroupingModal.story.vue +3 -3
  321. package/src/components/GroupingModal.vue +30 -391
  322. package/src/components/MultiSelect.vue +17 -12
  323. package/src/components/PlateMapEditor.vue +3 -8
  324. package/src/components/PluginIcon.story.vue +71 -0
  325. package/src/components/PluginIcon.vue +68 -0
  326. package/src/components/ProtocolStepEditor.vue +13 -22
  327. package/src/components/ReagentList.vue +25 -33
  328. package/src/components/SampleHierarchyTree.vue +12 -23
  329. package/src/components/SampleSelector.vue +42 -122
  330. package/src/components/SegmentedControl.vue +7 -3
  331. package/src/components/SettingsButton.story.vue +1 -1
  332. package/src/components/SettingsButton.vue +15 -27
  333. package/src/components/SettingsModal.story.vue +337 -45
  334. package/src/components/SettingsModal.vue +344 -66
  335. package/src/components/TagsInput.vue +29 -14
  336. package/src/components/ThemeToggle.vue +9 -7
  337. package/src/components/TimePicker.vue +19 -41
  338. package/src/components/ToastNotification.vue +4 -57
  339. package/src/components/Tooltip.vue +7 -12
  340. package/src/components/WellEditPopup.vue +3 -8
  341. package/src/components/WellPlate.vue +4 -10
  342. package/src/components/index.ts +12 -1
  343. package/src/components/internal/FormFieldRendererInternal.vue +50 -0
  344. package/src/components/internal/FormSectionRenderer.vue +78 -0
  345. package/src/composables/index.ts +212 -0
  346. package/src/composables/platformContextHelpers.ts +74 -0
  347. package/src/composables/useBioTemplateComponents.ts +93 -0
  348. package/src/composables/useBioTemplateControls.ts +41 -0
  349. package/src/composables/useBioTemplatePackWorkspace.ts +181 -0
  350. package/src/composables/useBioTemplatePresetWorkspace.ts +337 -0
  351. package/src/composables/useBioTemplateWorkspace.ts +139 -0
  352. package/src/composables/useCalendarGrid.ts +140 -0
  353. package/src/composables/useControlSchema.ts +1274 -0
  354. package/src/composables/useDebouncedWatch.ts +119 -0
  355. package/src/composables/useDropdownState.ts +83 -0
  356. package/src/composables/useEventListener.ts +111 -0
  357. package/src/composables/useExpansionSet.ts +117 -0
  358. package/src/composables/useExperimentData.ts +20 -11
  359. package/src/composables/useExperimentSave.ts +202 -50
  360. package/src/composables/useExperimentSelector.ts +86 -72
  361. package/src/composables/useForm.ts +49 -4
  362. package/src/composables/useFormBuilder.ts +93 -42
  363. package/src/composables/useGroupAssignment.ts +148 -0
  364. package/src/composables/useListSelection.ts +158 -0
  365. package/src/composables/usePluginApi.ts +7 -14
  366. package/src/composables/usePluginClient.ts +425 -0
  367. package/src/composables/usePluginConfig.ts +34 -13
  368. package/src/composables/useRequestSyncState.ts +126 -0
  369. package/src/composables/useSampleGroups.ts +126 -0
  370. package/src/composables/useSelectionLimit.ts +57 -0
  371. package/src/composables/useSortedItems.ts +118 -0
  372. package/src/composables/useTemplateCollection.ts +229 -0
  373. package/src/composables/useTextSearch.ts +60 -0
  374. package/src/composables/useTheme.ts +2 -28
  375. package/src/composables/useTimeUtils.ts +26 -2
  376. package/src/composables/useWellPlateEditor.ts +13 -9
  377. package/src/index.ts +228 -4
  378. package/src/install.ts +11 -4
  379. package/src/stores/settings.ts +13 -9
  380. package/src/styles/components/app-page-selector.css +23 -0
  381. package/src/styles/components/app-pill-nav.css +8 -2
  382. package/src/styles/components/app-top-bar.css +35 -2
  383. package/src/styles/components/button.css +3 -7
  384. package/src/styles/components/concentration-input.css +3 -142
  385. package/src/styles/components/dropdown-button.css +4 -4
  386. package/src/styles/components/empty-state.css +0 -16
  387. package/src/styles/components/input.css +4 -5
  388. package/src/styles/components/number-input.css +3 -3
  389. package/src/styles/components/plugin-icon.css +38 -0
  390. package/src/styles/components/segmented-control.css +4 -7
  391. package/src/styles/components/settings-button.css +3 -66
  392. package/src/styles/components/settings-modal.css +184 -0
  393. package/src/styles/components/tabs.css +1 -2
  394. package/src/styles/components/textarea.css +4 -5
  395. package/src/styles/components/theme-toggle.css +3 -66
  396. package/src/styles/components/unit-input.css +3 -3
  397. package/src/styles/index.css +0 -1
  398. package/src/templates/adapters.ts +785 -0
  399. package/src/templates/builders.ts +2149 -0
  400. package/src/templates/catalog.ts +245 -0
  401. package/src/templates/componentBindings.ts +615 -0
  402. package/src/templates/controlSchemas.ts +718 -0
  403. package/src/templates/index.ts +314 -0
  404. package/src/templates/lookup.ts +18 -0
  405. package/src/templates/packs.ts +156 -0
  406. package/src/templates/presets.ts +146 -0
  407. package/src/templates/types.ts +668 -0
  408. package/src/types/components.ts +80 -1
  409. package/src/types/form-builder.ts +7 -2
  410. package/src/types/index.ts +17 -0
  411. package/src/types/platform.ts +8 -1
  412. package/src/utils/formModelSync.ts +52 -0
  413. package/src/utils/items.ts +28 -0
  414. package/src/utils/options.ts +23 -0
  415. package/src/utils/pluginIcon.ts +30 -0
  416. package/dist/auth-DsI0rQ7_.js.map +0 -1
  417. package/dist/components-CzbQQPCb.js.map +0 -1
  418. package/dist/composables-BXklV5ii.js.map +0 -1
  419. package/dist/useScheduleDrag-CxBeqYcu.js +0 -7181
  420. package/dist/useScheduleDrag-CxBeqYcu.js.map +0 -1
  421. package/src/styles/components/grouping-modal.css +0 -323
@@ -0,0 +1,919 @@
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
+ controlsToTopBarTabs,
11
+ controlsToViewIds,
12
+ controlsToViewItems,
13
+ controlValuesToComponentProps,
14
+ defineControlModel,
15
+ defineDoseDesignControlModel,
16
+ defineDoseCalculatorControlProps,
17
+ defineControls,
18
+ defineWellPlateControlProps,
19
+ defineWellPlateDoseControlProps,
20
+ getDefaultControlView,
21
+ getControlDefaults,
22
+ useControlSchema,
23
+ useControlWorkspace,
24
+ } from '../../composables/useControlSchema'
25
+ import {
26
+ useBioTemplateComponents,
27
+ } from '../../composables/useBioTemplateComponents'
28
+ import {
29
+ useBioTemplateControls,
30
+ } from '../../composables/useBioTemplateControls'
31
+ import {
32
+ createWellPlateScreenCollection,
33
+ } from '../../templates'
34
+
35
+ describe('useControlSchema', () => {
36
+ const shorthandControls = defineControls({
37
+ threshold: 0.05,
38
+ model: ['linear', 'logistic'],
39
+ enabled: true,
40
+ title: 'Drug screen',
41
+ mode: [
42
+ { value: 'fast', label: 'Fast' },
43
+ { value: 'careful', label: 'Careful' },
44
+ ],
45
+ })
46
+
47
+ const controls = defineControls({
48
+ threshold: {
49
+ type: 'slider',
50
+ label: 'Threshold',
51
+ default: 0.05,
52
+ min: 0,
53
+ max: 1,
54
+ props: { step: 0.01 },
55
+ section: 'parameters',
56
+ sectionLabel: 'Parameters',
57
+ sectionSubtitle: 'Analysis controls',
58
+ view: 'analysis',
59
+ sidebar: {
60
+ label: 'Analysis Parameters',
61
+ icon: 'settings',
62
+ iconColor: '#6366f1',
63
+ iconBg: '#e0e7ff',
64
+ },
65
+ },
66
+ method: {
67
+ label: 'Model',
68
+ default: 'linear',
69
+ options: ['linear', 'logistic'],
70
+ required: true,
71
+ section: 'parameters',
72
+ view: 'analysis',
73
+ },
74
+ saveResults: {
75
+ label: 'Save results',
76
+ default: true,
77
+ section: 'output',
78
+ sectionLabel: 'Output',
79
+ view: 'analysis',
80
+ sidebar: { defaultOpen: false, showToggle: true },
81
+ },
82
+ notes: {
83
+ label: 'Notes',
84
+ placeholder: 'Optional notes',
85
+ section: 'output',
86
+ view: 'analysis',
87
+ sidebar: false,
88
+ },
89
+ diagnostics: {
90
+ label: 'Diagnostics',
91
+ default: false,
92
+ section: 'diagnostics',
93
+ sectionLabel: 'Diagnostics',
94
+ view: 'analysis',
95
+ sidebar: { enabled: false },
96
+ },
97
+ })
98
+
99
+ it('builds defaults from compact controls', () => {
100
+ expect(getControlDefaults(controls)).toEqual({
101
+ threshold: 0.05,
102
+ method: 'linear',
103
+ saveResults: true,
104
+ notes: '',
105
+ diagnostics: false,
106
+ })
107
+ })
108
+
109
+ it('accepts primitive and option-array shorthand controls', () => {
110
+ const schema = controlsToFormSchema(shorthandControls)
111
+
112
+ expect(getControlDefaults(shorthandControls)).toEqual({
113
+ threshold: 0.05,
114
+ model: 'linear',
115
+ enabled: true,
116
+ title: 'Drug screen',
117
+ mode: 'fast',
118
+ })
119
+ expect(schema.sections).toHaveLength(1)
120
+ expect(schema.sections[0].fields.map(field => [field.name, field.type, field.defaultValue])).toEqual([
121
+ ['threshold', 'number', 0.05],
122
+ ['model', 'select', 'linear'],
123
+ ['enabled', 'toggle', true],
124
+ ['title', 'text', 'Drug screen'],
125
+ ['mode', 'select', 'fast'],
126
+ ])
127
+ expect(schema.sections[0].fields[1].props?.options).toEqual([
128
+ { value: 'linear', label: 'Linear' },
129
+ { value: 'logistic', label: 'Logistic' },
130
+ ])
131
+ expect(schema.sections[0].fields[4].props?.options).toEqual([
132
+ { value: 'fast', label: 'Fast' },
133
+ { value: 'careful', label: 'Careful' },
134
+ ])
135
+ })
136
+
137
+ it('converts controls to a FormBuilder schema', () => {
138
+ const schema = controlsToFormSchema(controls, {
139
+ submitLabel: 'Run analysis',
140
+ sections: {
141
+ parameters: { columns: 2 },
142
+ },
143
+ })
144
+
145
+ expect(schema.submitLabel).toBe('Run analysis')
146
+ expect(schema.sections.map(section => section.id)).toEqual(['parameters', 'output', 'diagnostics'])
147
+ expect(schema.sections[0].title).toBe('Parameters')
148
+ expect(schema.sections[0].description).toBeUndefined()
149
+ expect(schema.sections[0].columns).toBe(2)
150
+ expect(schema.sections[0].fields.map(field => field.name)).toEqual(['threshold', 'method'])
151
+
152
+ const threshold = schema.sections[0].fields[0]
153
+ expect(threshold.type).toBe('slider')
154
+ expect(threshold.defaultValue).toBe(0.05)
155
+ expect(threshold.props).toMatchObject({ min: 0, max: 1, step: 0.01 })
156
+
157
+ const method = schema.sections[0].fields[1]
158
+ expect(method.type).toBe('select')
159
+ expect(method.validation?.required).toBe(true)
160
+ expect(method.props?.options).toEqual([
161
+ { label: 'Linear', value: 'linear' },
162
+ { label: 'Logistic', value: 'logistic' },
163
+ ])
164
+ })
165
+
166
+ it('converts controls to AppSidebar panels', () => {
167
+ const panels = controlsToSidebarPanels(controls)
168
+
169
+ expect(Object.keys(panels)).toEqual(['analysis'])
170
+ expect(panels.analysis).toEqual([
171
+ {
172
+ id: 'parameters',
173
+ label: 'Analysis Parameters',
174
+ subtitle: 'Analysis controls',
175
+ icon: 'settings',
176
+ iconColor: '#6366f1',
177
+ iconBg: '#e0e7ff',
178
+ defaultOpen: undefined,
179
+ showToggle: undefined,
180
+ },
181
+ {
182
+ id: 'output',
183
+ label: 'Output',
184
+ subtitle: undefined,
185
+ icon: undefined,
186
+ iconColor: undefined,
187
+ iconBg: undefined,
188
+ defaultOpen: false,
189
+ showToggle: true,
190
+ },
191
+ ])
192
+ })
193
+
194
+ it('returns view helpers that can drive AppPillNav and AppSidebar together', () => {
195
+ expect(controlsToViewIds(controls)).toEqual(['analysis'])
196
+ expect(controlsToViewItems(controls)).toEqual([{ id: 'analysis', label: 'Analysis' }])
197
+ expect(controlsToTopBarTabs(controls)).toEqual([{ id: 'analysis', label: 'Analysis' }])
198
+ expect(getDefaultControlView(controls)).toBe('analysis')
199
+ })
200
+
201
+ it('uses shared view metadata for AppTopBar tabs and AppPillNav items', () => {
202
+ const runIcon = 'M8 5v14l11-7z'
203
+ const options = {
204
+ views: {
205
+ analysis: {
206
+ label: 'Run',
207
+ icon: runIcon,
208
+ disabled: false,
209
+ },
210
+ },
211
+ }
212
+
213
+ expect(controlsToViewItems(controls, options)).toEqual([
214
+ {
215
+ id: 'analysis',
216
+ label: 'Run',
217
+ icon: runIcon,
218
+ disabled: false,
219
+ },
220
+ ])
221
+ expect(controlsToTopBarTabs(controls, options)).toEqual([
222
+ {
223
+ id: 'analysis',
224
+ label: 'Run',
225
+ icon: runIcon,
226
+ disabled: false,
227
+ },
228
+ ])
229
+ })
230
+
231
+ it('flattens nested control models into workspace bindings', () => {
232
+ const model = defineControlModel({
233
+ initialValues: { replicates: 4 },
234
+ topBarSettings: { title: 'Design settings', showAppearance: false },
235
+ views: {
236
+ design: {
237
+ label: 'Design',
238
+ icon: 'flask',
239
+ sections: {
240
+ dose: {
241
+ label: 'Dose design',
242
+ description: 'Dose and layout controls',
243
+ icon: 'settings',
244
+ columns: 2,
245
+ defaultOpen: false,
246
+ controls: {
247
+ concentrationUnit: ['uM', 'nM'],
248
+ replicates: { type: 'number', default: 3, min: 1 },
249
+ },
250
+ },
251
+ display: {
252
+ label: 'Display',
253
+ sidebar: false,
254
+ controls: {
255
+ showHeatmap: true,
256
+ },
257
+ },
258
+ },
259
+ },
260
+ results: {
261
+ label: 'Results',
262
+ section: 'plots',
263
+ sectionLabel: 'Plots',
264
+ controls: {
265
+ chartScale: ['linear', 'log'],
266
+ },
267
+ },
268
+ },
269
+ componentProps: {
270
+ mode: 'concentrationUnit',
271
+ targetWells: values => values.showHeatmap ? ['A1', 'A2'] : [],
272
+ disabled: values => !values.showHeatmap,
273
+ },
274
+ componentPropsById: {
275
+ dose: {
276
+ mode: 'concentrationUnit',
277
+ targetWells: values => values.showHeatmap ? ['A1', 'A2'] : [],
278
+ disabled: values => !values.showHeatmap,
279
+ },
280
+ plate: {
281
+ selectedWells: values => values.showHeatmap ? ['A1', 'A2'] : [],
282
+ disabled: values => !values.showHeatmap,
283
+ },
284
+ },
285
+ })
286
+
287
+ expect(model.controls.concentrationUnit).toMatchObject({
288
+ default: 'uM',
289
+ section: 'design-dose',
290
+ view: 'design',
291
+ })
292
+ expect(model.controls.showHeatmap).toMatchObject({
293
+ default: true,
294
+ section: 'design-display',
295
+ view: 'design',
296
+ sidebar: false,
297
+ })
298
+ expect(model.controlOptions.sections?.['design-dose']).toMatchObject({
299
+ id: 'design-dose',
300
+ label: 'Dose design',
301
+ description: 'Dose and layout controls',
302
+ icon: 'settings',
303
+ columns: 2,
304
+ defaultOpen: false,
305
+ })
306
+ expect(model.controlOptions.views?.design).toMatchObject({
307
+ label: 'Design',
308
+ icon: 'flask',
309
+ })
310
+ expect(model.componentProps).toBeDefined()
311
+ expect(model.componentPropsById).toBeDefined()
312
+
313
+ const workspace = useControlWorkspace(model.controls, model.controlOptions)
314
+
315
+ expect(workspace.values.replicates).toBe(4)
316
+ expect(workspace.topBar.tabs.map(tab => [tab.id, tab.label])).toEqual([
317
+ ['design', 'Design'],
318
+ ['results', 'Results'],
319
+ ])
320
+ expect(workspace.sidebarPanels.design.map(panel => panel.id)).toEqual(['design-dose'])
321
+ expect(workspace.sidebarPanels.results.map(panel => panel.id)).toEqual(['plots'])
322
+ expect(workspace.formSchema.sections.map(section => section.id)).toEqual([
323
+ 'design-dose',
324
+ 'design-display',
325
+ 'plots',
326
+ ])
327
+ expect(workspace.topBarSettings.settingsConfig.title).toBe('Design settings')
328
+ expect(workspace.topBarSettings.settingsConfig.showAppearance).toBe(false)
329
+ expect(workspace.getComponentProps(model.componentProps)).toEqual({
330
+ mode: 'uM',
331
+ targetWells: ['A1', 'A2'],
332
+ disabled: false,
333
+ })
334
+ expect(workspace.getComponentPropsById(model.componentPropsById)).toEqual({
335
+ dose: {
336
+ mode: 'uM',
337
+ targetWells: ['A1', 'A2'],
338
+ disabled: false,
339
+ },
340
+ plate: {
341
+ selectedWells: ['A1', 'A2'],
342
+ disabled: false,
343
+ },
344
+ })
345
+ })
346
+
347
+ it('keeps direct view controls in separate generated sections', () => {
348
+ const model = defineControlModel({
349
+ views: {
350
+ run: {
351
+ label: 'Run',
352
+ controls: { threshold: 0.05 },
353
+ },
354
+ results: {
355
+ label: 'Results',
356
+ controls: { chartScale: ['linear', 'log'] },
357
+ },
358
+ },
359
+ })
360
+ const workspace = useControlWorkspace(model.controls, model.controlOptions)
361
+
362
+ expect(model.controls.threshold).toMatchObject({ section: 'run-controls', view: 'run' })
363
+ expect(model.controls.chartScale).toMatchObject({ section: 'results-controls', view: 'results' })
364
+ expect(workspace.sectionSchemas['run-controls'].sections[0].fields.map(field => field.name)).toEqual([
365
+ 'threshold',
366
+ ])
367
+ expect(workspace.sectionSchemas['results-controls'].sections[0].fields.map(field => field.name)).toEqual([
368
+ 'chartScale',
369
+ ])
370
+ expect(workspace.sidebarPanels.run.map(panel => panel.id)).toEqual(['run-controls'])
371
+ expect(workspace.sidebarPanels.results.map(panel => panel.id)).toEqual(['results-controls'])
372
+ })
373
+
374
+ it('rejects duplicate control names in nested control models', () => {
375
+ expect(() =>
376
+ defineControlModel({
377
+ sections: {
378
+ first: { controls: { threshold: 0.05 } },
379
+ second: { controls: { threshold: 0.1 } },
380
+ },
381
+ }),
382
+ ).toThrow('Duplicate control "threshold"')
383
+ })
384
+
385
+ it('converts controls to a SettingsModal schema', () => {
386
+ const condition = { field: 'method', eq: 'linear' } as const
387
+ const schema = controlsToSettingsSchema(controls, {
388
+ sections: {
389
+ parameters: {
390
+ label: 'Analysis Settings',
391
+ description: 'Controls used by the analysis run',
392
+ icon: '<svg viewBox="0 0 24 24"></svg>',
393
+ columns: 2,
394
+ condition,
395
+ },
396
+ },
397
+ })
398
+
399
+ expect(schema.groups.map(group => group.id)).toEqual(['parameters', 'output', 'diagnostics'])
400
+ expect(schema.groups[0]).toMatchObject({
401
+ id: 'parameters',
402
+ label: 'Analysis Settings',
403
+ description: 'Controls used by the analysis run',
404
+ icon: '<svg viewBox="0 0 24 24"></svg>',
405
+ columns: 2,
406
+ condition,
407
+ })
408
+ expect(schema.groups[0].fields.map(field => field.name)).toEqual(['threshold', 'method'])
409
+ expect(schema.groups[1].label).toBe('Output')
410
+ expect(schema.groups[1].description).toBeUndefined()
411
+ })
412
+
413
+ it('converts controls to an AppTopBar settings config', () => {
414
+ const options = {
415
+ sections: {
416
+ parameters: { label: 'Analysis Settings' },
417
+ },
418
+ }
419
+ const config = controlsToTopBarSettingsConfig(controls, options, {
420
+ title: 'Analysis controls',
421
+ showAppearance: false,
422
+ layout: 'vertical',
423
+ })
424
+
425
+ expect(config.title).toBe('Analysis controls')
426
+ expect(config.showAppearance).toBe(false)
427
+ expect(config.layout).toBe('vertical')
428
+ expect(config.controls).toBe(controls)
429
+ expect(config.controlOptions).toBe(options)
430
+ expect(config.schema).toBeUndefined()
431
+ })
432
+
433
+ it('returns section-only schemas for sidebar slots', () => {
434
+ const schema = controlsToSectionFormSchema(controls, 'output')
435
+
436
+ expect(schema.sections).toHaveLength(1)
437
+ expect(schema.sections[0].id).toBe('output')
438
+ expect(schema.sections[0].title).toBe('')
439
+ expect(schema.sections[0].fields.map(field => field.name)).toEqual(['saveResults', 'notes'])
440
+ })
441
+
442
+ it('returns section-only schemas keyed by section for AppSidebar forms', () => {
443
+ const schemas = controlsToSectionFormSchemas(controls)
444
+
445
+ expect(Object.keys(schemas)).toEqual(['parameters', 'output', 'diagnostics'])
446
+ expect(schemas.parameters.sections[0].title).toBe('')
447
+ expect(schemas.parameters.sections[0].fields.map(field => field.name)).toEqual(['threshold', 'method'])
448
+ expect(schemas.output.sections[0].fields.map(field => field.name)).toEqual(['saveResults', 'notes'])
449
+ })
450
+
451
+ it('packages form schema, sidebar panels, defaults, and section helpers', () => {
452
+ const toolkit = useControlSchema(controls)
453
+
454
+ expect(toolkit.initialValues.saveResults).toBe(true)
455
+ expect(toolkit.formSchema.sections[0].id).toBe('parameters')
456
+ expect(toolkit.form.schema).toBe(toolkit.formSchema)
457
+ expect(toolkit.settingsSchema.groups[0].id).toBe('parameters')
458
+ expect(toolkit.settings.schema).toBe(toolkit.settingsSchema)
459
+ expect(toolkit.topBarSettingsConfig.controls).toBe(toolkit.controls)
460
+ expect(toolkit.topBarSettingsConfig.controlOptions).toEqual({})
461
+ expect(toolkit.topBarSettingsConfig.schema).toBeUndefined()
462
+ expect(toolkit.topBarSettings).toEqual({
463
+ showSettings: true,
464
+ settingsConfig: toolkit.topBarSettingsConfig,
465
+ })
466
+ expect(toolkit.sidebarPanels.analysis).toHaveLength(2)
467
+ expect(toolkit.viewIds).toEqual(['analysis'])
468
+ expect(toolkit.viewItems).toEqual([{ id: 'analysis', label: 'Analysis' }])
469
+ expect(toolkit.topBarTabs).toEqual([{ id: 'analysis', label: 'Analysis' }])
470
+ expect(toolkit.defaultView).toBe('analysis')
471
+ expect(toolkit.sidebar.panels).toBe(toolkit.sidebarPanels)
472
+ expect(toolkit.sidebar.forms).toBe(toolkit.sectionSchemas)
473
+ expect(toolkit.sidebar.viewIds).toBe(toolkit.viewIds)
474
+ expect(toolkit.sidebar.viewItems).toBe(toolkit.viewItems)
475
+ expect(toolkit.sidebar.defaultView).toBe('analysis')
476
+ expect(toolkit.fields.map(field => field.name)).toEqual([
477
+ 'threshold',
478
+ 'method',
479
+ 'saveResults',
480
+ 'notes',
481
+ 'diagnostics',
482
+ ])
483
+ expect(toolkit.sectionSchema('parameters').sections[0].fields).toHaveLength(2)
484
+ expect(toolkit.sectionSchemas.parameters.sections[0].fields).toHaveLength(2)
485
+ })
486
+
487
+ it('packages shared values and direct component bindings for control workspaces', () => {
488
+ const workspace = useControlWorkspace(shorthandControls, {
489
+ initialValues: { threshold: 0.2 },
490
+ topBarSettings: { title: 'Controls', showAppearance: false },
491
+ })
492
+
493
+ expect(workspace.schema.controls).toBe(workspace.controls)
494
+ expect(workspace.values).toMatchObject({
495
+ threshold: 0.2,
496
+ model: 'linear',
497
+ enabled: true,
498
+ title: 'Drug screen',
499
+ mode: 'fast',
500
+ })
501
+ expect(workspace.form.modelValue).toBe(workspace.values)
502
+ expect(workspace.sidebar.modelValue).toBe(workspace.values)
503
+ expect(workspace.sidebar.values).toBe(workspace.values)
504
+ expect(workspace.sidebar.activeView).toBe('default')
505
+ expect(workspace.topBar.tabs).toEqual([{ id: 'default', label: 'Default' }])
506
+ expect(workspace.topBar.currentTabId).toBe('default')
507
+ expect(workspace.pillNav.items).toEqual([{ id: 'default', label: 'Default' }])
508
+ expect(workspace.pillNav.currentItemId).toBe('default')
509
+ expect(workspace.topBarSettings.settingsConfig.values).toBe(workspace.values)
510
+ expect(workspace.topBarSettings.settingsConfig.title).toBe('Controls')
511
+ expect(workspace.topBarSettings.settingsConfig.showAppearance).toBe(false)
512
+ expect(workspace.topBarSettings.settingsConfig.controls).toBe(workspace.controls)
513
+ expect(workspace.topBarSettings.settingsConfig.controlOptions).toEqual({})
514
+ expect(workspace.topBarSettings.settingsConfig.schema).toBeUndefined()
515
+ expect(workspace.bindings.form).toBe(workspace.form)
516
+ expect(workspace.bindings.sidebar).toBe(workspace.sidebar)
517
+ expect(workspace.bindings.topBarSettings).toBe(workspace.topBarSettings)
518
+ expect(workspace.bindings.pillNav).toBe(workspace.pillNav)
519
+ expect(workspace.bindings.componentProps).toBe(workspace.componentProps)
520
+ expect(workspace.bindings.componentPropsById).toBe(workspace.componentPropsById)
521
+ expect(workspace.bindings.topBar.value).toMatchObject({
522
+ pillNav: workspace.pillNav.items,
523
+ currentPillId: 'default',
524
+ showSettings: true,
525
+ settingsConfig: workspace.topBarSettingsConfig,
526
+ })
527
+ expect(workspace.bindings.topBarTabs.value).toMatchObject({
528
+ tabs: workspace.topBar.tabs,
529
+ currentTabId: 'default',
530
+ showSettings: true,
531
+ settingsConfig: workspace.topBarSettingsConfig,
532
+ })
533
+
534
+ workspace.form['onUpdate:modelValue']({ threshold: 0.5, model: 'logistic' })
535
+ expect(workspace.values.threshold).toBe(0.5)
536
+ expect(workspace.values.model).toBe('logistic')
537
+
538
+ workspace.sidebar['onUpdate:values']({ enabled: false })
539
+ expect(workspace.values.enabled).toBe(false)
540
+
541
+ workspace.sidebar['onUpdate:modelValue']({ enabled: true, title: 'Updated screen' })
542
+ expect(workspace.values.enabled).toBe(true)
543
+ expect(workspace.values.title).toBe('Updated screen')
544
+
545
+ workspace.topBarSettings.onSettingsValuesChange({ mode: 'careful' })
546
+ expect(workspace.values.mode).toBe('careful')
547
+
548
+ workspace.topBar.onTabSelect({ id: 'missing', label: 'Missing' })
549
+ expect(workspace.activeView.value).toBe('default')
550
+
551
+ workspace.resetValues()
552
+ expect(workspace.values).toMatchObject({
553
+ threshold: 0.05,
554
+ model: 'linear',
555
+ enabled: true,
556
+ title: 'Drug screen',
557
+ mode: 'fast',
558
+ })
559
+ })
560
+
561
+ it('maps workspace values into direct component props', () => {
562
+ const workspace = useControlWorkspace(defineControls({
563
+ sampleNames: {
564
+ default: ['Control', 'Treatment'],
565
+ type: 'tags',
566
+ },
567
+ unit: 'uM',
568
+ enabled: true,
569
+ }))
570
+
571
+ expect(workspace.getComponentProps(['sampleNames', 'unit'])).toEqual({
572
+ sampleNames: ['Control', 'Treatment'],
573
+ unit: 'uM',
574
+ })
575
+ expect(workspace.getComponentProps({
576
+ samples: 'sampleNames',
577
+ concentrationUnit: 'unit',
578
+ disabled: values => !values.enabled,
579
+ })).toEqual({
580
+ samples: ['Control', 'Treatment'],
581
+ concentrationUnit: 'uM',
582
+ disabled: false,
583
+ })
584
+ expect(workspace.getComponentPropsById({
585
+ plate: {
586
+ modelValue: 'sampleNames',
587
+ disabled: values => !values.enabled,
588
+ },
589
+ dose: {
590
+ unit: 'unit',
591
+ readonly: values => !values.enabled,
592
+ },
593
+ })).toEqual({
594
+ plate: {
595
+ modelValue: ['Control', 'Treatment'],
596
+ disabled: false,
597
+ },
598
+ dose: {
599
+ unit: 'uM',
600
+ readonly: false,
601
+ },
602
+ })
603
+ expect(workspace.getComponentPropsById()).toEqual({})
604
+
605
+ workspace.setValues({ enabled: false, unit: 'nM' })
606
+
607
+ expect(controlValuesToComponentProps(workspace.values, {
608
+ unit: 'unit',
609
+ disabled: values => !values.enabled,
610
+ })).toEqual({
611
+ unit: 'nM',
612
+ disabled: true,
613
+ })
614
+ })
615
+
616
+ it('provides built-in WellPlate and DoseCalculator component prop mappings', () => {
617
+ const model = defineControlModel({
618
+ controls: {
619
+ selectedWells: { type: 'tags', default: ['A1', 'A2'] },
620
+ plateFormat: [96, 384],
621
+ doseMode: ['serial', 'dilution'],
622
+ disabled: false,
623
+ },
624
+ componentPropsById: defineWellPlateDoseControlProps(),
625
+ })
626
+ const workspace = useControlWorkspace(model.controls, model.controlOptions)
627
+ const componentPropsById = workspace.getComponentPropsById(model.componentPropsById)
628
+
629
+ expect(componentPropsById.plate.modelValue).toEqual(['A1', 'A2'])
630
+ expect(componentPropsById.plate.format).toBe(96)
631
+ expect(componentPropsById.plate.wells).toBeUndefined()
632
+ expect(componentPropsById.plate.disabled).toBe(false)
633
+ expect(componentPropsById.plate.readonly).toBeUndefined()
634
+ expect(componentPropsById.dose.mode).toBe('serial')
635
+ expect(componentPropsById.dose.targetWells).toEqual(['A1', 'A2'])
636
+ expect(componentPropsById.dose.disabled).toBe(false)
637
+
638
+ const updatePlateSelection = componentPropsById.plate['onUpdate:modelValue']
639
+ expect(typeof updatePlateSelection).toBe('function')
640
+ if (typeof updatePlateSelection !== 'function') throw new Error('Expected WellPlate update handler')
641
+ updatePlateSelection(['B1', 'B2'])
642
+
643
+ expect(workspace.values.selectedWells).toEqual(['B1', 'B2'])
644
+ const syncedPropsById = workspace.getComponentPropsById(model.componentPropsById)
645
+ expect(syncedPropsById.dose.targetWells).toEqual(['B1', 'B2'])
646
+
647
+ const applyToWells = syncedPropsById.dose.onApplyToWells
648
+ expect(typeof applyToWells).toBe('function')
649
+ if (typeof applyToWells !== 'function') throw new Error('Expected DoseCalculator apply handler')
650
+ applyToWells([
651
+ {
652
+ wellId: 'B1',
653
+ concentration: { value: 100, unit: 'µM' },
654
+ volume: { value: 50, unit: 'µL' },
655
+ },
656
+ {
657
+ wellId: 'B2',
658
+ concentration: { value: 50, unit: 'µM' },
659
+ },
660
+ ])
661
+
662
+ expect(workspace.values.wells).toMatchObject({
663
+ B1: {
664
+ state: 'filled',
665
+ value: 100,
666
+ metadata: {
667
+ concentration: { value: 100, unit: 'µM' },
668
+ volume: { value: 50, unit: 'µL' },
669
+ },
670
+ },
671
+ B2: {
672
+ state: 'filled',
673
+ value: 50,
674
+ metadata: {
675
+ concentration: { value: 50, unit: 'µM' },
676
+ },
677
+ },
678
+ })
679
+ expect(workspace.getComponentPropsById(model.componentPropsById).plate.wells).toBe(workspace.values.wells)
680
+ })
681
+
682
+ it('accepts a complete control model binding directly', () => {
683
+ const model = defineDoseDesignControlModel({
684
+ selectedWells: ['E1'],
685
+ plateFormat: 384,
686
+ })
687
+ const workspace = useControlWorkspace(model, {
688
+ initialValues: {
689
+ disabled: true,
690
+ },
691
+ })
692
+
693
+ expect(workspace.values.selectedWells).toEqual(['E1'])
694
+ expect(workspace.values.plateFormat).toBe(384)
695
+ expect(workspace.values.doseMode).toBe('serial')
696
+ expect(workspace.values.disabled).toBe(true)
697
+ expect(workspace.sidebar.panels.design.map(panel => panel.id)).toEqual(['design-dose'])
698
+ expect(workspace.componentProps.value).toEqual({})
699
+ expect(workspace.componentPropsById.value.plate.modelValue).toEqual(['E1'])
700
+ expect(workspace.componentPropsById.value.plate.format).toBe(384)
701
+ expect(workspace.componentPropsById.value.plate.disabled).toBe(true)
702
+ expect(workspace.componentPropsById.value.dose.mode).toBe('serial')
703
+ expect(workspace.componentPropsById.value.dose.targetWells).toEqual(['E1'])
704
+ expect(workspace.componentPropsById.value.dose.disabled).toBe(true)
705
+
706
+ const updatePlateSelection = workspace.componentPropsById.value.plate['onUpdate:modelValue']
707
+ expect(typeof updatePlateSelection).toBe('function')
708
+ if (typeof updatePlateSelection !== 'function') throw new Error('Expected WellPlate update handler')
709
+ updatePlateSelection(['F1', 'F2'])
710
+
711
+ expect(workspace.values.selectedWells).toEqual(['F1', 'F2'])
712
+ expect(workspace.componentPropsById.value.dose.targetWells).toEqual(['F1', 'F2'])
713
+ })
714
+
715
+ it('allows biology component prop mapping field overrides', () => {
716
+ const workspace = useControlWorkspace(defineControls({
717
+ wells: { type: 'tags', default: ['B1'] },
718
+ format: [384, 96],
719
+ mode: ['dilution', 'serial'],
720
+ locked: true,
721
+ molecularWeight: 300.44,
722
+ }))
723
+ const plateProps = workspace.getComponentProps(defineWellPlateControlProps({
724
+ selectedWells: 'wells',
725
+ format: 'format',
726
+ wells: () => undefined,
727
+ disabled: 'locked',
728
+ showSampleTypeIndicator: () => true,
729
+ }))
730
+
731
+ expect(plateProps.modelValue).toEqual(['B1'])
732
+ expect(plateProps.format).toBe(384)
733
+ expect(plateProps.wells).toBeUndefined()
734
+ expect(plateProps.disabled).toBe(true)
735
+ expect(plateProps.readonly).toBeUndefined()
736
+ expect(plateProps.showSampleTypeIndicator).toBe(true)
737
+ const updatePlateSelection = plateProps['onUpdate:modelValue']
738
+ expect(typeof updatePlateSelection).toBe('function')
739
+ if (typeof updatePlateSelection !== 'function') throw new Error('Expected WellPlate update handler')
740
+ updatePlateSelection(['C1', 'C2'])
741
+ expect(workspace.values.wells).toEqual(['C1', 'C2'])
742
+ const doseProps = workspace.getComponentProps(defineDoseCalculatorControlProps({
743
+ mode: 'mode',
744
+ targetWells: 'wells',
745
+ disabled: 'locked',
746
+ molecularWeight: 'molecularWeight',
747
+ }))
748
+
749
+ expect(doseProps.mode).toBe('dilution')
750
+ expect(doseProps.targetWells).toEqual(['C1', 'C2'])
751
+ expect(doseProps.disabled).toBe(true)
752
+ expect(doseProps.molecularWeight).toBe(300.44)
753
+ expect(doseProps.onApplyToWells).toBeUndefined()
754
+ })
755
+
756
+ it('creates a complete dose design control model from one helper', () => {
757
+ const model = defineDoseDesignControlModel({
758
+ selectedWells: ['C1', 'C2'],
759
+ plateFormat: 384,
760
+ includeMolecularWeight: true,
761
+ molecularWeight: 300.44,
762
+ })
763
+ const workspace = useControlWorkspace(model.controls, model.controlOptions)
764
+
765
+ expect(model.controlOptions.views?.design).toMatchObject({ label: 'Design' })
766
+ expect(model.controlOptions.sections?.['design-dose']).toMatchObject({
767
+ label: 'Dose design',
768
+ description: 'Well selection and dose calculator controls',
769
+ })
770
+ expect(workspace.values.selectedWells).toEqual(['C1', 'C2'])
771
+ expect(workspace.values.plateFormat).toBe(384)
772
+ expect(workspace.values.doseMode).toBe('serial')
773
+ expect(workspace.values.disabled).toBe(false)
774
+ expect(workspace.values.molecularWeight).toBe(300.44)
775
+ expect(workspace.sidebar.panels.design.map(panel => panel.id)).toEqual(['design-dose'])
776
+ const componentPropsById = workspace.getComponentPropsById(model.componentPropsById)
777
+ expect(componentPropsById.plate.modelValue).toEqual(['C1', 'C2'])
778
+ expect(componentPropsById.plate.format).toBe(384)
779
+ expect(componentPropsById.plate.wells).toBeUndefined()
780
+ expect(componentPropsById.plate.disabled).toBe(false)
781
+ expect(componentPropsById.plate.readonly).toBeUndefined()
782
+ expect(componentPropsById.dose.mode).toBe('serial')
783
+ expect(componentPropsById.dose.targetWells).toEqual(['C1', 'C2'])
784
+ expect(componentPropsById.dose.disabled).toBe(false)
785
+ expect(componentPropsById.dose.molecularWeight).toBe(300.44)
786
+
787
+ const updatePlateSelection = componentPropsById.plate['onUpdate:modelValue']
788
+ expect(typeof updatePlateSelection).toBe('function')
789
+ if (typeof updatePlateSelection !== 'function') throw new Error('Expected WellPlate update handler')
790
+ updatePlateSelection(['D1'])
791
+
792
+ expect(workspace.values.selectedWells).toEqual(['D1'])
793
+ expect(workspace.getComponentPropsById(model.componentPropsById).dose.targetWells).toEqual(['D1'])
794
+ })
795
+
796
+ it('customizes the generated dose design control model ids and component mappings', () => {
797
+ const model = defineDoseDesignControlModel({
798
+ viewId: 'setup',
799
+ viewLabel: 'Setup',
800
+ sectionId: 'plate',
801
+ sectionLabel: 'Plate setup',
802
+ componentProps: {
803
+ plateId: 'wellPlate',
804
+ doseId: 'doseCalculator',
805
+ plate: { showLegend: () => true },
806
+ },
807
+ })
808
+ const workspace = useControlWorkspace(model.controls, model.controlOptions)
809
+
810
+ expect(model.controlOptions.views?.setup).toMatchObject({ label: 'Setup' })
811
+ expect(model.controlOptions.sections?.['setup-plate']).toMatchObject({ label: 'Plate setup' })
812
+ const componentPropsById = workspace.getComponentPropsById(model.componentPropsById)
813
+ expect(componentPropsById.wellPlate.modelValue).toEqual(['A1', 'A2'])
814
+ expect(componentPropsById.wellPlate.showLegend).toBe(true)
815
+ expect(componentPropsById.doseCalculator.mode).toBe('serial')
816
+ expect(componentPropsById.doseCalculator.targetWells).toEqual(['A1', 'A2'])
817
+ })
818
+
819
+ it('keeps AppTopBar tabs and AppSidebar active view on the same workspace state', async () => {
820
+ const workspace = useControlWorkspace({
821
+ threshold: {
822
+ default: 0.05,
823
+ section: 'parameters',
824
+ view: 'analysis',
825
+ },
826
+ chartScale: {
827
+ default: 'linear',
828
+ section: 'display',
829
+ view: 'results',
830
+ },
831
+ }, {
832
+ views: {
833
+ analysis: { label: 'Run' },
834
+ results: { label: 'Results' },
835
+ },
836
+ })
837
+
838
+ expect(workspace.activeView.value).toBe('analysis')
839
+ expect(workspace.sidebar.activeView).toBe('analysis')
840
+ expect(workspace.topBar.currentTabId).toBe('analysis')
841
+ expect(workspace.pillNav.currentItemId).toBe('analysis')
842
+ expect(workspace.bindings.topBar.value.currentPillId).toBe('analysis')
843
+ expect(workspace.bindings.topBarTabs.value.currentTabId).toBe('analysis')
844
+ expect(workspace.topBar.tabs.map(tab => tab.label)).toEqual(['Run', 'Results'])
845
+ expect(workspace.pillNav.items.map(item => item.label)).toEqual(['Run', 'Results'])
846
+
847
+ workspace.topBar.onTabSelect({ id: 'results', label: 'Results' })
848
+ expect(workspace.activeView.value).toBe('results')
849
+ expect(workspace.sidebar.activeView).toBe('results')
850
+ expect(workspace.topBar.currentTabId).toBe('results')
851
+ expect(workspace.pillNav.currentItemId).toBe('results')
852
+ expect(workspace.bindings.topBar.value.currentPillId).toBe('results')
853
+ expect(workspace.bindings.topBarTabs.value.currentTabId).toBe('results')
854
+
855
+ workspace.setActiveView('analysis')
856
+ expect(workspace.activeView.value).toBe('analysis')
857
+ expect(workspace.sidebar.activeView).toBe('analysis')
858
+ expect(workspace.topBar.currentTabId).toBe('analysis')
859
+ expect(workspace.pillNav.currentItemId).toBe('analysis')
860
+ workspace.bindings.topBar.value.onPillSelect({ id: 'results', label: 'Results' })
861
+ expect(workspace.activeView.value).toBe('results')
862
+ expect(workspace.bindings.topBar.value.currentPillId).toBe('results')
863
+ expect(workspace.bindings.topBarTabs.value.currentTabId).toBe('results')
864
+
865
+ workspace.bindings.topBarTabs.value.onTabSelect({ id: 'analysis', label: 'Run' })
866
+ expect(workspace.activeView.value).toBe('analysis')
867
+ expect(workspace.bindings.topBar.value.currentPillId).toBe('analysis')
868
+ expect(workspace.bindings.topBarTabs.value.currentTabId).toBe('analysis')
869
+
870
+ workspace.pillNav.onSelect({ id: 'results', label: 'Results' })
871
+ expect(workspace.activeView.value).toBe('results')
872
+ expect(workspace.sidebar.activeView).toBe('results')
873
+ expect(workspace.topBar.currentTabId).toBe('results')
874
+ expect(workspace.pillNav.currentItemId).toBe('results')
875
+
876
+ workspace.activeView.value = 'analysis'
877
+ await nextTick()
878
+
879
+ expect(workspace.sidebar.activeView).toBe('analysis')
880
+ expect(workspace.topBar.currentTabId).toBe('analysis')
881
+ expect(workspace.pillNav.currentItemId).toBe('analysis')
882
+
883
+ workspace.pillNav.currentItemId = 'results'
884
+ await nextTick()
885
+
886
+ expect(workspace.activeView.value).toBe('results')
887
+ expect(workspace.sidebar.activeView).toBe('results')
888
+ expect(workspace.topBar.currentTabId).toBe('results')
889
+ })
890
+
891
+ it('builds controls and component bindings for biology templates', () => {
892
+ const controlsToolkit = useBioTemplateControls('wellplate-screen')
893
+ const collection = createWellPlateScreenCollection({
894
+ samples: ['Control', 'Treatment'],
895
+ compounds: { 'Drug A': [10, 1] },
896
+ })
897
+ const componentsToolkit = useBioTemplateComponents(collection)
898
+
899
+ expect(controlsToolkit.initialValues.plateFormat).toBe(96)
900
+ expect(controlsToolkit.sidebarPanels.design.map(panel => panel.id)).toContain('dose')
901
+ expect(componentsToolkit.bindings.map(binding => binding.component)).toContain('WellPlate')
902
+ expect(componentsToolkit.imports[0].statement).toContain('DoseCalculator')
903
+ expect(componentsToolkit.componentProps.map(binding => binding.component)).toContain('DoseCalculator')
904
+ expect(componentsToolkit.componentPropsById['dose-response:DoseCalculator'].mode).toBe('serial')
905
+ expect(componentsToolkit.componentPropsByComponent.WellPlate).toHaveLength(2)
906
+ expect(componentsToolkit.getComponentProps('DoseCalculator')?.mode).toBe('serial')
907
+ expect(componentsToolkit.snippets.map(snippet => snippet.component)).toContain('DoseCalculator')
908
+ expect(componentsToolkit.usage?.template).toContain('<DoseCalculator')
909
+
910
+ const presetOnlyToolkit = useBioTemplateComponents('wellplate-screen')
911
+ expect(presetOnlyToolkit.imports[0].statement).toContain('PlateMapEditor')
912
+ expect(presetOnlyToolkit.componentProps).toEqual([])
913
+ expect(presetOnlyToolkit.componentPropsById).toEqual({})
914
+ expect(presetOnlyToolkit.componentPropsByComponent).toEqual({})
915
+ expect(presetOnlyToolkit.getComponentProps('WellPlate')).toBeUndefined()
916
+ expect(presetOnlyToolkit.snippets).toEqual([])
917
+ expect(presetOnlyToolkit.usage).toBeNull()
918
+ })
919
+ })