@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,392 @@
1
+ <script setup lang="ts">
2
+ /** Complete editable WellPlate/DoseCalculator workspace for dose design that rebuilds generated AppSidebar preset wiring when preset props load late. */
3
+ import { computed, effectScope, onScopeDispose, shallowRef, toRaw, unref, watch, type EffectScope } from 'vue'
4
+ import type { BioTemplateControlValues, TemplatePresetId } from '../templates'
5
+ import type {
6
+ UseBioTemplatePresetWorkspaceOptions,
7
+ UseBioTemplatePresetWorkspaceReturn,
8
+ } from '../composables/useBioTemplatePresetWorkspace'
9
+ import { useBioTemplatePresetWorkspace } from '../composables/useBioTemplatePresetWorkspace'
10
+ import AlertBox from './AlertBox.vue'
11
+ import AppPillNav from './AppPillNav.vue'
12
+ import AppSidebar from './AppSidebar.vue'
13
+ import BaseButton from './BaseButton.vue'
14
+ import BioTemplateRenderer from './BioTemplateRenderer.vue'
15
+
16
+ interface BioTemplatePresetWorkspaceSlotProps {
17
+ workspace: UseBioTemplatePresetWorkspaceReturn
18
+ bindings: UseBioTemplatePresetWorkspaceReturn['bindings']
19
+ collection: UseBioTemplatePresetWorkspaceReturn['collection']['value']
20
+ renderer: UseBioTemplatePresetWorkspaceReturn['renderer']['value']
21
+ componentProps: UseBioTemplatePresetWorkspaceReturn['componentProps']['value']
22
+ componentPropsById: UseBioTemplatePresetWorkspaceReturn['componentPropsById']['value']
23
+ componentPropsByComponent: UseBioTemplatePresetWorkspaceReturn['componentPropsByComponent']['value']
24
+ getComponentProps: UseBioTemplatePresetWorkspaceReturn['getComponentProps']
25
+ }
26
+
27
+ interface Props {
28
+ /** Workspace returned by useBioTemplatePresetWorkspace(). Use for full manual control. */
29
+ workspace?: UseBioTemplatePresetWorkspaceReturn
30
+ /** Built-in preset id used to create the workspace when workspace is not provided. */
31
+ preset?: TemplatePresetId
32
+ /** Options passed to the internally generated useBioTemplatePresetWorkspace() call. */
33
+ workspaceOptions?: UseBioTemplatePresetWorkspaceOptions
34
+ /** Initial control values for the internally generated preset workspace. */
35
+ initialValues?: BioTemplateControlValues
36
+ /** External control values for the internally generated preset workspace. Supports default v-model. */
37
+ modelValue?: BioTemplateControlValues
38
+ /** External control values for the internally generated preset workspace. Supports v-model:values. */
39
+ values?: BioTemplateControlValues
40
+ /** Label shown in the status banner. Defaults to the humanized preset id. */
41
+ label?: string
42
+ /** Sidebar CSS width. */
43
+ sidebarWidth?: string
44
+ /** Compact sidebar and preview layout. */
45
+ dense?: boolean
46
+ /** Render preview components in read-only mode. */
47
+ readonly?: boolean
48
+ /** Show save timestamp. */
49
+ showStatus?: boolean
50
+ /** Show cards for templates included in the preset collection. */
51
+ showTemplateSummary?: boolean
52
+ /** Show component binding count. */
53
+ showComponentSummary?: boolean
54
+ saveLabel?: string
55
+ resetLabel?: string
56
+ }
57
+
58
+ const emit = defineEmits<{
59
+ /** Emitted when generated preset controls change through the embedded sidebar or reset action. */
60
+ 'update:modelValue': [values: BioTemplateControlValues]
61
+ /** Emitted when generated preset controls change through the embedded sidebar or reset action. */
62
+ 'update:values': [values: BioTemplateControlValues]
63
+ }>()
64
+
65
+ defineSlots<{
66
+ default?: (props: BioTemplatePresetWorkspaceSlotProps) => unknown
67
+ }>()
68
+
69
+ const props = withDefaults(defineProps<Props>(), {
70
+ workspace: undefined,
71
+ preset: 'wellplate-screen',
72
+ workspaceOptions: () => ({}),
73
+ initialValues: undefined,
74
+ modelValue: undefined,
75
+ values: undefined,
76
+ sidebarWidth: '320px',
77
+ dense: true,
78
+ readonly: true,
79
+ showStatus: true,
80
+ showTemplateSummary: true,
81
+ showComponentSummary: false,
82
+ saveLabel: 'Save',
83
+ resetLabel: 'Reset defaults',
84
+ })
85
+
86
+ const externalValues = computed(() => props.modelValue ?? props.values)
87
+ const resolvedWorkspaceOptions = computed<UseBioTemplatePresetWorkspaceOptions>(() => {
88
+ if (props.initialValues === undefined && externalValues.value === undefined) return props.workspaceOptions
89
+
90
+ return {
91
+ ...props.workspaceOptions,
92
+ initialValues: {
93
+ ...(props.workspaceOptions.initialValues ?? {}),
94
+ ...(props.initialValues ?? {}),
95
+ ...(externalValues.value ?? {}),
96
+ },
97
+ }
98
+ })
99
+
100
+ let generatedWorkspaceScope: EffectScope | undefined
101
+
102
+ function createGeneratedWorkspace(): UseBioTemplatePresetWorkspaceReturn {
103
+ generatedWorkspaceScope?.stop()
104
+ generatedWorkspaceScope = effectScope()
105
+ return generatedWorkspaceScope.run(() => useBioTemplatePresetWorkspace(props.preset, resolvedWorkspaceOptions.value))!
106
+ }
107
+
108
+ const generatedWorkspace = shallowRef<UseBioTemplatePresetWorkspaceReturn>(createGeneratedWorkspace())
109
+ const resolvedWorkspace = computed(() => props.workspace ?? generatedWorkspace.value)
110
+
111
+ watch(
112
+ [() => props.preset, () => props.workspaceOptions, () => props.initialValues],
113
+ () => {
114
+ generatedWorkspace.value = createGeneratedWorkspace()
115
+ },
116
+ { deep: true },
117
+ )
118
+
119
+ onScopeDispose(() => generatedWorkspaceScope?.stop())
120
+
121
+ watch(
122
+ externalValues,
123
+ (values) => {
124
+ if (values === undefined || props.workspace !== undefined) return
125
+ if (recordsEqual(toPlainRecord(generatedWorkspace.value.controlValues.value), values)) return
126
+ generatedWorkspace.value.setControlValues(values)
127
+ },
128
+ { deep: true },
129
+ )
130
+
131
+ watch(
132
+ () => generatedWorkspace.value.controlValues.value,
133
+ (values) => {
134
+ if (props.workspace !== undefined) return
135
+ const nextValues = toPlainRecord(values)
136
+ if (externalValues.value !== undefined && recordsEqual(nextValues, externalValues.value)) return
137
+ emit('update:modelValue', nextValues)
138
+ emit('update:values', nextValues)
139
+ },
140
+ { deep: true },
141
+ )
142
+
143
+ const presetId = computed(() => resolvedWorkspace.value.presetId)
144
+ const label = computed(() => props.label ?? unref(resolvedWorkspace.value.presetLabel))
145
+ const collection = computed(() => unref(resolvedWorkspace.value.collection))
146
+ const templateIds = computed(() => unref(resolvedWorkspace.value.templateIds))
147
+ const currentExperimentId = computed(() => unref(resolvedWorkspace.value.currentExperimentId))
148
+ const hasCurrentExperiment = computed(() => unref(resolvedWorkspace.value.hasCurrentExperiment))
149
+ const isSaving = computed(() => unref(resolvedWorkspace.value.saving))
150
+ const error = computed(() => unref(resolvedWorkspace.value.error))
151
+ const lastSavedAt = computed(() => unref(resolvedWorkspace.value.lastSavedAt))
152
+ const renderer = computed(() => unref(resolvedWorkspace.value.renderer))
153
+ const componentProps = computed(() => unref(resolvedWorkspace.value.componentProps))
154
+ const componentPropsById = computed(() => unref(resolvedWorkspace.value.componentPropsById))
155
+ const componentPropsByComponent = computed(() => unref(resolvedWorkspace.value.componentPropsByComponent))
156
+ const componentPropsCount = computed(() => unref(resolvedWorkspace.value.componentProps).length)
157
+ const bindings = computed(() => resolvedWorkspace.value.bindings)
158
+
159
+ function toPlainRecord(value: Record<string, unknown>): Record<string, unknown> {
160
+ return toPlain(value) as Record<string, unknown>
161
+ }
162
+
163
+ function toPlain(value: unknown): unknown {
164
+ const raw = toRaw(value)
165
+ if (Array.isArray(raw)) return raw.map(item => toPlain(item))
166
+ if (raw && typeof raw === 'object') {
167
+ return Object.fromEntries(
168
+ Object.entries(raw as Record<string, unknown>).map(([key, item]) => [key, toPlain(item)])
169
+ )
170
+ }
171
+ return raw
172
+ }
173
+
174
+ function recordsEqual(left: Record<string, unknown>, right: Record<string, unknown>): boolean {
175
+ const leftKeys = Object.keys(left)
176
+ const rightKeys = Object.keys(right)
177
+ if (leftKeys.length !== rightKeys.length) return false
178
+ return leftKeys.every(key => valuesEqual(left[key], right[key]))
179
+ }
180
+
181
+ function valuesEqual(left: unknown, right: unknown): boolean {
182
+ const leftRaw = toRaw(left)
183
+ const rightRaw = toRaw(right)
184
+
185
+ if (Object.is(leftRaw, rightRaw)) return true
186
+
187
+ if (Array.isArray(leftRaw) || Array.isArray(rightRaw)) {
188
+ if (!Array.isArray(leftRaw) || !Array.isArray(rightRaw)) return false
189
+ if (leftRaw.length !== rightRaw.length) return false
190
+ return leftRaw.every((item, index) => valuesEqual(item, rightRaw[index]))
191
+ }
192
+
193
+ if (isPlainRecord(leftRaw) || isPlainRecord(rightRaw)) {
194
+ if (!isPlainRecord(leftRaw) || !isPlainRecord(rightRaw)) return false
195
+ return recordsEqual(leftRaw, rightRaw)
196
+ }
197
+
198
+ return false
199
+ }
200
+
201
+ function isPlainRecord(value: unknown): value is Record<string, unknown> {
202
+ return typeof value === 'object' && value !== null && !Array.isArray(value)
203
+ }
204
+ </script>
205
+
206
+ <template>
207
+ <div class="mint-bio-template-preset-workspace">
208
+ <div class="mint-bio-template-preset-workspace__toolbar">
209
+ <AlertBox type="info" class="mint-bio-template-preset-workspace__status">
210
+ {{ label }} preset saves to the current experiment design_data.
211
+ </AlertBox>
212
+
213
+ <div class="mint-bio-template-preset-workspace__actions">
214
+ <BaseButton
215
+ variant="secondary"
216
+ :disabled="isSaving"
217
+ @click="resolvedWorkspace.resetToDefaultCollection()"
218
+ >
219
+ {{ resetLabel }}
220
+ </BaseButton>
221
+ <BaseButton
222
+ variant="primary"
223
+ :loading="isSaving"
224
+ :disabled="!hasCurrentExperiment"
225
+ @click="resolvedWorkspace.saveCurrent()"
226
+ >
227
+ {{ saveLabel }}
228
+ </BaseButton>
229
+ </div>
230
+ </div>
231
+
232
+ <AlertBox v-if="error" type="error">
233
+ {{ error }}
234
+ </AlertBox>
235
+
236
+ <p v-if="showStatus" class="mint-bio-template-preset-workspace__meta">
237
+ saved: {{ lastSavedAt?.toLocaleString() ?? 'not saved' }}
238
+ </p>
239
+
240
+ <div class="mint-bio-template-preset-workspace__main">
241
+ <AppSidebar
242
+ v-bind="resolvedWorkspace.sidebar"
243
+ :floating="false"
244
+ :width="sidebarWidth"
245
+ :dense="dense"
246
+ >
247
+ <template #header>
248
+ <p class="mint-bio-template-preset-workspace__sidebar-title">
249
+ Controls
250
+ </p>
251
+ <AppPillNav
252
+ v-if="resolvedWorkspace.pillNav.items.length > 1"
253
+ class="mint-bio-template-preset-workspace__view-nav"
254
+ :items="resolvedWorkspace.pillNav.items"
255
+ :current-item-id="resolvedWorkspace.pillNav.currentItemId"
256
+ @select="resolvedWorkspace.pillNav.onSelect"
257
+ />
258
+ <p class="mint-bio-template-preset-workspace__sidebar-subtitle">
259
+ {{ presetId }}
260
+ </p>
261
+ </template>
262
+ </AppSidebar>
263
+
264
+ <slot
265
+ :workspace="resolvedWorkspace"
266
+ :bindings="bindings"
267
+ :collection="collection"
268
+ :renderer="renderer"
269
+ :component-props="componentProps"
270
+ :component-props-by-id="componentPropsById"
271
+ :component-props-by-component="componentPropsByComponent"
272
+ :get-component-props="resolvedWorkspace.getComponentProps"
273
+ >
274
+ <BioTemplateRenderer
275
+ v-bind="renderer"
276
+ :dense="dense"
277
+ :readonly="readonly"
278
+ />
279
+ </slot>
280
+ </div>
281
+
282
+ <div
283
+ v-if="showTemplateSummary"
284
+ class="mint-bio-template-preset-workspace__templates"
285
+ >
286
+ <div
287
+ v-for="templateId in templateIds"
288
+ :key="templateId"
289
+ class="mint-bio-template-preset-workspace__template"
290
+ >
291
+ <p class="mint-bio-template-preset-workspace__template-id">
292
+ {{ templateId }}
293
+ </p>
294
+ <p class="mint-bio-template-preset-workspace__template-meta">
295
+ {{ presetId }} - Experiment {{ currentExperimentId ?? 'not selected' }}
296
+ </p>
297
+ </div>
298
+ </div>
299
+
300
+ <p v-if="showComponentSummary" class="mint-bio-template-preset-workspace__meta">
301
+ component prop bindings: {{ componentPropsCount }}
302
+ </p>
303
+ </div>
304
+ </template>
305
+
306
+ <style scoped>
307
+ .mint-bio-template-preset-workspace {
308
+ display: flex;
309
+ flex-direction: column;
310
+ gap: 1rem;
311
+ min-width: 0;
312
+ }
313
+
314
+ .mint-bio-template-preset-workspace__toolbar {
315
+ display: flex;
316
+ align-items: center;
317
+ justify-content: space-between;
318
+ gap: 0.75rem;
319
+ }
320
+
321
+ .mint-bio-template-preset-workspace__status {
322
+ flex: 1 1 auto;
323
+ min-width: 0;
324
+ }
325
+
326
+ .mint-bio-template-preset-workspace__actions {
327
+ display: flex;
328
+ flex: 0 0 auto;
329
+ gap: 0.5rem;
330
+ }
331
+
332
+ .mint-bio-template-preset-workspace__meta,
333
+ .mint-bio-template-preset-workspace__sidebar-subtitle,
334
+ .mint-bio-template-preset-workspace__template-meta {
335
+ margin: 0;
336
+ color: var(--text-secondary, #64748b);
337
+ font-family: var(--font-mono, 'Fira Code', monospace);
338
+ font-size: 0.75rem;
339
+ }
340
+
341
+ .mint-bio-template-preset-workspace__main {
342
+ display: grid;
343
+ grid-template-columns: minmax(16rem, max-content) minmax(0, 1fr);
344
+ gap: 1rem;
345
+ align-items: start;
346
+ min-width: 0;
347
+ }
348
+
349
+ .mint-bio-template-preset-workspace__sidebar-title {
350
+ margin: 0;
351
+ color: var(--text-primary, #0f172a);
352
+ font-size: 0.875rem;
353
+ font-weight: 600;
354
+ }
355
+
356
+ .mint-bio-template-preset-workspace__view-nav {
357
+ margin-top: 0.5rem;
358
+ }
359
+
360
+ .mint-bio-template-preset-workspace__templates {
361
+ display: grid;
362
+ grid-template-columns: repeat(auto-fit, minmax(min(100%, 12rem), 1fr));
363
+ gap: 0.75rem;
364
+ }
365
+
366
+ .mint-bio-template-preset-workspace__template {
367
+ min-width: 0;
368
+ padding: 1rem;
369
+ border: 1px solid var(--border-color, #e5e7eb);
370
+ border-radius: 0.5rem;
371
+ background: var(--bg-primary, #ffffff);
372
+ }
373
+
374
+ .mint-bio-template-preset-workspace__template-id {
375
+ margin: 0;
376
+ color: var(--text-primary, #0f172a);
377
+ font-family: var(--font-mono, 'Fira Code', monospace);
378
+ font-size: 0.875rem;
379
+ }
380
+
381
+ @media (max-width: 48rem) {
382
+ .mint-bio-template-preset-workspace__toolbar,
383
+ .mint-bio-template-preset-workspace__actions {
384
+ align-items: stretch;
385
+ flex-direction: column;
386
+ }
387
+
388
+ .mint-bio-template-preset-workspace__main {
389
+ grid-template-columns: minmax(0, 1fr);
390
+ }
391
+ }
392
+ </style>
@@ -0,0 +1,57 @@
1
+ <script setup lang="ts">
2
+ import BioTemplateRenderer from './BioTemplateRenderer.vue'
3
+ import {
4
+ createLcmsBatchCollection,
5
+ createQpcrExpressionCollection,
6
+ createWellPlateScreenCollection,
7
+ } from '../templates'
8
+
9
+ const wellplateScreen = createWellPlateScreenCollection({
10
+ samples: ['Control', 'Treatment'],
11
+ compounds: {
12
+ 'Drug A': [10, 1, 0.1],
13
+ },
14
+ unit: 'uM',
15
+ })
16
+
17
+ const qpcrExpression = createQpcrExpressionCollection({
18
+ samples: ['Control', 'Treatment'],
19
+ targets: ['ACTB', 'GAPDH'],
20
+ replicates: 2,
21
+ })
22
+
23
+ const lcmsBatch = createLcmsBatchCollection({
24
+ samples: ['S001', 'S002', 'S003'],
25
+ features: ['Glucose', 'Lactate', 'Citrate'],
26
+ })
27
+ </script>
28
+
29
+ <template>
30
+ <Story title="Templates/BioTemplateRenderer">
31
+ <Variant title="Wellplate Screen">
32
+ <div style="padding: 2rem; background: var(--bg-primary, #f8fafc);">
33
+ <BioTemplateRenderer :target="wellplateScreen" />
34
+ </div>
35
+ </Variant>
36
+
37
+ <Variant title="qPCR Expression">
38
+ <div style="padding: 2rem; background: var(--bg-primary, #f8fafc);">
39
+ <BioTemplateRenderer
40
+ :target="qpcrExpression"
41
+ :include="['WellPlate', 'DataFrame']"
42
+ dense
43
+ />
44
+ </div>
45
+ </Variant>
46
+
47
+ <Variant title="LCMS Batch Tables">
48
+ <div style="padding: 2rem; background: var(--bg-primary, #f8fafc);">
49
+ <BioTemplateRenderer
50
+ :target="lcmsBatch"
51
+ :include="['DataFrame']"
52
+ dense
53
+ />
54
+ </div>
55
+ </Variant>
56
+ </Story>
57
+ </template>