@morscherlab/mint-sdk 1.0.0-rc.2 → 1.0.0-rc.5
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.
- package/dist/__tests__/components/AppTopBar.navigation.test.d.ts +1 -0
- package/dist/__tests__/components/DoseCalculatorVolumeField.test.d.ts +1 -0
- package/dist/__tests__/components/PlateMapEditorToolbarInternal.test.d.ts +1 -0
- package/dist/__tests__/components/PluginWorkspaceView.controls.test.d.ts +1 -0
- package/dist/__tests__/components/PluginWorkspaceView.navigation.test.d.ts +1 -0
- package/dist/__tests__/components/PluginWorkspaceView.shell.test.d.ts +1 -0
- package/dist/__tests__/components/ProtocolStep.presentation.test.d.ts +1 -0
- package/dist/__tests__/components/ProtocolStepEditor.state.test.d.ts +1 -0
- package/dist/__tests__/components/ProtocolStepParameterField.test.d.ts +1 -0
- package/dist/__tests__/components/ReagentList.presentation.test.d.ts +1 -0
- package/dist/__tests__/components/SampleSelector.colors.test.d.ts +1 -0
- package/dist/__tests__/components/SampleSelector.drag.test.d.ts +1 -0
- package/dist/__tests__/components/SampleSelector.groups.test.d.ts +1 -0
- package/dist/__tests__/components/SampleSelector.selection.test.d.ts +1 -0
- package/dist/__tests__/components/SampleSelectorSampleRow.test.d.ts +1 -0
- package/dist/__tests__/components/ScheduleCalendar.test.d.ts +1 -0
- package/dist/__tests__/components/SettingsModal.schema.test.d.ts +1 -0
- package/dist/__tests__/components/WellPlate.colors.test.d.ts +1 -0
- package/dist/__tests__/components/WellPlate.conditions.test.d.ts +1 -0
- package/dist/__tests__/components/WellPlate.geometry.test.d.ts +1 -0
- package/dist/__tests__/components/WellPlate.interaction.test.d.ts +1 -0
- package/dist/__tests__/components/WellPlate.legend.test.d.ts +1 -0
- package/dist/__tests__/components/WellPlate.rendering.test.d.ts +1 -0
- package/dist/__tests__/components/WellPlate.sampleDrop.test.d.ts +1 -0
- package/dist/__tests__/composables/autoGroup/classify.test.d.ts +1 -0
- package/dist/__tests__/composables/autoGroup/columns.test.d.ts +1 -0
- package/dist/__tests__/composables/autoGroup/compose.test.d.ts +1 -0
- package/dist/__tests__/composables/autoGroup/cooccurrence.test.d.ts +1 -0
- package/dist/__tests__/composables/autoGroup/fingerprint.test.d.ts +1 -0
- package/dist/__tests__/composables/autoGroup/integration.test.d.ts +1 -0
- package/dist/__tests__/composables/autoGroup/template.test.d.ts +1 -0
- package/dist/__tests__/composables/autoGroup/tokenize.test.d.ts +1 -0
- package/dist/__tests__/composables/useAutoGroupInputSources.test.d.ts +1 -0
- package/dist/__tests__/composables/useScheduleCalendarLayout.test.d.ts +1 -0
- package/dist/__tests__/docs/extractDocsComponents.test.d.ts +1 -0
- package/dist/__tests__/docs/extractDocsExports.test.d.ts +1 -0
- package/dist/__tests__/docs/extractDocsParsing.test.d.ts +1 -0
- package/dist/__tests__/docs/extractDocsTemplates.test.d.ts +1 -0
- package/dist/__tests__/docs/extractDocsTheme.test.d.ts +1 -0
- package/dist/components/AppTopBar.navigation.d.ts +11 -0
- package/dist/components/BaseButton.vue.d.ts +1 -1
- package/dist/components/BaseCheckbox.vue.d.ts +1 -1
- package/dist/components/BaseInput.vue.d.ts +2 -2
- package/dist/components/BasePill.vue.d.ts +1 -1
- package/dist/components/BaseRadioGroup.vue.d.ts +1 -1
- package/dist/components/BaseSelect.vue.d.ts +1 -1
- package/dist/components/BaseSlider.vue.d.ts +2 -2
- package/dist/components/BaseTextarea.vue.d.ts +2 -2
- package/dist/components/BaseToggle.vue.d.ts +1 -1
- package/dist/components/BioTemplateExperimentWorkspaceView.vue.d.ts +2 -2
- package/dist/components/BioTemplatePackWorkspaceView.vue.d.ts +1 -1
- package/dist/components/ColorSlider.vue.d.ts +2 -2
- package/dist/components/ConcentrationInput.vue.d.ts +2 -2
- package/dist/components/DatePicker.vue.d.ts +1 -1
- package/dist/components/DateTimePicker.vue.d.ts +2 -2
- package/dist/components/DoseCalculatorVolumeField.vue.d.ts +15 -0
- package/dist/components/DropdownButton.vue.d.ts +1 -1
- package/dist/components/FileUploader.vue.d.ts +2 -2
- package/dist/components/FormulaInput.vue.d.ts +2 -2
- package/dist/components/IconButton.vue.d.ts +1 -1
- package/dist/components/LoadingSpinner.vue.d.ts +1 -1
- package/dist/components/MoleculeInput.vue.d.ts +2 -2
- package/dist/components/MultiSelect.vue.d.ts +1 -1
- package/dist/components/NumberInput.vue.d.ts +1 -1
- package/dist/components/PlateMapEditor.vue.d.ts +6 -6
- package/dist/components/PluginWorkspaceView.controls.d.ts +28 -0
- package/dist/components/PluginWorkspaceView.navigation.d.ts +29 -0
- package/dist/components/PluginWorkspaceView.props.d.ts +151 -0
- package/dist/components/PluginWorkspaceView.shell.d.ts +19 -0
- package/dist/components/PluginWorkspaceView.vue.d.ts +46 -195
- package/dist/components/ProgressBar.vue.d.ts +1 -1
- package/dist/components/ProtocolStep.presentation.d.ts +4 -0
- package/dist/components/ProtocolStepEditor.state.d.ts +18 -0
- package/dist/components/ProtocolStepParameterField.vue.d.ts +12 -0
- package/dist/components/ReagentList.presentation.d.ts +16 -0
- package/dist/components/ResourceCard.vue.d.ts +1 -1
- package/dist/components/SampleSelector.colors.d.ts +13 -0
- package/dist/components/SampleSelector.drag.d.ts +24 -0
- package/dist/components/SampleSelector.groups.d.ts +15 -0
- package/dist/components/SampleSelector.selection.d.ts +26 -0
- package/dist/components/SampleSelector.vue.d.ts +4 -1
- package/dist/components/SampleSelectorSampleRow.vue.d.ts +21 -0
- package/dist/components/SegmentedControl.vue.d.ts +1 -1
- package/dist/components/SequenceInput.vue.d.ts +2 -2
- package/dist/components/SequenceProgressBar.vue.d.ts +1 -1
- package/dist/components/SettingsModal.schema.d.ts +9 -0
- package/dist/components/StatusIndicator.vue.d.ts +1 -1
- package/dist/components/TagsInput.vue.d.ts +2 -2
- package/dist/components/TimePicker.vue.d.ts +2 -2
- package/dist/components/TimeRangeInput.vue.d.ts +1 -1
- package/dist/components/UnitInput.vue.d.ts +2 -2
- package/dist/components/WellPlate.colors.d.ts +9 -0
- package/dist/components/WellPlate.conditions.d.ts +26 -0
- package/dist/components/WellPlate.geometry.d.ts +23 -0
- package/dist/components/WellPlate.interaction.d.ts +71 -0
- package/dist/components/WellPlate.legend.d.ts +2 -0
- package/dist/components/WellPlate.rendering.d.ts +24 -0
- package/dist/components/WellPlate.sampleDrop.d.ts +8 -0
- package/dist/components/WellPlate.vue.d.ts +1 -1
- package/dist/components/index.js +2 -2
- package/dist/components/internal/ActionItemInternal.vue.d.ts +1 -1
- package/dist/components/internal/PlateMapEditorToolbarInternal.vue.d.ts +28 -0
- package/dist/{components-BhK-dW99.js → components-DtHA2bgp.js} +3754 -2991
- package/dist/components-DtHA2bgp.js.map +1 -0
- package/dist/composables/autoGroup/classKey.d.ts +4 -0
- package/dist/composables/autoGroup/classify.d.ts +28 -0
- package/dist/composables/autoGroup/colors.d.ts +2 -0
- package/dist/composables/autoGroup/columns.d.ts +10 -0
- package/dist/composables/autoGroup/compose.d.ts +8 -0
- package/dist/composables/autoGroup/cooccurrence.d.ts +2 -0
- package/dist/composables/autoGroup/csv-shim.d.ts +2 -0
- package/dist/composables/autoGroup/fingerprint.d.ts +3 -0
- package/dist/composables/autoGroup/index.d.ts +16 -0
- package/dist/composables/autoGroup/replicatePreGroup.d.ts +38 -0
- package/dist/composables/autoGroup/template.d.ts +15 -0
- package/dist/composables/autoGroup/tokenize.d.ts +8 -0
- package/dist/composables/autoGroupConstants.d.ts +1 -0
- package/dist/composables/autoGroupGrouping.d.ts +3 -0
- package/dist/composables/controlComponentBindings.d.ts +7 -0
- package/dist/composables/controlSchemaAdapters.d.ts +20 -0
- package/dist/composables/controlSchemaDoseDesign.d.ts +11 -0
- package/dist/composables/controlSchemaFormFields.d.ts +3 -0
- package/dist/composables/controlSchemaLayout.d.ts +7 -0
- package/dist/composables/controlSchemaModel.d.ts +5 -0
- package/dist/composables/controlSchemaNormalize.d.ts +15 -0
- package/dist/composables/controlSchemaTypes.d.ts +305 -0
- package/dist/composables/controlSchemaUtils.d.ts +9 -0
- package/dist/composables/controlWorkspaceOptions.d.ts +2 -0
- package/dist/composables/formBuilderSchema.d.ts +18 -0
- package/dist/composables/index.js +3 -3
- package/dist/composables/pluginEndpointBuilder.d.ts +13 -0
- package/dist/composables/protocolTemplateCatalog.d.ts +26 -0
- package/dist/composables/useAutoGroup.d.ts +61 -74
- package/dist/composables/useAutoGroupInputSources.d.ts +32 -0
- package/dist/composables/useBioTemplateControls.d.ts +1 -1
- package/dist/composables/useBioTemplatePresetWorkspace.d.ts +1 -1
- package/dist/composables/useBioTemplateWorkspace.d.ts +1 -1
- package/dist/composables/useControlSchema.d.ts +8 -346
- package/dist/composables/useControlWorkspace.d.ts +5 -0
- package/dist/composables/useForm.d.ts +2 -33
- package/dist/composables/useFormBuilder.d.ts +2 -9
- package/dist/composables/useFormValidation.d.ts +34 -0
- package/dist/composables/usePluginClient.d.ts +1 -4
- package/dist/composables/useProtocolTemplates.d.ts +2 -24
- package/dist/composables/useScheduleCalendarLayout.d.ts +49 -0
- package/dist/{composables-Bg7CFuNz.js → composables-Dlg8jenH.js} +33 -31
- package/dist/composables-Dlg8jenH.js.map +1 -0
- package/dist/index.js +4 -4
- package/dist/install.js +2 -2
- package/dist/styles.css +547 -516
- package/dist/templates/adapters.d.ts +14 -47
- package/dist/templates/assayLookups.d.ts +3 -0
- package/dist/templates/assayMatrixAdapters.d.ts +6 -0
- package/dist/templates/assayMatrixBuilder.d.ts +2 -0
- package/dist/templates/assayNormalizers.d.ts +4 -0
- package/dist/templates/builderDefaults.d.ts +1 -0
- package/dist/templates/builderIdUtils.d.ts +4 -0
- package/dist/templates/builderPresetControls.d.ts +10 -0
- package/dist/templates/builderReadUtils.d.ts +8 -0
- package/dist/templates/builders.d.ts +26 -67
- package/dist/templates/calibrationCurveAdapters.d.ts +4 -0
- package/dist/templates/calibrationCurveBuilder.d.ts +2 -0
- package/dist/templates/calibrationNormalizers.d.ts +5 -0
- package/dist/templates/componentBindingCatalog.d.ts +25 -0
- package/dist/templates/componentBindingHelpers.d.ts +17 -0
- package/dist/templates/componentDoseResponseProps.d.ts +3 -0
- package/dist/templates/componentGenericProps.d.ts +8 -0
- package/dist/templates/componentPlateHelpers.d.ts +5 -0
- package/dist/templates/componentPlateMapProps.d.ts +3 -0
- package/dist/templates/componentPropsFactory.d.ts +3 -0
- package/dist/templates/componentQpcrPlateProps.d.ts +3 -0
- package/dist/templates/componentTargetResolvers.d.ts +5 -0
- package/dist/templates/componentTemplateProps.d.ts +3 -0
- package/dist/templates/controlSchemaClone.d.ts +4 -0
- package/dist/templates/controlSchemaConstants.d.ts +10 -0
- package/dist/templates/controlSchemaMerge.d.ts +3 -0
- package/dist/templates/controlSchemaTargets.d.ts +17 -0
- package/dist/templates/controlSchemaTypes.d.ts +4 -0
- package/dist/templates/controlSchemas.d.ts +4 -4
- package/dist/templates/defaultBioTemplateBuilder.d.ts +2 -0
- package/dist/templates/doseResponseAdapters.d.ts +4 -0
- package/dist/templates/doseResponseBuilder.d.ts +2 -0
- package/dist/templates/elisaAssayCollectionBuilder.d.ts +2 -0
- package/dist/templates/flowCytometryAssayCollectionBuilder.d.ts +2 -0
- package/dist/templates/flowCytometryPanelBuilder.d.ts +2 -0
- package/dist/templates/flowNormalizers.d.ts +8 -0
- package/dist/templates/flowPanelAdapters.d.ts +4 -0
- package/dist/templates/index.js +1 -1
- package/dist/templates/instrumentRunAdapterHelpers.d.ts +8 -0
- package/dist/templates/instrumentRunAdapters.d.ts +8 -0
- package/dist/templates/instrumentRunBuilder.d.ts +2 -0
- package/dist/templates/lcmsBatchCollectionBuilder.d.ts +2 -0
- package/dist/templates/plateGeometry.d.ts +4 -0
- package/dist/templates/plateMapAdapters.d.ts +3 -0
- package/dist/templates/plateMapBuilder.d.ts +2 -0
- package/dist/templates/presetControlSchemas.d.ts +534 -0
- package/dist/templates/protocolAdapters.d.ts +5 -0
- package/dist/templates/protocolNormalizers.d.ts +6 -0
- package/dist/templates/protocolStepsBuilder.d.ts +2 -0
- package/dist/templates/qpcrAdapters.d.ts +5 -0
- package/dist/templates/qpcrExpressionCollectionBuilder.d.ts +2 -0
- package/dist/templates/qpcrPlateBuilder.d.ts +2 -0
- package/dist/templates/reagentAdapters.d.ts +5 -0
- package/dist/templates/reagentListBuilder.d.ts +2 -0
- package/dist/templates/runNormalizers.d.ts +10 -0
- package/dist/templates/sampleNormalizers.d.ts +4 -0
- package/dist/templates/samplePrepAdapters.d.ts +4 -0
- package/dist/templates/samplePrepBuilder.d.ts +2 -0
- package/dist/templates/sampleSheetAdapters.d.ts +5 -0
- package/dist/templates/sampleSheetBuilder.d.ts +2 -0
- package/dist/templates/targetedMetabolomicsCollectionBuilder.d.ts +2 -0
- package/dist/templates/targetedMetabolomicsHelpers.d.ts +5 -0
- package/dist/templates/templateAdapterTypes.d.ts +48 -0
- package/dist/templates/templateControlSchemas.d.ts +400 -0
- package/dist/templates/templateCreateOptions.d.ts +165 -0
- package/dist/templates/templateEnvelopes.d.ts +9 -0
- package/dist/templates/templatePackCollectionBuilder.d.ts +2 -0
- package/dist/templates/templatePresetCollectionBuilder.d.ts +18 -0
- package/dist/templates/templateQpcrTypes.d.ts +42 -0
- package/dist/templates/templateValidators.d.ts +13 -0
- package/dist/templates/timeCourseAdapters.d.ts +5 -0
- package/dist/templates/timeCourseBuilder.d.ts +2 -0
- package/dist/templates/types.d.ts +5 -250
- package/dist/templates/wellPlateScreenCollectionBuilder.d.ts +2 -0
- package/dist/templates/westernBlotAssayCollectionBuilder.d.ts +2 -0
- package/dist/{templates-BorLR_7p.js → templates-DtdUvJ4c.js} +3565 -3411
- package/dist/templates-DtdUvJ4c.js.map +1 -0
- package/dist/types/auto-group.d.ts +79 -9
- package/dist/types/componentLabTypes.d.ts +161 -0
- package/dist/types/componentWorkflowTypes.d.ts +150 -0
- package/dist/types/components.d.ts +2 -311
- package/dist/{useProtocolTemplates-n6AJqSqv.js → useProtocolTemplates-Bm5vyH4_.js} +1220 -454
- package/dist/useProtocolTemplates-Bm5vyH4_.js.map +1 -0
- package/package.json +1 -1
- package/src/__tests__/components/AppTopBar.navigation.test.ts +70 -0
- package/src/__tests__/components/DoseCalculatorVolumeField.test.ts +53 -0
- package/src/__tests__/components/PlateMapEditorToolbarInternal.test.ts +54 -0
- package/src/__tests__/components/PluginWorkspaceView.controls.test.ts +156 -0
- package/src/__tests__/components/PluginWorkspaceView.navigation.test.ts +102 -0
- package/src/__tests__/components/PluginWorkspaceView.shell.test.ts +41 -0
- package/src/__tests__/components/ProtocolStep.presentation.test.ts +31 -0
- package/src/__tests__/components/ProtocolStepEditor.state.test.ts +165 -0
- package/src/__tests__/components/ProtocolStepParameterField.test.ts +44 -0
- package/src/__tests__/components/ReagentList.presentation.test.ts +68 -0
- package/src/__tests__/components/SampleSelector.colors.test.ts +49 -0
- package/src/__tests__/components/SampleSelector.drag.test.ts +100 -0
- package/src/__tests__/components/SampleSelector.groups.test.ts +81 -0
- package/src/__tests__/components/SampleSelector.selection.test.ts +70 -0
- package/src/__tests__/components/SampleSelector.test.ts +32 -0
- package/src/__tests__/components/SampleSelectorSampleRow.test.ts +37 -0
- package/src/__tests__/components/ScheduleCalendar.test.ts +44 -0
- package/src/__tests__/components/SettingsModal.schema.test.ts +97 -0
- package/src/__tests__/components/WellPlate.colors.test.ts +28 -0
- package/src/__tests__/components/WellPlate.conditions.test.ts +68 -0
- package/src/__tests__/components/WellPlate.geometry.test.ts +54 -0
- package/src/__tests__/components/WellPlate.interaction.test.ts +171 -0
- package/src/__tests__/components/WellPlate.legend.test.ts +13 -0
- package/src/__tests__/components/WellPlate.rendering.test.ts +122 -0
- package/src/__tests__/components/WellPlate.sampleDrop.test.ts +70 -0
- package/src/__tests__/composables/autoGroup/classify.test.ts +107 -0
- package/src/__tests__/composables/autoGroup/columns.test.ts +135 -0
- package/src/__tests__/composables/autoGroup/compose.test.ts +227 -0
- package/src/__tests__/composables/autoGroup/cooccurrence.test.ts +91 -0
- package/src/__tests__/composables/autoGroup/fingerprint.test.ts +50 -0
- package/src/__tests__/composables/autoGroup/integration.test.ts +79 -0
- package/src/__tests__/composables/autoGroup/template.test.ts +70 -0
- package/src/__tests__/composables/autoGroup/tokenize.test.ts +33 -0
- package/src/__tests__/composables/useAutoGroup.test.ts +129 -625
- package/src/__tests__/composables/useAutoGroupInputSources.test.ts +107 -0
- package/src/__tests__/composables/useControlSchema.test.ts +23 -0
- package/src/__tests__/composables/useScheduleCalendarLayout.test.ts +89 -0
- package/src/__tests__/docs/extractDocsComponents.test.ts +142 -0
- package/src/__tests__/docs/extractDocsExports.test.ts +77 -0
- package/src/__tests__/docs/extractDocsParsing.test.ts +69 -0
- package/src/__tests__/docs/extractDocsTemplates.test.ts +54 -0
- package/src/__tests__/docs/extractDocsTheme.test.ts +89 -0
- package/src/__tests__/docs/frontendDocsCatalog.test.ts +1 -1
- package/src/__tests__/fixtures/auto-group/mixed-lc-ms-batch.txt +187 -0
- package/src/components/AppSidebar.vue +2 -6
- package/src/components/AppTopBar.navigation.ts +62 -0
- package/src/components/AppTopBar.vue +17 -44
- package/src/components/AutoGroupModal.story.vue +50 -0
- package/src/components/AutoGroupModal.vue +441 -158
- package/src/components/ControlWorkspaceView.vue +2 -6
- package/src/components/DoseCalculator.vue +13 -73
- package/src/components/DoseCalculatorVolumeField.vue +61 -0
- package/src/components/ExperimentTimeline.vue +6 -31
- package/src/components/FormBuilder.vue +2 -7
- package/src/components/PlateMapEditor.vue +32 -106
- package/src/components/PluginWorkspaceView.controls.ts +182 -0
- package/src/components/PluginWorkspaceView.navigation.ts +106 -0
- package/src/components/PluginWorkspaceView.props.ts +174 -0
- package/src/components/PluginWorkspaceView.shell.ts +66 -0
- package/src/components/PluginWorkspaceView.vue +85 -404
- package/src/components/ProtocolStep.presentation.ts +31 -0
- package/src/components/ProtocolStepEditor.state.ts +104 -0
- package/src/components/ProtocolStepEditor.vue +48 -179
- package/src/components/ProtocolStepParameterField.vue +134 -0
- package/src/components/ReagentList.presentation.ts +105 -0
- package/src/components/ReagentList.vue +16 -79
- package/src/components/SampleSelector.colors.ts +43 -0
- package/src/components/SampleSelector.drag.ts +164 -0
- package/src/components/SampleSelector.groups.ts +109 -0
- package/src/components/SampleSelector.selection.ts +103 -0
- package/src/components/SampleSelector.vue +82 -349
- package/src/components/SampleSelectorSampleRow.vue +64 -0
- package/src/components/ScheduleCalendar.vue +44 -199
- package/src/components/SettingsModal.schema.ts +71 -0
- package/src/components/SettingsModal.vue +16 -46
- package/src/components/WellPlate.colors.ts +56 -0
- package/src/components/WellPlate.conditions.ts +100 -0
- package/src/components/WellPlate.geometry.ts +91 -0
- package/src/components/WellPlate.interaction.ts +272 -0
- package/src/components/WellPlate.legend.ts +8 -0
- package/src/components/WellPlate.rendering.ts +105 -0
- package/src/components/WellPlate.sampleDrop.ts +73 -0
- package/src/components/WellPlate.vue +102 -550
- package/src/components/internal/PlateMapEditorToolbarInternal.vue +128 -0
- package/src/composables/autoGroup/classKey.ts +5 -0
- package/src/composables/autoGroup/classify.ts +205 -0
- package/src/composables/autoGroup/colors.ts +6 -0
- package/src/composables/autoGroup/columns.ts +226 -0
- package/src/composables/autoGroup/compose.ts +156 -0
- package/src/composables/autoGroup/cooccurrence.ts +46 -0
- package/src/composables/autoGroup/csv-shim.ts +44 -0
- package/src/composables/autoGroup/fingerprint.ts +49 -0
- package/src/composables/autoGroup/index.ts +20 -0
- package/src/composables/autoGroup/replicatePreGroup.ts +90 -0
- package/src/composables/autoGroup/template.ts +126 -0
- package/src/composables/autoGroup/tokenize.ts +41 -0
- package/src/composables/autoGroup/vocab.json +67 -0
- package/src/composables/autoGroupConstants.ts +4 -0
- package/src/composables/autoGroupGrouping.ts +148 -0
- package/src/composables/controlComponentBindings.ts +80 -0
- package/src/composables/controlSchemaAdapters.ts +196 -0
- package/src/composables/controlSchemaDoseDesign.ts +215 -0
- package/src/composables/controlSchemaFormFields.ts +61 -0
- package/src/composables/controlSchemaLayout.ts +59 -0
- package/src/composables/controlSchemaModel.ts +163 -0
- package/src/composables/controlSchemaNormalize.ts +101 -0
- package/src/composables/controlSchemaTypes.ts +364 -0
- package/src/composables/controlSchemaUtils.ts +36 -0
- package/src/composables/controlWorkspaceOptions.ts +21 -0
- package/src/composables/formBuilderSchema.ts +153 -0
- package/src/composables/pluginEndpointBuilder.ts +203 -0
- package/src/composables/protocolTemplateCatalog.ts +325 -0
- package/src/composables/useAutoGroup.ts +395 -549
- package/src/composables/useAutoGroupInputSources.ts +147 -0
- package/src/composables/useBioTemplateControls.ts +1 -1
- package/src/composables/useBioTemplatePresetWorkspace.ts +1 -1
- package/src/composables/useBioTemplateWorkspace.ts +1 -1
- package/src/composables/useControlSchema.ts +64 -1312
- package/src/composables/useControlWorkspace.ts +201 -0
- package/src/composables/useForm.ts +5 -187
- package/src/composables/useFormBuilder.ts +11 -153
- package/src/composables/useFormValidation.ts +154 -0
- package/src/composables/usePluginClient.ts +10 -193
- package/src/composables/useProtocolTemplates.ts +10 -328
- package/src/composables/useScheduleCalendarLayout.ts +287 -0
- package/src/styles/components/auto-group-modal.css +248 -310
- package/src/templates/adapters.ts +89 -930
- package/src/templates/assayLookups.ts +33 -0
- package/src/templates/assayMatrixAdapters.ts +78 -0
- package/src/templates/assayMatrixBuilder.ts +59 -0
- package/src/templates/assayNormalizers.ts +34 -0
- package/src/templates/builderDefaults.ts +11 -0
- package/src/templates/builderIdUtils.ts +20 -0
- package/src/templates/builderPresetControls.ts +165 -0
- package/src/templates/builderReadUtils.ts +57 -0
- package/src/templates/builders.ts +122 -2350
- package/src/templates/calibrationCurveAdapters.ts +59 -0
- package/src/templates/calibrationCurveBuilder.ts +99 -0
- package/src/templates/calibrationNormalizers.ts +60 -0
- package/src/templates/componentBindingCatalog.ts +90 -0
- package/src/templates/componentBindingHelpers.ts +93 -0
- package/src/templates/componentBindings.ts +12 -461
- package/src/templates/componentDoseResponseProps.ts +42 -0
- package/src/templates/componentGenericProps.ts +77 -0
- package/src/templates/componentPlateHelpers.ts +29 -0
- package/src/templates/componentPlateMapProps.ts +32 -0
- package/src/templates/componentPropsFactory.ts +21 -0
- package/src/templates/componentQpcrPlateProps.ts +28 -0
- package/src/templates/componentTargetResolvers.ts +69 -0
- package/src/templates/componentTemplateProps.ts +78 -0
- package/src/templates/controlSchemaClone.ts +32 -0
- package/src/templates/controlSchemaConstants.ts +11 -0
- package/src/templates/controlSchemaMerge.ts +40 -0
- package/src/templates/controlSchemaTargets.ts +87 -0
- package/src/templates/controlSchemaTypes.ts +20 -0
- package/src/templates/controlSchemas.ts +22 -704
- package/src/templates/defaultBioTemplateBuilder.ts +124 -0
- package/src/templates/doseResponseAdapters.ts +45 -0
- package/src/templates/doseResponseBuilder.ts +44 -0
- package/src/templates/elisaAssayCollectionBuilder.ts +62 -0
- package/src/templates/flowCytometryAssayCollectionBuilder.ts +41 -0
- package/src/templates/flowCytometryPanelBuilder.ts +53 -0
- package/src/templates/flowNormalizers.ts +58 -0
- package/src/templates/flowPanelAdapters.ts +58 -0
- package/src/templates/instrumentRunAdapterHelpers.ts +94 -0
- package/src/templates/instrumentRunAdapters.ts +163 -0
- package/src/templates/instrumentRunBuilder.ts +97 -0
- package/src/templates/lcmsBatchCollectionBuilder.ts +38 -0
- package/src/templates/plateGeometry.ts +62 -0
- package/src/templates/plateMapAdapters.ts +36 -0
- package/src/templates/plateMapBuilder.ts +43 -0
- package/src/templates/presetControlSchemas.ts +258 -0
- package/src/templates/protocolAdapters.ts +69 -0
- package/src/templates/protocolNormalizers.ts +37 -0
- package/src/templates/protocolStepsBuilder.ts +36 -0
- package/src/templates/qpcrAdapters.ts +104 -0
- package/src/templates/qpcrExpressionCollectionBuilder.ts +33 -0
- package/src/templates/qpcrPlateBuilder.ts +96 -0
- package/src/templates/reagentAdapters.ts +77 -0
- package/src/templates/reagentListBuilder.ts +30 -0
- package/src/templates/runNormalizers.ts +63 -0
- package/src/templates/sampleNormalizers.ts +58 -0
- package/src/templates/samplePrepAdapters.ts +63 -0
- package/src/templates/samplePrepBuilder.ts +51 -0
- package/src/templates/sampleSheetAdapters.ts +75 -0
- package/src/templates/sampleSheetBuilder.ts +23 -0
- package/src/templates/targetedMetabolomicsCollectionBuilder.ts +79 -0
- package/src/templates/targetedMetabolomicsHelpers.ts +102 -0
- package/src/templates/templateAdapterTypes.ts +58 -0
- package/src/templates/templateControlSchemas.ts +320 -0
- package/src/templates/templateCreateOptions.ts +208 -0
- package/src/templates/templateEnvelopes.ts +137 -0
- package/src/templates/templatePackCollectionBuilder.ts +23 -0
- package/src/templates/templatePresetCollectionBuilder.ts +139 -0
- package/src/templates/templateQpcrTypes.ts +48 -0
- package/src/templates/templateValidators.ts +414 -0
- package/src/templates/timeCourseAdapters.ts +73 -0
- package/src/templates/timeCourseBuilder.ts +64 -0
- package/src/templates/types.ts +79 -275
- package/src/templates/wellPlateScreenCollectionBuilder.ts +36 -0
- package/src/templates/westernBlotAssayCollectionBuilder.ts +68 -0
- package/src/types/auto-group.ts +107 -9
- package/src/types/componentLabTypes.ts +235 -0
- package/src/types/componentWorkflowTypes.ts +190 -0
- package/src/types/components.ts +74 -424
- package/dist/components-BhK-dW99.js.map +0 -1
- package/dist/composables-Bg7CFuNz.js.map +0 -1
- package/dist/templates-BorLR_7p.js.map +0 -1
- package/dist/useProtocolTemplates-n6AJqSqv.js.map +0 -1
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@morscherlab/mint-sdk",
|
|
3
|
-
"version": "1.0.0-rc.
|
|
3
|
+
"version": "1.0.0-rc.5",
|
|
4
4
|
"description": "MINT Platform SDK — Vue 3 components, composables, and types for plugin development. MINT = Mass-spec INtegrated Toolkit.",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"main": "./dist/index.js",
|
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
import { describe, expect, it } from 'vitest'
|
|
2
|
+
import {
|
|
3
|
+
currentItemIdFromLocation,
|
|
4
|
+
currentPagePath,
|
|
5
|
+
normalizeNavPath,
|
|
6
|
+
pageIdFromPath,
|
|
7
|
+
pluginNavItemToPageSelectorItem,
|
|
8
|
+
} from '../../components/AppTopBar.navigation'
|
|
9
|
+
|
|
10
|
+
describe('AppTopBar navigation helpers', () => {
|
|
11
|
+
it('normalizes blank, relative, and trailing-slash paths', () => {
|
|
12
|
+
expect(normalizeNavPath()).toBe('/')
|
|
13
|
+
expect(normalizeNavPath('results')).toBe('/results')
|
|
14
|
+
expect(normalizeNavPath('/results/')).toBe('/results')
|
|
15
|
+
expect(normalizeNavPath(' /analysis/run/ ')).toBe('/analysis/run')
|
|
16
|
+
})
|
|
17
|
+
|
|
18
|
+
it('derives stable page ids from paths', () => {
|
|
19
|
+
expect(pageIdFromPath('/', 'fallback')).toBe('dashboard')
|
|
20
|
+
expect(pageIdFromPath('/analysis/run', 'fallback')).toBe('analysis-run')
|
|
21
|
+
expect(pageIdFromPath('', 'fallback')).toBe('dashboard')
|
|
22
|
+
})
|
|
23
|
+
|
|
24
|
+
it('maps plugin nav items to page selector items with plugin fallbacks', () => {
|
|
25
|
+
expect(
|
|
26
|
+
pluginNavItemToPageSelectorItem(
|
|
27
|
+
{ label: 'Results', path: 'results' },
|
|
28
|
+
1,
|
|
29
|
+
{ pluginIcon: 'flask', pluginName: 'Assay Plugin' },
|
|
30
|
+
),
|
|
31
|
+
).toEqual({
|
|
32
|
+
id: 'results',
|
|
33
|
+
label: 'Results',
|
|
34
|
+
to: '/results',
|
|
35
|
+
icon: 'flask',
|
|
36
|
+
hint: 'Assay Plugin',
|
|
37
|
+
})
|
|
38
|
+
})
|
|
39
|
+
|
|
40
|
+
it('prefers explicit nav item id, icon, and description', () => {
|
|
41
|
+
expect(
|
|
42
|
+
pluginNavItemToPageSelectorItem(
|
|
43
|
+
{
|
|
44
|
+
id: 'custom',
|
|
45
|
+
label: 'Custom',
|
|
46
|
+
path: '/custom',
|
|
47
|
+
icon: 'sparkles',
|
|
48
|
+
description: 'Custom page',
|
|
49
|
+
},
|
|
50
|
+
0,
|
|
51
|
+
{ pluginIcon: 'flask', pluginName: 'Assay Plugin' },
|
|
52
|
+
),
|
|
53
|
+
).toMatchObject({
|
|
54
|
+
id: 'custom',
|
|
55
|
+
icon: 'sparkles',
|
|
56
|
+
hint: 'Custom page',
|
|
57
|
+
})
|
|
58
|
+
})
|
|
59
|
+
|
|
60
|
+
it('matches current route-relative page ids for integrated plugins', () => {
|
|
61
|
+
const pages = [
|
|
62
|
+
{ id: 'overview', to: '/' },
|
|
63
|
+
{ id: 'results', to: '/results' },
|
|
64
|
+
]
|
|
65
|
+
|
|
66
|
+
expect(currentPagePath('/plugins/assay', '/plugins/assay')).toBe('/')
|
|
67
|
+
expect(currentPagePath('/plugins/assay/results', '/plugins/assay')).toBe('/results')
|
|
68
|
+
expect(currentItemIdFromLocation(pages, '/plugins/assay/results', '/plugins/assay')).toBe('results')
|
|
69
|
+
})
|
|
70
|
+
})
|
|
@@ -0,0 +1,53 @@
|
|
|
1
|
+
import { mount } from '@vue/test-utils'
|
|
2
|
+
import { describe, expect, it } from 'vitest'
|
|
3
|
+
import DoseCalculatorVolumeField from '../../components/DoseCalculatorVolumeField.vue'
|
|
4
|
+
|
|
5
|
+
describe('DoseCalculatorVolumeField', () => {
|
|
6
|
+
it('emits value updates while preserving the current unit', async () => {
|
|
7
|
+
const wrapper = mount(DoseCalculatorVolumeField, {
|
|
8
|
+
props: {
|
|
9
|
+
label: 'Final Volume',
|
|
10
|
+
modelValue: { value: 1, unit: 'mL' },
|
|
11
|
+
units: ['µL', 'mL', 'L'],
|
|
12
|
+
},
|
|
13
|
+
})
|
|
14
|
+
|
|
15
|
+
await wrapper.get('input').setValue('2.5')
|
|
16
|
+
|
|
17
|
+
expect(wrapper.emitted('update:modelValue')?.at(-1)).toEqual([
|
|
18
|
+
{ value: 2.5, unit: 'mL' },
|
|
19
|
+
])
|
|
20
|
+
})
|
|
21
|
+
|
|
22
|
+
it('emits unit updates while preserving the current value', async () => {
|
|
23
|
+
const wrapper = mount(DoseCalculatorVolumeField, {
|
|
24
|
+
props: {
|
|
25
|
+
label: 'Volume per Well',
|
|
26
|
+
modelValue: { value: 100, unit: 'µL' },
|
|
27
|
+
units: ['µL', 'mL', 'L'],
|
|
28
|
+
},
|
|
29
|
+
})
|
|
30
|
+
|
|
31
|
+
await wrapper.get('select').setValue('mL')
|
|
32
|
+
|
|
33
|
+
expect(wrapper.emitted('update:modelValue')?.at(-1)).toEqual([
|
|
34
|
+
{ value: 100, unit: 'mL' },
|
|
35
|
+
])
|
|
36
|
+
})
|
|
37
|
+
|
|
38
|
+
it('ignores non-numeric value input', async () => {
|
|
39
|
+
const wrapper = mount(DoseCalculatorVolumeField, {
|
|
40
|
+
props: {
|
|
41
|
+
label: 'Final Volume',
|
|
42
|
+
modelValue: { value: 1, unit: 'mL' },
|
|
43
|
+
units: ['µL', 'mL', 'L'],
|
|
44
|
+
},
|
|
45
|
+
})
|
|
46
|
+
|
|
47
|
+
const input = wrapper.get('input')
|
|
48
|
+
;(input.element as HTMLInputElement).value = ''
|
|
49
|
+
await input.trigger('input')
|
|
50
|
+
|
|
51
|
+
expect(wrapper.emitted('update:modelValue')).toBeUndefined()
|
|
52
|
+
})
|
|
53
|
+
})
|
|
@@ -0,0 +1,54 @@
|
|
|
1
|
+
import { mount } from '@vue/test-utils'
|
|
2
|
+
import { describe, expect, it, vi } from 'vitest'
|
|
3
|
+
import PlateMapEditorToolbarInternal from '../../components/internal/PlateMapEditorToolbarInternal.vue'
|
|
4
|
+
import type { PlateMap } from '../../types'
|
|
5
|
+
|
|
6
|
+
const plates: PlateMap[] = [
|
|
7
|
+
{ id: 'plate-1', name: 'Plate 1', format: 96, wells: {} },
|
|
8
|
+
{ id: 'plate-2', name: 'Plate 2', format: 96, wells: {} },
|
|
9
|
+
]
|
|
10
|
+
|
|
11
|
+
function mountToolbar(props = {}) {
|
|
12
|
+
return mount(PlateMapEditorToolbarInternal, {
|
|
13
|
+
props: {
|
|
14
|
+
plates,
|
|
15
|
+
activePlateId: 'plate-1',
|
|
16
|
+
plateWellCounts: { 'plate-1': 3, 'plate-2': 0 },
|
|
17
|
+
canAddPlate: true,
|
|
18
|
+
canUndo: false,
|
|
19
|
+
canRedo: true,
|
|
20
|
+
slotColorForPlate: vi.fn((_plateId: string, index: number) => (index === 0 ? '#ef4444' : '#22c55e')),
|
|
21
|
+
...props,
|
|
22
|
+
},
|
|
23
|
+
})
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
describe('PlateMapEditorToolbarInternal', () => {
|
|
27
|
+
it('renders plate tabs, active state, and well counts', () => {
|
|
28
|
+
const wrapper = mountToolbar()
|
|
29
|
+
|
|
30
|
+
const tabs = wrapper.findAll('.mint-plate-editor__tab')
|
|
31
|
+
expect(tabs).toHaveLength(2)
|
|
32
|
+
expect(tabs[0].classes()).toContain('mint-plate-editor__tab--active')
|
|
33
|
+
expect(wrapper.find('.mint-plate-editor__tab-count').text()).toBe('3')
|
|
34
|
+
expect(wrapper.find('.mint-plate-editor__add-plate').exists()).toBe(true)
|
|
35
|
+
})
|
|
36
|
+
|
|
37
|
+
it('emits tab and action events', async () => {
|
|
38
|
+
const wrapper = mountToolbar()
|
|
39
|
+
|
|
40
|
+
await wrapper.findAll('.mint-plate-editor__tab')[1].trigger('click')
|
|
41
|
+
await wrapper.findAll('.mint-plate-editor__tab-remove')[0].trigger('click')
|
|
42
|
+
await wrapper.find('.mint-plate-editor__add-plate').trigger('click')
|
|
43
|
+
await wrapper.find('[title="Redo (Ctrl+Shift+Z)"]').trigger('click')
|
|
44
|
+
await wrapper.find('[title="Import"]').trigger('click')
|
|
45
|
+
await wrapper.find('[title="Export JSON"]').trigger('click')
|
|
46
|
+
|
|
47
|
+
expect(wrapper.emitted('select-plate')?.[0]).toEqual(['plate-2'])
|
|
48
|
+
expect(wrapper.emitted('remove-plate')?.[0]).toEqual(['plate-1'])
|
|
49
|
+
expect(wrapper.emitted('add-plate')).toHaveLength(1)
|
|
50
|
+
expect(wrapper.emitted('redo')).toHaveLength(1)
|
|
51
|
+
expect(wrapper.emitted('import')).toHaveLength(1)
|
|
52
|
+
expect(wrapper.emitted('export-json')).toHaveLength(1)
|
|
53
|
+
})
|
|
54
|
+
})
|
|
@@ -0,0 +1,156 @@
|
|
|
1
|
+
import { nextTick, ref } from 'vue'
|
|
2
|
+
import { describe, expect, it, vi } from 'vitest'
|
|
3
|
+
import { defineDoseDesignControlModel } from '../../composables/useControlSchema'
|
|
4
|
+
import { usePluginWorkspaceControls } from '../../components/PluginWorkspaceView.controls'
|
|
5
|
+
import type {
|
|
6
|
+
ControlComponentBindingsConfig,
|
|
7
|
+
ControlComponentPropsByIdMap,
|
|
8
|
+
ControlComponentPropsMap,
|
|
9
|
+
ControlSchema,
|
|
10
|
+
} from '../../composables/useControlSchema'
|
|
11
|
+
|
|
12
|
+
interface CreateControlsOptions {
|
|
13
|
+
controls?: ControlSchema
|
|
14
|
+
modelValue?: () => Record<string, unknown> | undefined
|
|
15
|
+
values?: () => Record<string, unknown> | undefined
|
|
16
|
+
componentProps?: () => ControlComponentPropsMap | undefined
|
|
17
|
+
componentPropsById?: () => ControlComponentPropsByIdMap | undefined
|
|
18
|
+
componentBindings?: () => ControlComponentBindingsConfig | undefined
|
|
19
|
+
emitValues?: (values: Record<string, unknown>) => void
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
function createControls({
|
|
23
|
+
controls,
|
|
24
|
+
modelValue,
|
|
25
|
+
values,
|
|
26
|
+
componentProps,
|
|
27
|
+
componentPropsById,
|
|
28
|
+
componentBindings,
|
|
29
|
+
emitValues = vi.fn(),
|
|
30
|
+
}: CreateControlsOptions = {}) {
|
|
31
|
+
return usePluginWorkspaceControls({
|
|
32
|
+
model: () => undefined,
|
|
33
|
+
controls: () => controls,
|
|
34
|
+
controlOptions: () => ({}),
|
|
35
|
+
modelValue: modelValue ?? (() => undefined),
|
|
36
|
+
values: values ?? (() => undefined),
|
|
37
|
+
componentBindings: componentBindings ?? (() => undefined),
|
|
38
|
+
componentProps: componentProps ?? (() => undefined),
|
|
39
|
+
componentPropsById: componentPropsById ?? (() => undefined),
|
|
40
|
+
emitValues,
|
|
41
|
+
})
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
describe('PluginWorkspaceView control state helpers', () => {
|
|
45
|
+
it('merges control defaults with external values and resyncs when values change', async () => {
|
|
46
|
+
const externalValues = ref<Record<string, unknown> | undefined>({ threshold: 0.2 })
|
|
47
|
+
const state = createControls({
|
|
48
|
+
controls: {
|
|
49
|
+
threshold: { label: 'Threshold', default: 0.05 },
|
|
50
|
+
mode: { label: 'Mode', default: 'fast' },
|
|
51
|
+
},
|
|
52
|
+
modelValue: () => externalValues.value,
|
|
53
|
+
})
|
|
54
|
+
|
|
55
|
+
expect(state.internalControlValues).toEqual({
|
|
56
|
+
threshold: 0.2,
|
|
57
|
+
mode: 'fast',
|
|
58
|
+
})
|
|
59
|
+
|
|
60
|
+
externalValues.value = { threshold: 0.8 }
|
|
61
|
+
await nextTick()
|
|
62
|
+
|
|
63
|
+
expect(state.internalControlValues).toEqual({
|
|
64
|
+
threshold: 0.8,
|
|
65
|
+
mode: 'fast',
|
|
66
|
+
})
|
|
67
|
+
})
|
|
68
|
+
|
|
69
|
+
it('emits updates from sidebar form values without dropping generated defaults', () => {
|
|
70
|
+
const emitValues = vi.fn()
|
|
71
|
+
const state = createControls({
|
|
72
|
+
controls: {
|
|
73
|
+
threshold: { label: 'Threshold', default: 0.05 },
|
|
74
|
+
mode: { label: 'Mode', default: 'fast' },
|
|
75
|
+
},
|
|
76
|
+
emitValues,
|
|
77
|
+
})
|
|
78
|
+
|
|
79
|
+
state.handleControlValuesUpdate({ threshold: 0.4 })
|
|
80
|
+
|
|
81
|
+
expect(state.internalControlValues).toEqual({
|
|
82
|
+
threshold: 0.4,
|
|
83
|
+
mode: 'fast',
|
|
84
|
+
})
|
|
85
|
+
expect(emitValues).toHaveBeenCalledWith({
|
|
86
|
+
threshold: 0.4,
|
|
87
|
+
mode: 'fast',
|
|
88
|
+
})
|
|
89
|
+
})
|
|
90
|
+
|
|
91
|
+
it('derives component props and bindings from one internal value model', () => {
|
|
92
|
+
const state = createControls({
|
|
93
|
+
controls: {
|
|
94
|
+
threshold: { label: 'Threshold', default: 0.05 },
|
|
95
|
+
mode: { label: 'Mode', default: 'fast' },
|
|
96
|
+
},
|
|
97
|
+
componentProps: () => ['threshold'],
|
|
98
|
+
componentPropsById: () => ({
|
|
99
|
+
chart: ['threshold', 'mode'],
|
|
100
|
+
}),
|
|
101
|
+
componentBindings: () => ({
|
|
102
|
+
chart: {
|
|
103
|
+
component: 'ChartContainer',
|
|
104
|
+
props: ['threshold'],
|
|
105
|
+
},
|
|
106
|
+
}),
|
|
107
|
+
})
|
|
108
|
+
|
|
109
|
+
expect(state.resolvedComponentProps.value).toEqual({ threshold: 0.05 })
|
|
110
|
+
expect(state.resolvedComponentPropsById.value).toEqual({
|
|
111
|
+
chart: { threshold: 0.05, mode: 'fast' },
|
|
112
|
+
})
|
|
113
|
+
expect(state.resolvedComponentBindingsById.value.chart).toMatchObject({
|
|
114
|
+
component: 'ChartContainer',
|
|
115
|
+
props: { threshold: 0.05 },
|
|
116
|
+
})
|
|
117
|
+
})
|
|
118
|
+
|
|
119
|
+
it('uses compact control model navigation and binding defaults', () => {
|
|
120
|
+
const model = defineDoseDesignControlModel({
|
|
121
|
+
selectedWells: ['A1', 'A2'],
|
|
122
|
+
includeMolecularWeight: true,
|
|
123
|
+
})
|
|
124
|
+
const state = usePluginWorkspaceControls({
|
|
125
|
+
model: () => model,
|
|
126
|
+
controls: () => undefined,
|
|
127
|
+
controlOptions: () => ({}),
|
|
128
|
+
modelValue: () => ({
|
|
129
|
+
selectedWells: ['C1'],
|
|
130
|
+
plateFormat: 384,
|
|
131
|
+
doseMode: 'dilution',
|
|
132
|
+
disabled: true,
|
|
133
|
+
molecularWeight: 450,
|
|
134
|
+
}),
|
|
135
|
+
values: () => undefined,
|
|
136
|
+
componentBindings: () => undefined,
|
|
137
|
+
componentProps: () => undefined,
|
|
138
|
+
componentPropsById: () => undefined,
|
|
139
|
+
emitValues: vi.fn(),
|
|
140
|
+
})
|
|
141
|
+
|
|
142
|
+
expect(state.generatedPillNav.value.map(item => item.id)).toEqual(['design'])
|
|
143
|
+
expect(state.generatedDefaultView.value).toBe('design')
|
|
144
|
+
expect(state.resolvedComponentBindingsById.value.plate.props).toMatchObject({
|
|
145
|
+
modelValue: ['C1'],
|
|
146
|
+
format: 384,
|
|
147
|
+
disabled: true,
|
|
148
|
+
})
|
|
149
|
+
expect(state.resolvedComponentPropsById.value.dose).toMatchObject({
|
|
150
|
+
mode: 'dilution',
|
|
151
|
+
targetWells: ['C1'],
|
|
152
|
+
disabled: true,
|
|
153
|
+
molecularWeight: 450,
|
|
154
|
+
})
|
|
155
|
+
})
|
|
156
|
+
})
|
|
@@ -0,0 +1,102 @@
|
|
|
1
|
+
import { ref } from 'vue'
|
|
2
|
+
import { describe, expect, it } from 'vitest'
|
|
3
|
+
import type {
|
|
4
|
+
PageSelectorItem,
|
|
5
|
+
PageSelectorItemInput,
|
|
6
|
+
PillNavItem,
|
|
7
|
+
PillNavItemInput,
|
|
8
|
+
} from '../../types'
|
|
9
|
+
import { usePluginWorkspaceNavigation } from '../../components/PluginWorkspaceView.navigation'
|
|
10
|
+
|
|
11
|
+
function makeNavigation(overrides: {
|
|
12
|
+
activeView?: string
|
|
13
|
+
defaultActiveView?: string
|
|
14
|
+
currentPillId?: string
|
|
15
|
+
currentPageSelectorId?: string
|
|
16
|
+
pillNav?: PillNavItemInput[]
|
|
17
|
+
pageSelector?: PageSelectorItemInput[]
|
|
18
|
+
panelIds?: string[]
|
|
19
|
+
generatedPillNav?: PillNavItem[]
|
|
20
|
+
generatedDefaultView?: string
|
|
21
|
+
} = {}) {
|
|
22
|
+
const activeView = ref(overrides.activeView)
|
|
23
|
+
const defaultActiveView = ref(overrides.defaultActiveView)
|
|
24
|
+
const currentPillId = ref(overrides.currentPillId)
|
|
25
|
+
const currentPageSelectorId = ref(overrides.currentPageSelectorId)
|
|
26
|
+
const pillNav = ref(overrides.pillNav)
|
|
27
|
+
const pageSelector = ref(overrides.pageSelector)
|
|
28
|
+
const panelIds = ref(overrides.panelIds ?? [])
|
|
29
|
+
const generatedPillNav = ref<PillNavItem[]>(overrides.generatedPillNav ?? [])
|
|
30
|
+
const generatedDefaultView = ref(overrides.generatedDefaultView ?? '')
|
|
31
|
+
const emitted = {
|
|
32
|
+
activeView: [] as string[],
|
|
33
|
+
page: [] as PageSelectorItem[],
|
|
34
|
+
pill: [] as PillNavItem[],
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
const navigation = usePluginWorkspaceNavigation({
|
|
38
|
+
activeView: () => activeView.value,
|
|
39
|
+
defaultActiveView: () => defaultActiveView.value,
|
|
40
|
+
currentPillId: () => currentPillId.value,
|
|
41
|
+
currentPageSelectorId: () => currentPageSelectorId.value,
|
|
42
|
+
pillNav: () => pillNav.value,
|
|
43
|
+
pageSelector: () => pageSelector.value,
|
|
44
|
+
panelIds: () => panelIds.value,
|
|
45
|
+
generatedPillNav,
|
|
46
|
+
generatedDefaultView,
|
|
47
|
+
emitActiveView: viewId => emitted.activeView.push(viewId),
|
|
48
|
+
emitPageSelectorSelect: page => emitted.page.push(page),
|
|
49
|
+
emitPillSelect: item => emitted.pill.push(item),
|
|
50
|
+
})
|
|
51
|
+
|
|
52
|
+
return { navigation, emitted, activeView, pillNav, generatedPillNav }
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
describe('usePluginWorkspaceNavigation', () => {
|
|
56
|
+
it('derives the initial view from explicit pill nav and syncs pill selection', () => {
|
|
57
|
+
const data = { id: 'data', label: 'Data' }
|
|
58
|
+
const results = { id: 'results', label: 'Results' }
|
|
59
|
+
const { navigation, emitted } = makeNavigation({
|
|
60
|
+
pillNav: [data, results],
|
|
61
|
+
panelIds: ['fallback'],
|
|
62
|
+
})
|
|
63
|
+
|
|
64
|
+
expect(navigation.resolvedPillNav.value).toEqual([data, results])
|
|
65
|
+
expect(navigation.resolvedActiveView.value).toBe('data')
|
|
66
|
+
|
|
67
|
+
navigation.handlePillSelect(results)
|
|
68
|
+
|
|
69
|
+
expect(navigation.resolvedActiveView.value).toBe('results')
|
|
70
|
+
expect(emitted.activeView).toEqual(['results'])
|
|
71
|
+
expect(emitted.pill).toEqual([results])
|
|
72
|
+
})
|
|
73
|
+
|
|
74
|
+
it('emits page selector changes while keeping controlled active view stable', () => {
|
|
75
|
+
const overview = { id: 'overview', label: 'Overview' }
|
|
76
|
+
const { navigation, emitted } = makeNavigation({
|
|
77
|
+
activeView: 'controlled',
|
|
78
|
+
pageSelector: [overview],
|
|
79
|
+
generatedDefaultView: 'generated',
|
|
80
|
+
})
|
|
81
|
+
|
|
82
|
+
navigation.handlePageSelectorSelect(overview)
|
|
83
|
+
|
|
84
|
+
expect(navigation.resolvedActiveView.value).toBe('controlled')
|
|
85
|
+
expect(navigation.resolvedCurrentPageSelectorId.value).toBe('controlled')
|
|
86
|
+
expect(emitted.activeView).toEqual(['overview'])
|
|
87
|
+
expect(emitted.page).toEqual([overview])
|
|
88
|
+
})
|
|
89
|
+
|
|
90
|
+
it('uses generated navigation when explicit pill nav is absent', () => {
|
|
91
|
+
const generated = { id: 'design', label: 'Design' }
|
|
92
|
+
const { navigation } = makeNavigation({
|
|
93
|
+
generatedPillNav: [generated],
|
|
94
|
+
generatedDefaultView: 'design',
|
|
95
|
+
pageSelector: ['page-fallback'],
|
|
96
|
+
})
|
|
97
|
+
|
|
98
|
+
expect(navigation.resolvedPillNav.value).toEqual([generated])
|
|
99
|
+
expect(navigation.firstConfiguredViewId.value).toBe('design')
|
|
100
|
+
expect(navigation.resolvedCurrentPillId.value).toBe('design')
|
|
101
|
+
})
|
|
102
|
+
})
|
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
import { describe, expect, it } from 'vitest'
|
|
2
|
+
import {
|
|
3
|
+
getPluginWorkspaceSidebarSlotNames,
|
|
4
|
+
getPluginWorkspaceTopBarSlotNames,
|
|
5
|
+
hasPluginWorkspaceSidebarSurface,
|
|
6
|
+
} from '../../components/PluginWorkspaceView.shell'
|
|
7
|
+
|
|
8
|
+
describe('PluginWorkspaceView shell helpers', () => {
|
|
9
|
+
it('selects forwarded topbar and sidebar slots by key without depending on Vue slot return types', () => {
|
|
10
|
+
const slots: Readonly<Record<string, unknown>> = {
|
|
11
|
+
default: undefined,
|
|
12
|
+
actions: undefined,
|
|
13
|
+
'settings-tab-analysis': undefined,
|
|
14
|
+
header: undefined,
|
|
15
|
+
'section-import': undefined,
|
|
16
|
+
unrelated: undefined,
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
expect(getPluginWorkspaceTopBarSlotNames(slots)).toEqual([
|
|
20
|
+
'actions',
|
|
21
|
+
'settings-tab-analysis',
|
|
22
|
+
])
|
|
23
|
+
expect(getPluginWorkspaceSidebarSlotNames(slots)).toEqual([
|
|
24
|
+
'header',
|
|
25
|
+
'section-import',
|
|
26
|
+
])
|
|
27
|
+
})
|
|
28
|
+
|
|
29
|
+
it('keeps the sidebar surface hidden when all sidebar inputs are empty', () => {
|
|
30
|
+
expect(hasPluginWorkspaceSidebarSurface({
|
|
31
|
+
showSidebar: true,
|
|
32
|
+
panels: {},
|
|
33
|
+
forms: {},
|
|
34
|
+
model: undefined,
|
|
35
|
+
controls: undefined,
|
|
36
|
+
showSidebarWhenEmpty: false,
|
|
37
|
+
hasSidebarSlot: false,
|
|
38
|
+
sidebarSlotCount: 0,
|
|
39
|
+
})).toBe(false)
|
|
40
|
+
})
|
|
41
|
+
})
|
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
import { describe, expect, it } from 'vitest'
|
|
2
|
+
|
|
3
|
+
import {
|
|
4
|
+
PROTOCOL_STEP_TYPE_COLORS,
|
|
5
|
+
PROTOCOL_STEP_TYPE_ICONS,
|
|
6
|
+
formatProtocolDuration,
|
|
7
|
+
} from '../../components/ProtocolStep.presentation'
|
|
8
|
+
|
|
9
|
+
describe('ProtocolStep presentation helpers', () => {
|
|
10
|
+
it('formats protocol durations for editor and timeline displays', () => {
|
|
11
|
+
expect(formatProtocolDuration(undefined)).toBe('')
|
|
12
|
+
expect(formatProtocolDuration(15)).toBe('15m')
|
|
13
|
+
expect(formatProtocolDuration(60)).toBe('1h')
|
|
14
|
+
expect(formatProtocolDuration(95)).toBe('1h 35m')
|
|
15
|
+
})
|
|
16
|
+
|
|
17
|
+
it('keeps icons and colors available for every protocol step type', () => {
|
|
18
|
+
expect(Object.keys(PROTOCOL_STEP_TYPE_ICONS).sort()).toEqual([
|
|
19
|
+
'addition',
|
|
20
|
+
'centrifuge',
|
|
21
|
+
'custom',
|
|
22
|
+
'incubation',
|
|
23
|
+
'measurement',
|
|
24
|
+
'mix',
|
|
25
|
+
'transfer',
|
|
26
|
+
'wash',
|
|
27
|
+
])
|
|
28
|
+
expect(PROTOCOL_STEP_TYPE_COLORS.measurement).toBe('#8B5CF6')
|
|
29
|
+
expect(PROTOCOL_STEP_TYPE_ICONS.custom).toContain('M10.325')
|
|
30
|
+
})
|
|
31
|
+
})
|
|
@@ -0,0 +1,165 @@
|
|
|
1
|
+
import { describe, expect, it } from 'vitest'
|
|
2
|
+
import type { ProtocolStep } from '../../types'
|
|
3
|
+
import type { StepTemplate } from '../../composables/useProtocolTemplates'
|
|
4
|
+
import {
|
|
5
|
+
buildCustomProtocolTemplate,
|
|
6
|
+
buildProtocolPreviewParams,
|
|
7
|
+
buildProtocolStepEditorState,
|
|
8
|
+
buildProtocolStepPreview,
|
|
9
|
+
buildProtocolTemplateDefaults,
|
|
10
|
+
} from '../../components/ProtocolStepEditor.state'
|
|
11
|
+
|
|
12
|
+
const template: StepTemplate = {
|
|
13
|
+
id: 'builtin-incubation',
|
|
14
|
+
type: 'incubation',
|
|
15
|
+
name: 'Incubation',
|
|
16
|
+
description: 'Hold samples under controlled conditions',
|
|
17
|
+
defaultDuration: 60,
|
|
18
|
+
parameters: [
|
|
19
|
+
{
|
|
20
|
+
key: 'temperature',
|
|
21
|
+
label: 'Temperature',
|
|
22
|
+
type: 'temperature',
|
|
23
|
+
unit: 'C',
|
|
24
|
+
default: 37,
|
|
25
|
+
required: true,
|
|
26
|
+
},
|
|
27
|
+
{
|
|
28
|
+
key: 'notes',
|
|
29
|
+
label: 'Notes',
|
|
30
|
+
type: 'text',
|
|
31
|
+
},
|
|
32
|
+
],
|
|
33
|
+
isBuiltIn: true,
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
describe('ProtocolStepEditor state helpers', () => {
|
|
37
|
+
it('builds editor defaults from template parameter defaults', () => {
|
|
38
|
+
expect(buildProtocolTemplateDefaults(template)).toEqual({
|
|
39
|
+
name: 'Incubation',
|
|
40
|
+
description: 'Hold samples under controlled conditions',
|
|
41
|
+
duration: 60,
|
|
42
|
+
parameters: {
|
|
43
|
+
temperature: 37,
|
|
44
|
+
},
|
|
45
|
+
})
|
|
46
|
+
})
|
|
47
|
+
|
|
48
|
+
it('initializes editor fields from an existing step', () => {
|
|
49
|
+
const step: ProtocolStep = {
|
|
50
|
+
id: 'step-1',
|
|
51
|
+
type: 'incubation',
|
|
52
|
+
name: 'Overnight incubation',
|
|
53
|
+
description: 'Long run',
|
|
54
|
+
duration: 720,
|
|
55
|
+
status: 'in_progress',
|
|
56
|
+
parameters: {
|
|
57
|
+
temperature: 4,
|
|
58
|
+
},
|
|
59
|
+
order: 3,
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
const state = buildProtocolStepEditorState(step, (type) => (
|
|
63
|
+
type === 'incubation' ? template : undefined
|
|
64
|
+
))
|
|
65
|
+
|
|
66
|
+
expect(state).toEqual({
|
|
67
|
+
selectedTemplateId: 'builtin-incubation',
|
|
68
|
+
fields: {
|
|
69
|
+
name: 'Overnight incubation',
|
|
70
|
+
description: 'Long run',
|
|
71
|
+
duration: 720,
|
|
72
|
+
parameters: {
|
|
73
|
+
temperature: 4,
|
|
74
|
+
},
|
|
75
|
+
},
|
|
76
|
+
})
|
|
77
|
+
})
|
|
78
|
+
|
|
79
|
+
it('preserves source step metadata when building a preview', () => {
|
|
80
|
+
const preview = buildProtocolStepPreview(
|
|
81
|
+
template,
|
|
82
|
+
{
|
|
83
|
+
name: '',
|
|
84
|
+
description: 'Preview description',
|
|
85
|
+
parameters: {
|
|
86
|
+
temperature: 25,
|
|
87
|
+
},
|
|
88
|
+
},
|
|
89
|
+
{
|
|
90
|
+
id: 'step-2',
|
|
91
|
+
type: 'incubation',
|
|
92
|
+
name: 'Source',
|
|
93
|
+
status: 'completed',
|
|
94
|
+
order: 5,
|
|
95
|
+
}
|
|
96
|
+
)
|
|
97
|
+
|
|
98
|
+
expect(preview).toEqual({
|
|
99
|
+
id: 'step-2',
|
|
100
|
+
type: 'incubation',
|
|
101
|
+
name: 'Incubation',
|
|
102
|
+
description: 'Preview description',
|
|
103
|
+
duration: 60,
|
|
104
|
+
status: 'completed',
|
|
105
|
+
parameters: {
|
|
106
|
+
temperature: 25,
|
|
107
|
+
},
|
|
108
|
+
order: 5,
|
|
109
|
+
})
|
|
110
|
+
})
|
|
111
|
+
|
|
112
|
+
it('formats only populated preview parameters', () => {
|
|
113
|
+
const formatted = buildProtocolPreviewParams(
|
|
114
|
+
template,
|
|
115
|
+
{
|
|
116
|
+
temperature: 37,
|
|
117
|
+
notes: '',
|
|
118
|
+
},
|
|
119
|
+
(value, parameter) => `${parameter.label}: ${value}`
|
|
120
|
+
)
|
|
121
|
+
|
|
122
|
+
expect(formatted).toBe('Temperature: 37')
|
|
123
|
+
})
|
|
124
|
+
|
|
125
|
+
it('creates a custom template from the current editor fields', () => {
|
|
126
|
+
expect(
|
|
127
|
+
buildCustomProtocolTemplate(
|
|
128
|
+
template,
|
|
129
|
+
{
|
|
130
|
+
name: 'Cold incubation',
|
|
131
|
+
description: 'Cold room workflow',
|
|
132
|
+
duration: 120,
|
|
133
|
+
parameters: {
|
|
134
|
+
temperature: 4,
|
|
135
|
+
notes: 'keep covered',
|
|
136
|
+
},
|
|
137
|
+
},
|
|
138
|
+
'custom-fixed'
|
|
139
|
+
)
|
|
140
|
+
).toEqual({
|
|
141
|
+
id: 'custom-fixed',
|
|
142
|
+
type: 'incubation',
|
|
143
|
+
name: 'Cold incubation',
|
|
144
|
+
description: 'Cold room workflow',
|
|
145
|
+
defaultDuration: 120,
|
|
146
|
+
parameters: [
|
|
147
|
+
{
|
|
148
|
+
key: 'temperature',
|
|
149
|
+
label: 'Temperature',
|
|
150
|
+
type: 'temperature',
|
|
151
|
+
unit: 'C',
|
|
152
|
+
default: 4,
|
|
153
|
+
required: true,
|
|
154
|
+
},
|
|
155
|
+
{
|
|
156
|
+
key: 'notes',
|
|
157
|
+
label: 'Notes',
|
|
158
|
+
type: 'text',
|
|
159
|
+
default: 'keep covered',
|
|
160
|
+
},
|
|
161
|
+
],
|
|
162
|
+
isBuiltIn: false,
|
|
163
|
+
})
|
|
164
|
+
})
|
|
165
|
+
})
|