From a2e4bdd2acb29a5e4bd00fd5a4ba5e760c073b64 Mon Sep 17 00:00:00 2001 From: Jethary Alcid <66035149+jerader@users.noreply.github.com> Date: Wed, 18 Dec 2024 13:59:56 -0500 Subject: [PATCH] feat(protocol-designer, components): add dropdown field deck highlights (#17122) closes AUTH-1124 --- .../BaseDeck/WasteChuteFixture.tsx | 85 +++-- .../BaseDeck/WasteChuteStagingAreaFixture.tsx | 7 + .../src/hardware-sim/Deck/DeckFromLayers.tsx | 2 + .../src/hardware-sim/Deck/FlexTrash.tsx | 19 + .../src/hardware-sim/Deck/OT2Layers.tsx | 13 +- .../DeckLabel/__tests__/DeckLabel.test.tsx | 2 +- components/src/molecules/DeckLabel/index.tsx | 6 +- .../src/molecules/DropdownMenu/index.tsx | 6 + .../__tests__/DeckLabelSet.test.tsx | 2 + .../src/organisms/DeckLabelSet/index.tsx | 29 +- .../assets/localization/en/application.json | 5 + .../molecules/DropdownStepFormField/index.tsx | 49 ++- .../organisms/DesignerNavigation/index.tsx | 11 +- .../Designer/DeckSetup/DeckItemHighlight.tsx | 94 +++++ .../Designer/DeckSetup/DeckSetupDetails.tsx | 28 +- .../Designer/DeckSetup/FixtureRender.tsx | 55 ++- .../Designer/DeckSetup/HighlightItems.tsx | 326 ++++++++++++++++++ .../pages/Designer/DeckSetup/HoveredItems.tsx | 1 + .../pages/Designer/DeckSetup/ModuleLabel.tsx | 37 +- .../DeckSetup/SelectedHoveredItems.tsx | 4 + .../__tests__/DeckSetupContainer.test.tsx | 13 +- .../src/pages/Designer/HighlightLabware.tsx | 5 + .../src/pages/Designer/LabwareLabel.tsx | 26 +- .../Designer/Offdeck/HighlightOffdeckSlot.tsx | 76 ++++ .../pages/Designer/Offdeck/OffDeckDetails.tsx | 41 ++- .../Offdeck/__tests__/OffDeckDetails.test.tsx | 16 +- .../StepForm/PipetteFields/LabwareField.tsx | 12 +- .../StepForm/StepFormToolbox.tsx | 17 +- .../StepTools/HeaterShakerTools/index.tsx | 11 +- .../StepForm/StepTools/MagnetTools/index.tsx | 73 +--- .../MoveLabwareTools/LabwareLocationField.tsx | 15 +- .../MoveLabwareTools/MoveLabwareField.tsx | 12 +- .../StepTools/TemperatureTools/index.tsx | 11 +- .../StepTools/ThermocyclerTools/index.tsx | 4 +- .../StepTools/__tests__/MagnetTools.test.tsx | 3 - .../src/pages/Designer/index.tsx | 11 +- .../test/createPresavedStepForm.test.ts | 1 + .../utils/createPresavedStepForm.ts | 79 ++++- .../steps/actions/__tests__/actions.test.ts | 34 +- .../__tests__/addAndSelectStep.test.ts | 316 +++++++++++++++++ .../src/ui/steps/actions/actions.ts | 127 ++++++- .../src/ui/steps/actions/thunks/index.ts | 121 ++++++- .../src/ui/steps/actions/types.ts | 18 + protocol-designer/src/ui/steps/reducers.ts | 49 +++ protocol-designer/src/ui/steps/selectors.ts | 9 + shared-data/js/fixtures.ts | 4 +- 46 files changed, 1653 insertions(+), 232 deletions(-) create mode 100644 protocol-designer/src/pages/Designer/DeckSetup/DeckItemHighlight.tsx create mode 100644 protocol-designer/src/pages/Designer/DeckSetup/HighlightItems.tsx create mode 100644 protocol-designer/src/pages/Designer/Offdeck/HighlightOffdeckSlot.tsx diff --git a/components/src/hardware-sim/BaseDeck/WasteChuteFixture.tsx b/components/src/hardware-sim/BaseDeck/WasteChuteFixture.tsx index 2253e0f8726..cc977a4e8b6 100644 --- a/components/src/hardware-sim/BaseDeck/WasteChuteFixture.tsx +++ b/components/src/hardware-sim/BaseDeck/WasteChuteFixture.tsx @@ -8,6 +8,7 @@ import { JUSTIFY_CENTER, TEXT_ALIGN_CENTER, } from '../../styles' +import { DeckLabelSet } from '../../organisms' import { SPACING, TYPOGRAPHY } from '../../ui-style-constants' import { COLORS } from '../../helix-design-system' import { RobotCoordsForeignObject } from '../Deck/RobotCoordsForeignObject' @@ -17,7 +18,13 @@ import type { DeckDefinition, ModuleType, } from '@opentrons/shared-data' +import type { DeckLabelProps } from '../../molecules' +const WASTE_CHUTE_WIDTH = 130 +const WASTE_CHUTE_HEIGHT = 138 +const WASTE_CHUTE_X = 322 +const WASTE_CHUTE_Y = -51 +const TAG_HEIGHT = 28 interface WasteChuteFixtureProps extends React.SVGProps { cutoutId: typeof WASTE_CHUTE_CUTOUT deckDefinition: DeckDefinition @@ -25,6 +32,10 @@ interface WasteChuteFixtureProps extends React.SVGProps { fixtureBaseColor?: React.SVGProps['fill'] wasteChuteColor?: string showExtensions?: boolean + /** optional prop to highlight the border of the wasteChute */ + showHighlight?: boolean + /** optional tag info to display a tag below the waste */ + tagInfo?: DeckLabelProps[] } export function WasteChuteFixture( @@ -35,6 +46,8 @@ export function WasteChuteFixture( deckDefinition, fixtureBaseColor = COLORS.grey35, wasteChuteColor = COLORS.grey50, + showHighlight, + tagInfo, ...restProps } = props @@ -64,6 +77,8 @@ export function WasteChuteFixture( ) @@ -72,43 +87,57 @@ export function WasteChuteFixture( interface WasteChuteProps { wasteIconColor: string backgroundColor: string + showHighlight?: boolean + tagInfo?: DeckLabelProps[] } /** * a deck map foreign object representing the physical location of the waste chute connected to the deck */ export function WasteChute(props: WasteChuteProps): JSX.Element { - const { wasteIconColor, backgroundColor } = props + const { wasteIconColor, backgroundColor, showHighlight, tagInfo } = props return ( - - + - - - Waste chute - - - + + + Waste chute + + + + {tagInfo != null && tagInfo.length > 0 ? ( + + ) : null} + ) } diff --git a/components/src/hardware-sim/BaseDeck/WasteChuteStagingAreaFixture.tsx b/components/src/hardware-sim/BaseDeck/WasteChuteStagingAreaFixture.tsx index 0034439ce12..17539777257 100644 --- a/components/src/hardware-sim/BaseDeck/WasteChuteStagingAreaFixture.tsx +++ b/components/src/hardware-sim/BaseDeck/WasteChuteStagingAreaFixture.tsx @@ -8,6 +8,7 @@ import { SlotClip } from './SlotClip' import { WasteChute } from './WasteChuteFixture' import type { DeckDefinition, ModuleType } from '@opentrons/shared-data' +import type { DeckLabelProps } from '../../molecules' interface WasteChuteStagingAreaFixtureProps extends React.SVGProps { @@ -18,6 +19,8 @@ interface WasteChuteStagingAreaFixtureProps slotClipColor?: React.SVGProps['stroke'] wasteChuteColor?: string showExtensions?: boolean + showHighlight?: boolean + tagInfo?: DeckLabelProps[] } export function WasteChuteStagingAreaFixture( @@ -29,6 +32,8 @@ export function WasteChuteStagingAreaFixture( fixtureBaseColor = COLORS.grey35, slotClipColor = COLORS.grey60, wasteChuteColor = COLORS.grey50, + showHighlight, + tagInfo, ...restProps } = props @@ -62,6 +67,8 @@ export function WasteChuteStagingAreaFixture( ) diff --git a/components/src/hardware-sim/Deck/DeckFromLayers.tsx b/components/src/hardware-sim/Deck/DeckFromLayers.tsx index badd7e80ca1..14d5b5337f4 100644 --- a/components/src/hardware-sim/Deck/DeckFromLayers.tsx +++ b/components/src/hardware-sim/Deck/DeckFromLayers.tsx @@ -15,6 +15,8 @@ import { ALL_OT2_DECK_LAYERS } from './constants' import type { RobotType } from '@opentrons/shared-data' +export * from './OT2Layers' + export interface DeckFromLayersProps { robotType: RobotType layerBlocklist: string[] diff --git a/components/src/hardware-sim/Deck/FlexTrash.tsx b/components/src/hardware-sim/Deck/FlexTrash.tsx index 9b3c4c9fef9..00ae3fb2a1b 100644 --- a/components/src/hardware-sim/Deck/FlexTrash.tsx +++ b/components/src/hardware-sim/Deck/FlexTrash.tsx @@ -4,6 +4,7 @@ import { opentrons1Trash3200MlFixedV1 as trashLabwareDef, } from '@opentrons/shared-data' import { Icon } from '../../icons' +import { DeckLabelSet } from '../../organisms' import { Flex, Text } from '../../primitives' import { ALIGN_CENTER, JUSTIFY_CENTER } from '../../styles' import { SPACING, TYPOGRAPHY } from '../../ui-style-constants' @@ -11,6 +12,7 @@ import { COLORS, BORDERS } from '../../helix-design-system' import { RobotCoordsForeignObject } from './RobotCoordsForeignObject' import type { RobotType } from '@opentrons/shared-data' +import type { DeckLabelProps } from '../../molecules' // only allow edge cutout locations (columns 1 and 3) export type TrashCutoutId = @@ -23,11 +25,16 @@ export type TrashCutoutId = | 'cutoutC3' | 'cutoutD3' +const HEIGHT_OF_TAG = 28 interface FlexTrashProps { robotType: RobotType trashIconColor: string backgroundColor: string trashCutoutId?: TrashCutoutId + /** optional prop to highlight the border of the trashBin */ + showHighlight?: boolean + /** optional tag info to display a tag below the trash */ + tagInfo?: DeckLabelProps[] } /** @@ -40,6 +47,8 @@ export const FlexTrash = ({ trashIconColor, backgroundColor, trashCutoutId, + showHighlight, + tagInfo, }: FlexTrashProps): JSX.Element | null => { // be sure we don't try to render for an OT-2 if (robotType !== FLEX_ROBOT_TYPE) return null @@ -96,6 +105,7 @@ export const FlexTrash = ({ justifyContent={JUSTIFY_CENTER} gridGap={SPACING.spacing8} width="100%" + border={showHighlight ? `3px solid ${COLORS.blue50}` : 'none'} > {rotateDegrees === '180' ? ( + {tagInfo != null && tagInfo.length > 0 ? ( + + ) : null} ) : null } diff --git a/components/src/hardware-sim/Deck/OT2Layers.tsx b/components/src/hardware-sim/Deck/OT2Layers.tsx index fe103ded268..cc5b3143aa1 100644 --- a/components/src/hardware-sim/Deck/OT2Layers.tsx +++ b/components/src/hardware-sim/Deck/OT2Layers.tsx @@ -1,3 +1,5 @@ +import { COLORS } from '../../helix-design-system' + export function FixedBase(): JSX.Element { return ( @@ -14,15 +16,20 @@ export function FixedBase(): JSX.Element { ) } -export function FixedTrash(): JSX.Element { +interface FixedTrashProps { + highlight?: boolean +} +export function FixedTrash(props: FixedTrashProps): JSX.Element { + const { highlight = false } = props return ( diff --git a/components/src/molecules/DeckLabel/__tests__/DeckLabel.test.tsx b/components/src/molecules/DeckLabel/__tests__/DeckLabel.test.tsx index 088536e706a..f041585b1cd 100644 --- a/components/src/molecules/DeckLabel/__tests__/DeckLabel.test.tsx +++ b/components/src/molecules/DeckLabel/__tests__/DeckLabel.test.tsx @@ -1,4 +1,3 @@ -import type * as React from 'react' import { screen } from '@testing-library/react' import { describe, it, beforeEach, expect } from 'vitest' @@ -20,6 +19,7 @@ describe('DeckLabel', () => { text: 'mock DeckLabel text', isSelected: false, isLast: true, + isZoomed: true, } }) diff --git a/components/src/molecules/DeckLabel/index.tsx b/components/src/molecules/DeckLabel/index.tsx index d3a8c02b975..67be739c7ab 100644 --- a/components/src/molecules/DeckLabel/index.tsx +++ b/components/src/molecules/DeckLabel/index.tsx @@ -11,6 +11,7 @@ import type { FlattenSimpleInterpolation } from 'styled-components' import type { ModuleModel } from '@opentrons/shared-data' export interface DeckLabelProps { + isZoomed: boolean text: string isSelected: boolean moduleModel?: ModuleModel @@ -26,6 +27,7 @@ export function DeckLabel({ moduleModel, maxWidth = FLEX_MAX_CONTENT, isLast = false, + isZoomed, }: DeckLabelProps): JSX.Element { const DECK_LABEL_BASE_STYLE = ( labelBorderRadius?: string @@ -59,7 +61,7 @@ export function DeckLabel({ return ( - {moduleModel != null ? ( + {moduleModel != null && isZoomed ? ( ) : null} diff --git a/components/src/molecules/DropdownMenu/index.tsx b/components/src/molecules/DropdownMenu/index.tsx index 29dfe03988d..2e2f2d8b8ca 100644 --- a/components/src/molecules/DropdownMenu/index.tsx +++ b/components/src/molecules/DropdownMenu/index.tsx @@ -65,6 +65,8 @@ export interface DropdownMenuProps { disabled?: boolean /** optional placement of the menu */ menuPlacement?: 'auto' | 'top' | 'bottom' + onEnter?: (id: string) => void + onExit?: () => void } // TODO: (smb: 4/15/22) refactor this to use html select for accessibility @@ -84,6 +86,8 @@ export function DropdownMenu(props: DropdownMenuProps): JSX.Element { disabled = false, onFocus, onBlur, + onEnter, + onExit, menuPlacement = 'auto', } = props const [targetProps, tooltipProps] = useHoverTooltip() @@ -290,6 +294,8 @@ export function DropdownMenu(props: DropdownMenuProps): JSX.Element { setShowDropdownMenu(false) }} border="none" + onMouseEnter={() => onEnter?.(option.value)} + onMouseLeave={onExit} > ): JSX.Element => { - const { deckLabels, x, y, width, height } = props + const { deckLabels, x, y, width, height, invert = false } = props return ( - - + + 0 ? deckLabels[0].isZoomed : true} + /> {deckLabels.length > 0 ? deckLabels.map((deckLabel, index) => ( @@ -46,9 +60,14 @@ export const DeckLabelSet = React.forwardRef( DeckLabelSetComponent ) -const StyledBox = styled(Box)` +interface StyledBoxProps { + isZoomed: boolean +} + +const StyledBox = styled(Box)` border-radius: ${BORDERS.borderRadius4}; - border: 1.5px solid ${COLORS.blue50}; + border: ${({ isZoomed }) => + isZoomed ? `1.5px solid ${COLORS.blue50}` : `3px solid ${COLORS.blue50}`}; ` const LabelContainer = styled.div` diff --git a/protocol-designer/src/assets/localization/en/application.json b/protocol-designer/src/assets/localization/en/application.json index 0cbdb9cc6d3..3692c9a1fac 100644 --- a/protocol-designer/src/assets/localization/en/application.json +++ b/protocol-designer/src/assets/localization/en/application.json @@ -5,12 +5,14 @@ "cancel": "cancel", "date_created": "Date Created", "description": "Description", + "dest": "Destination", "edit": "edit", "exit_batch_edit": "exit batch edit", "go_back": "Go back", "information": "Information", "labware": "labware", "last_exported": "Last Exported", + "location": "Location", "magnet_height_caption": "Must be between {{low}} to {{high}}.", "magnet_recommended": "The recommended height is {{default}}", "manually": "Manually", @@ -42,6 +44,9 @@ "temperature": "temperature", "thermocycler": "thermocycler" }, + "select": "Select", + "selected": "Selected", + "source": "Source", "temperature": "Temperature (˚C)", "time": "Time", "units": { diff --git a/protocol-designer/src/molecules/DropdownStepFormField/index.tsx b/protocol-designer/src/molecules/DropdownStepFormField/index.tsx index 01f82972d60..75334fe830a 100644 --- a/protocol-designer/src/molecules/DropdownStepFormField/index.tsx +++ b/protocol-designer/src/molecules/DropdownStepFormField/index.tsx @@ -1,5 +1,5 @@ import { useTranslation } from 'react-i18next' -import { useEffect } from 'react' +import { useDispatch } from 'react-redux' import { COLORS, DIRECTION_COLUMN, @@ -9,6 +9,7 @@ import { SPACING, StyledText, } from '@opentrons/components' +import { selectDropdownItem } from '../../ui/steps/actions/actions' import type { Options } from '@opentrons/components' import type { FieldProps } from '../../pages/Designer/ProtocolSteps/StepForm/types' @@ -16,8 +17,13 @@ export interface DropdownStepFormFieldProps extends FieldProps { options: Options title: string width?: string + onEnter?: (id: string) => void + onExit?: () => void } +const FIRST_FIELDS = ['aspirate_labware', 'labware', 'moduleId'] +const SECOND_FIELDS = ['dispense_labware', 'newLocation'] + export function DropdownStepFormField( props: DropdownStepFormFieldProps ): JSX.Element { @@ -31,16 +37,44 @@ export function DropdownStepFormField( padding = `0 ${SPACING.spacing16}`, width = '17.5rem', onFieldFocus, + onEnter, + onExit, onFieldBlur, + name: fieldName, } = props - const { t } = useTranslation('tooltip') + const { t } = useTranslation(['tooltip', 'application']) + const dispatch = useDispatch() const availableOptionId = options.find(opt => opt.value === value) + const handleSelection = (value: string): void => { + let text = t('application:selected') + if (fieldName === 'newLocation') { + text = t('application:location') + } else if (fieldName === 'aspirate_labware') { + text = t('application:source') + } else if (fieldName === 'dispense_labware') { + text = t('application:dest') + } - useEffect(() => { - if (options.length === 1) { - updateValue(options[0].value) + const selection = { + id: value, + text, + } + if (FIRST_FIELDS.includes(fieldName)) { + dispatch( + selectDropdownItem({ + selection: { ...selection, field: '1' }, + mode: 'add', + }) + ) + } else if (SECOND_FIELDS.includes(fieldName)) { + dispatch( + selectDropdownItem({ + selection: { ...selection, field: '2' }, + mode: 'add', + }) + ) } - }, []) + } return ( @@ -59,7 +93,10 @@ export function DropdownStepFormField( } onClick={value => { updateValue(value) + handleSelection(value) }} + onEnter={onEnter} + onExit={onExit} /> ) : ( selected.id === itemId && selected.field === '2' + ) + + if ( + tab === 'startingDeck' || + slotPosition === null || + (!isHovered && !isSelected) + ) { + return null + } + + return ( + <> + + + + ) +} diff --git a/protocol-designer/src/pages/Designer/DeckSetup/DeckSetupDetails.tsx b/protocol-designer/src/pages/Designer/DeckSetup/DeckSetupDetails.tsx index 53a61df3faf..110bf9535a2 100644 --- a/protocol-designer/src/pages/Designer/DeckSetup/DeckSetupDetails.tsx +++ b/protocol-designer/src/pages/Designer/DeckSetup/DeckSetupDetails.tsx @@ -29,6 +29,7 @@ import { HoveredItems } from './HoveredItems' import { SelectedHoveredItems } from './SelectedHoveredItems' import { getAdjacentLabware } from './utils' import { SlotWarning } from './SlotWarning' +import { HighlightItems } from './HighlightItems' import type { ComponentProps, Dispatch, SetStateAction } from 'react' import type { ThermocyclerVizProps } from '@opentrons/components' @@ -258,17 +259,19 @@ export function DeckSetupDetails(props: DeckSetupDetailsProps): JSX.Element { ) : null} {labwareLoadedOnModule == null ? ( - + <> + + ) : null} @@ -431,6 +434,9 @@ export function DeckSetupDetails(props: DeckSetupDetailsProps): JSX.Element { ) })} + {/* highlight items from Protocol steps */} + + {/* selected hardware + labware */} { - const { fixture, cutout, deckDef, robotType } = props + const { fixture, cutout, deckDef, robotType, showHighlight, tagInfo } = props const deckSetup = useSelector(getInitialDeckSetup) const { labware } = deckSetup const adjacentLabware = getAdjacentLabware(fixture, cutout, labware) @@ -61,22 +68,28 @@ export const FixtureRender = (props: FixtureRenderProps): JSX.Element => { ) } case 'trashBin': { - return ( - - - - - ) + if (robotType === OT2_ROBOT_TYPE && showHighlight) { + return + } else { + return ( + + + + + ) + } } case 'wasteChute': { return ( @@ -85,6 +98,8 @@ export const FixtureRender = (props: FixtureRenderProps): JSX.Element => { cutoutId={cutout as typeof WASTE_CHUTE_CUTOUT} deckDefinition={deckDef} fixtureBaseColor={lightFill} + showHighlight={showHighlight} + tagInfo={tagInfo} /> ) } @@ -95,6 +110,8 @@ export const FixtureRender = (props: FixtureRenderProps): JSX.Element => { cutoutId={cutout as typeof WASTE_CHUTE_CUTOUT} deckDefinition={deckDef} fixtureBaseColor={lightFill} + showHighlight={showHighlight} + tagInfo={tagInfo} /> {renderLabwareOnDeck()} diff --git a/protocol-designer/src/pages/Designer/DeckSetup/HighlightItems.tsx b/protocol-designer/src/pages/Designer/DeckSetup/HighlightItems.tsx new file mode 100644 index 00000000000..fd548de360b --- /dev/null +++ b/protocol-designer/src/pages/Designer/DeckSetup/HighlightItems.tsx @@ -0,0 +1,326 @@ +import { useSelector } from 'react-redux' +import { useTranslation } from 'react-i18next' + +import { + STANDARD_FLEX_SLOTS, + STANDARD_OT2_SLOTS, + THERMOCYCLER_MODULE_TYPE, + WASTE_CHUTE_CUTOUT, + getAddressableAreaFromSlotId, + getPositionFromSlotId, + inferModuleOrientationFromXCoordinate, +} from '@opentrons/shared-data' +import { getDeckSetupForActiveItem } from '../../../top-selectors/labware-locations' +import { + getHoveredDropdownItem, + getSelectedDropdownItem, +} from '../../../ui/steps/selectors' +import { getDesignerTab } from '../../../file-data/selectors' +import { LabwareLabel } from '../LabwareLabel' +import { ModuleLabel } from './ModuleLabel' +import { FixtureRender } from './FixtureRender' +import { DeckItemHighlight } from './DeckItemHighlight' +import type { AdditionalEquipmentName } from '@opentrons/step-generation' +import type { + RobotType, + DeckDefinition, + CutoutId, + AddressableAreaName, +} from '@opentrons/shared-data' +import type { LabwareOnDeck, ModuleOnDeck } from '../../../step-forms' +import type { Fixture } from './constants' + +interface HighlightItemsProps { + deckDef: DeckDefinition + robotType: RobotType +} + +const SLOTS = [ + ...STANDARD_FLEX_SLOTS, + ...STANDARD_OT2_SLOTS, + 'A4', + 'B4', + 'C4', + 'D4', + 'cutoutD3', +] + +export function HighlightItems(props: HighlightItemsProps): JSX.Element | null { + const { robotType, deckDef } = props + const { t } = useTranslation('application') + const tab = useSelector(getDesignerTab) + const { labware, modules, additionalEquipmentOnDeck } = useSelector( + getDeckSetupForActiveItem + ) + const hoveredItem = useSelector(getHoveredDropdownItem) + const selectedDropdownItems = useSelector(getSelectedDropdownItem) + + if ( + hoveredItem == null && + (selectedDropdownItems == null || selectedDropdownItems.length === 0) + ) { + return null + } + + const hoveredItemLabware: LabwareOnDeck | null = + hoveredItem?.id != null && labware[hoveredItem.id] != null + ? labware[hoveredItem.id] + : null + const selectedItemLabwares = selectedDropdownItems.filter( + selected => selected.id != null && labware[selected.id] + ) + const hoveredItemModule: ModuleOnDeck | null = + hoveredItem?.id != null && modules[hoveredItem.id] != null + ? modules[hoveredItem.id] + : null + const selectedItemModule = selectedDropdownItems.find( + selected => selected.id != null && modules[selected.id] + ) + const hoveredItemTrash: { + name: AdditionalEquipmentName + id: string + location?: string | undefined + } | null = + hoveredItem?.id != null && additionalEquipmentOnDeck[hoveredItem.id] != null + ? additionalEquipmentOnDeck[hoveredItem.id] + : null + const selectedItemTrash = selectedDropdownItems.find( + selected => selected.id != null && additionalEquipmentOnDeck[selected.id] + ) + + const hoveredDeckItem: string | null = + hoveredItem?.id != null && + SLOTS.includes(hoveredItem.id as AddressableAreaName) + ? hoveredItem.id + : null + const selectedItemSlot = selectedDropdownItems.find( + selected => + selected.id != null && SLOTS.includes(selected.id as AddressableAreaName) + ) + + const getLabwareItems = (): JSX.Element[] => { + const items: JSX.Element[] = [] + + if (hoveredItemLabware != null || selectedItemLabwares.length > 0) { + const selectedLabwaresOnDeck = selectedItemLabwares + .map(item => (item?.id != null ? labware[item.id] : null)) + .filter(Boolean) + + const labwaresToRender = + hoveredItemLabware != null + ? [hoveredItemLabware] + : selectedLabwaresOnDeck + + labwaresToRender.forEach((labwareOnDeck, index) => { + if (!labwareOnDeck) { + console.warn( + `labwareOnDeck was null as ${labwareOnDeck}, expected to find a matching entity` + ) + return + } + + let labwareSlot = labwareOnDeck.slot + const hasTC = Object.values(modules).some( + module => module.type === THERMOCYCLER_MODULE_TYPE + ) + + if (modules[labwareSlot]) { + labwareSlot = modules[labwareSlot].slot + } else if (labware[labwareSlot]) { + const adapter = labware[labwareSlot] + labwareSlot = modules[adapter.slot]?.slot ?? adapter.slot + } + + const position = getPositionFromSlotId(labwareSlot, deckDef) + if (position != null) { + items.push( + selected.id === labwareOnDeck.id + )} + isLast={true} + position={ + hasTC && labwareSlot === 'B1' ? [-20, 282, 0] : position + } + labwareDef={labwareOnDeck.def} + labelText={ + hoveredItemLabware == null + ? selectedItemLabwares.find( + selected => selected.id === labwareOnDeck.id + )?.text ?? '' + : hoveredItem.text ?? '' + } + /> + ) + } + }) + } + + return items + } + + const getModuleItems = (): JSX.Element[] => { + const items: JSX.Element[] = [] + + if (hoveredItemModule != null || selectedItemModule != null) { + const selectedModuleOnDeck = + selectedItemModule?.id != null ? modules[selectedItemModule.id] : null + const moduleOnDeck = hoveredItemModule ?? selectedModuleOnDeck + + if (!moduleOnDeck) { + console.warn( + `moduleOnDeck was null as ${moduleOnDeck}, expected to find a matching entity` + ) + return items + } + + const position = getPositionFromSlotId(moduleOnDeck.slot, deckDef) + if (position != null) { + items.push( + + ) + } + } + + return items + } + + const getTrashItems = (): JSX.Element[] => { + const items: JSX.Element[] = [] + + if (hoveredItemTrash != null || selectedItemTrash != null) { + const selectedTrashOnDeck = + selectedItemTrash?.id != null + ? additionalEquipmentOnDeck[selectedItemTrash.id] + : null + const trashOnDeck = hoveredItemTrash ?? selectedTrashOnDeck + + if (!trashOnDeck) { + console.warn( + `trashOnDeck was null as ${trashOnDeck}, expected to find a matching entity` + ) + return [] + } + + if (hoveredItemTrash != null) { + items.push( + + ) + } + + if (selectedTrashOnDeck != null && selectedItemTrash != null) { + items.push( + + ) + } + } + + return items + } + + const getDeckItems = (): JSX.Element[] => { + const items: JSX.Element[] = [] + + if (hoveredDeckItem != null || selectedItemSlot != null) { + const slot = hoveredDeckItem ?? selectedItemSlot?.id + + if (slot === WASTE_CHUTE_CUTOUT) { + items.push( + + ) + } else { + const addressableArea = + slot != null && slot !== WASTE_CHUTE_CUTOUT + ? getAddressableAreaFromSlotId(slot, deckDef) + : null + + if (!addressableArea) { + console.warn( + `addressableArea was null as ${addressableArea}, expected to find a matching entity` + ) + return [] + } + items.push( + + ) + } + } + + return items + } + + const renderItems = (): JSX.Element[] => { + return [ + ...getLabwareItems(), + ...getModuleItems(), + ...getTrashItems(), + ...getDeckItems(), + ] + } + + return <>{renderItems()} +} diff --git a/protocol-designer/src/pages/Designer/DeckSetup/HoveredItems.tsx b/protocol-designer/src/pages/Designer/DeckSetup/HoveredItems.tsx index 79bba166f88..a96d2418607 100644 --- a/protocol-designer/src/pages/Designer/DeckSetup/HoveredItems.tsx +++ b/protocol-designer/src/pages/Designer/DeckSetup/HoveredItems.tsx @@ -76,6 +76,7 @@ export const HoveredItems = ( text: selectedLabwareDef.metadata.displayName, isLast: false, isSelected: true, + isZoomed: true, }, ] : [] diff --git a/protocol-designer/src/pages/Designer/DeckSetup/ModuleLabel.tsx b/protocol-designer/src/pages/Designer/DeckSetup/ModuleLabel.tsx index 1b8eaa4e73b..12849eba08a 100644 --- a/protocol-designer/src/pages/Designer/DeckSetup/ModuleLabel.tsx +++ b/protocol-designer/src/pages/Designer/DeckSetup/ModuleLabel.tsx @@ -1,11 +1,15 @@ import { useRef, useState, useEffect } from 'react' +import { useSelector } from 'react-redux' import { DeckLabelSet } from '@opentrons/components' import { + FLEX_ROBOT_TYPE, HEATERSHAKER_MODULE_TYPE, MAGNETIC_MODULE_TYPE, TEMPERATURE_MODULE_TYPE, + THERMOCYCLER_MODULE_TYPE, getModuleDef2, } from '@opentrons/shared-data' +import { getRobotType } from '../../../file-data/selectors' import type { DeckLabelProps } from '@opentrons/components' import type { CoordinateTuple, ModuleModel } from '@opentrons/shared-data' @@ -15,7 +19,9 @@ interface ModuleLabelProps { orientation: 'left' | 'right' isSelected: boolean isLast: boolean + isZoomed?: boolean labwareInfos?: DeckLabelProps[] + labelName?: string } export const ModuleLabel = (props: ModuleLabelProps): JSX.Element => { const { @@ -25,7 +31,10 @@ export const ModuleLabel = (props: ModuleLabelProps): JSX.Element => { isSelected, isLast, labwareInfos = [], + isZoomed = true, + labelName, } = props + const robotType = useSelector(getRobotType) const labelContainerRef = useRef(null) const [labelContainerHeight, setLabelContainerHeight] = useState(12) @@ -40,14 +49,25 @@ export const ModuleLabel = (props: ModuleLabelProps): JSX.Element => { def?.dimensions.labwareInterfaceXDimension != null ? def.dimensions.xDimension - def?.dimensions.labwareInterfaceXDimension : 0 - // TODO(ja 9/6/24): definitely need to refine these overhang values let leftOverhang = overhang - if (def?.moduleType === TEMPERATURE_MODULE_TYPE) { - leftOverhang = overhang * 2 - } else if (def?.moduleType === HEATERSHAKER_MODULE_TYPE) { - leftOverhang = overhang + 14 - } else if (def?.moduleType === MAGNETIC_MODULE_TYPE) { - leftOverhang = overhang + 8 + + switch (def?.moduleType) { + case TEMPERATURE_MODULE_TYPE: + leftOverhang = overhang * 2 + break + case HEATERSHAKER_MODULE_TYPE: + leftOverhang = overhang + 14 + break + case MAGNETIC_MODULE_TYPE: + leftOverhang = overhang + 8 + break + case THERMOCYCLER_MODULE_TYPE: + if (!isZoomed && robotType === FLEX_ROBOT_TYPE) { + leftOverhang = overhang + 20 + } + break + default: + break } return ( @@ -55,10 +75,11 @@ export const ModuleLabel = (props: ModuleLabelProps): JSX.Element => { ref={labelContainerRef} deckLabels={[ { - text: def?.displayName, + text: labelName ?? def?.displayName, isSelected, isLast, moduleModel: def?.model, + isZoomed: isZoomed, }, ...labwareInfos, ]} diff --git a/protocol-designer/src/pages/Designer/DeckSetup/SelectedHoveredItems.tsx b/protocol-designer/src/pages/Designer/DeckSetup/SelectedHoveredItems.tsx index 33bb727fa38..2400271ee22 100644 --- a/protocol-designer/src/pages/Designer/DeckSetup/SelectedHoveredItems.tsx +++ b/protocol-designer/src/pages/Designer/DeckSetup/SelectedHoveredItems.tsx @@ -115,6 +115,7 @@ export const SelectedHoveredItems = ( text: def.metadata.displayName, isSelected: true, isLast: hoveredLabware == null && selectedNestedLabwareDefUri == null, + isZoomed: true, } labwareInfos.push(selectedLabwareLabel) } @@ -123,6 +124,7 @@ export const SelectedHoveredItems = ( text: selectedNestedLabwareDef.metadata.displayName, isSelected: true, isLast: hoveredLabware == null, + isZoomed: true, } labwareInfos.push(selectedNestedLabwareLabel) } @@ -136,6 +138,7 @@ export const SelectedHoveredItems = ( text: hoveredLabwareDef.metadata.displayName, isSelected: false, isLast: true, + isZoomed: true, } labwareInfos.push(hoverLabelLabel) } @@ -208,6 +211,7 @@ export const SelectedHoveredItems = ( selectedNestedLabwareDef?.metadata.displayName ?? 'unknown name', isSelected: true, isLast: true, + isZoomed: true, }, ]} /> diff --git a/protocol-designer/src/pages/Designer/DeckSetup/__tests__/DeckSetupContainer.test.tsx b/protocol-designer/src/pages/Designer/DeckSetup/__tests__/DeckSetupContainer.test.tsx index eb77c79190a..a5d0226472d 100644 --- a/protocol-designer/src/pages/Designer/DeckSetup/__tests__/DeckSetupContainer.test.tsx +++ b/protocol-designer/src/pages/Designer/DeckSetup/__tests__/DeckSetupContainer.test.tsx @@ -8,14 +8,19 @@ import { renderWithProviders } from '../../../../__testing-utils__' import { selectors } from '../../../../labware-ingred/selectors' import { getDeckSetupForActiveItem } from '../../../../top-selectors/labware-locations' -import { DeckSetupTools } from '../DeckSetupTools' -import { DeckSetupContainer } from '../DeckSetupContainer' +import { + getHoveredDropdownItem, + getSelectedDropdownItem, +} from '../../../../ui/steps/selectors' import { getSelectedTerminalItemId } from '../../../../ui/steps' import { getDisableModuleRestrictions } from '../../../../feature-flags/selectors' import { getRobotType } from '../../../../file-data/selectors' import { DeckSetupDetails } from '../DeckSetupDetails' +import { DeckSetupTools } from '../DeckSetupTools' +import { DeckSetupContainer } from '../DeckSetupContainer' import type * as OpentronsComponents from '@opentrons/components' +vi.mock('../../../../ui/steps/selectors') vi.mock('../../../../top-selectors/labware-locations') vi.mock('../../../../feature-flags/selectors') vi.mock('../DeckSetupTools') @@ -41,6 +46,10 @@ describe('DeckSetupContainer', () => { slot: 'D3', cutout: 'cutoutD3', }) + vi.mocked(getSelectedDropdownItem).mockReturnValue([ + { id: null, text: null }, + ]) + vi.mocked(getHoveredDropdownItem).mockReturnValue({ id: null, text: null }) vi.mocked(DeckSetupTools).mockReturnValue(
mock DeckSetupTools
) vi.mocked(DeckSetupDetails).mockReturnValue(
mock DeckSetupDetails
diff --git a/protocol-designer/src/pages/Designer/HighlightLabware.tsx b/protocol-designer/src/pages/Designer/HighlightLabware.tsx index c2bddc9fbd4..9e2359e66c1 100644 --- a/protocol-designer/src/pages/Designer/HighlightLabware.tsx +++ b/protocol-designer/src/pages/Designer/HighlightLabware.tsx @@ -1,6 +1,7 @@ import { useSelector } from 'react-redux' import { getLabwareEntities } from '../../step-forms/selectors' import { getHoveredStepLabware } from '../../ui/steps' +import { getDesignerTab } from '../../file-data/selectors' import { LabwareLabel } from './LabwareLabel' import type { CoordinateTuple } from '@opentrons/shared-data' import type { LabwareOnDeck } from '../../step-forms' @@ -16,6 +17,7 @@ export function HighlightLabware( const { labwareOnDeck, position } = props const labwareEntities = useSelector(getLabwareEntities) const hoveredLabware = useSelector(getHoveredStepLabware) + const tab = useSelector(getDesignerTab) const adapterId = labwareEntities[labwareOnDeck.slot] != null ? labwareEntities[labwareOnDeck.slot].id @@ -23,6 +25,9 @@ export function HighlightLabware( const highlighted = hoveredLabware.includes(adapterId ?? labwareOnDeck.id) + if (tab === 'protocolSteps') { + return null + } if (highlighted) { return ( { +export const LabwareLabel = (props: LabwareLabelProps): JSX.Element => { const { labwareDef, position, isSelected, isLast, nestedLabwareInfo = [], + labelText = labwareDef.metadata.displayName, } = props const labelContainerRef = useRef(null) const designerTab = useSelector(getDesignerTab) const [labelContainerHeight, setLabelContainerHeight] = useState(0) - const deckLabels = - designerTab === 'startingDeck' - ? [ - ...nestedLabwareInfo, - { - text: labwareDef.metadata.displayName, - isSelected: isSelected, - isLast: isLast, - }, - ] - : [] + const deckLabels = [ + ...nestedLabwareInfo, + { + text: labelText, + isSelected: isSelected, + isLast: isLast, + isZoomed: designerTab === 'startingDeck', + }, + ] useEffect(() => { if (labelContainerRef.current) { diff --git a/protocol-designer/src/pages/Designer/Offdeck/HighlightOffdeckSlot.tsx b/protocol-designer/src/pages/Designer/Offdeck/HighlightOffdeckSlot.tsx new file mode 100644 index 00000000000..4f8ddfdbbea --- /dev/null +++ b/protocol-designer/src/pages/Designer/Offdeck/HighlightOffdeckSlot.tsx @@ -0,0 +1,76 @@ +import { useSelector } from 'react-redux' +import { useTranslation } from 'react-i18next' +import { DeckLabelSet, Flex, POSITION_RELATIVE } from '@opentrons/components' +import { + getHoveredDropdownItem, + getSelectedDropdownItem, +} from '../../../ui/steps/selectors' +import type { CoordinateTuple } from '@opentrons/shared-data' +import type { LabwareOnDeck } from '../../../step-forms' + +interface HighlightOffdeckSlotProps { + labwareOnDeck?: LabwareOnDeck + position: CoordinateTuple +} + +export function HighlightOffdeckSlot( + props: HighlightOffdeckSlotProps +): JSX.Element | null { + const { labwareOnDeck, position } = props + const { t } = useTranslation('application') + const hoveredDropdownItem = useSelector(getHoveredDropdownItem) + const selectedDropdownSelection = useSelector(getSelectedDropdownItem) + + if (labwareOnDeck != null) { + const isLabwareSelectionSelected = selectedDropdownSelection.some( + selected => selected.id === labwareOnDeck?.id + ) + const highlighted = hoveredDropdownItem.id === labwareOnDeck?.id + if (highlighted ?? isLabwareSelectionSelected) { + return ( + + + + ) + } + } else { + const highlightedNewLocation = hoveredDropdownItem.id === 'offDeck' + const selected = selectedDropdownSelection.some( + selected => selected.id === 'offDeck' + ) + if (highlightedNewLocation ?? selected) { + return ( + + ) + } + } + return null +} diff --git a/protocol-designer/src/pages/Designer/Offdeck/OffDeckDetails.tsx b/protocol-designer/src/pages/Designer/Offdeck/OffDeckDetails.tsx index 3759aabf4d5..05720f81555 100644 --- a/protocol-designer/src/pages/Designer/Offdeck/OffDeckDetails.tsx +++ b/protocol-designer/src/pages/Designer/Offdeck/OffDeckDetails.tsx @@ -23,12 +23,17 @@ import { DeckItemHover } from '../DeckSetup/DeckItemHover' import { SlotDetailsContainer } from '../../../organisms' import { wellFillFromWellContents } from '../../../organisms/LabwareOnDeck/utils' import { getRobotType } from '../../../file-data/selectors' +import { + getHoveredDropdownItem, + getSelectedDropdownItem, +} from '../../../ui/steps/selectors' import { SlotOverflowMenu } from '../DeckSetup/SlotOverflowMenu' -import type { DeckSlotId } from '@opentrons/shared-data' +import { HighlightOffdeckSlot } from './HighlightOffdeckSlot' +import type { CoordinateTuple, DeckSlotId } from '@opentrons/shared-data' import type { DeckSetupTabType } from '../types' const OFFDECK_MAP_WIDTH = '41.625rem' - +const ZERO_SLOT_POSITION: CoordinateTuple = [0, 0, 0] interface OffDeckDetailsProps extends DeckSetupTabType { addLabware: () => void } @@ -39,6 +44,8 @@ export function OffDeckDetails(props: OffDeckDetailsProps): JSX.Element { const [menuListId, setShowMenuListForId] = useState(null) const robotType = useSelector(getRobotType) const deckSetup = useSelector(getDeckSetupForActiveItem) + const hoveredDropdownItem = useSelector(getHoveredDropdownItem) + const selectedDropdownSelection = useSelector(getSelectedDropdownItem) const offDeckLabware = Object.values(deckSetup.labware).filter( lw => lw.slot === 'offDeck' ) @@ -98,7 +105,7 @@ export function OffDeckDetails(props: OffDeckDetailsProps): JSX.Element {
- + {offDeckLabware.map(lw => { const wellContents = allWellContentsForActiveItem ? allWellContentsForActiveItem[lw.id] @@ -110,8 +117,21 @@ export function OffDeckDetails(props: OffDeckDetailsProps): JSX.Element { yDimension: dimensions.yDimension ?? 0, zDimension: dimensions.zDimension ?? 0, } + const isLabwareSelectionSelected = selectedDropdownSelection.some( + selected => selected.id === lw.id + ) + const highlighted = hoveredDropdownItem.id === lw.id return ( - + + )} + {menuListId === lw.id ? ( - // TODO fix this rendering position { setShowMenuListForId(null) }} - menuListSlotPosition={[0, 0, 0]} + menuListSlotPosition={ZERO_SLOT_POSITION} invertY /> @@ -161,6 +185,9 @@ export function OffDeckDetails(props: OffDeckDetailsProps): JSX.Element { ) })} + + + {tab === 'startingDeck' ? ( { }) vi.mocked(selectors.getLiquidDisplayColors).mockReturnValue([]) vi.mocked(getAllWellContentsForActiveItem).mockReturnValue({}) + vi.mocked(HighlightOffdeckSlot).mockReturnValue( +
Highlight Offdeck Slot
+ ) + vi.mocked(getSelectedDropdownItem).mockReturnValue([]) + vi.mocked(getHoveredDropdownItem).mockReturnValue({ id: null, text: null }) }) it('renders off-deck overview with 1 labware', () => { @@ -62,5 +73,6 @@ describe('OffDeckDetails', () => { screen.getByText('OFF-DECK LABWARE') screen.getByText('mock LabwareRender') screen.getByText('Add labware') + expect(screen.getAllByText('Highlight Offdeck Slot')).toHaveLength(2) }) }) diff --git a/protocol-designer/src/pages/Designer/ProtocolSteps/StepForm/PipetteFields/LabwareField.tsx b/protocol-designer/src/pages/Designer/ProtocolSteps/StepForm/PipetteFields/LabwareField.tsx index 5fb840e980b..07e08b8a299 100644 --- a/protocol-designer/src/pages/Designer/ProtocolSteps/StepForm/PipetteFields/LabwareField.tsx +++ b/protocol-designer/src/pages/Designer/ProtocolSteps/StepForm/PipetteFields/LabwareField.tsx @@ -1,17 +1,19 @@ -import { useSelector } from 'react-redux' +import { useDispatch, useSelector } from 'react-redux' import { useTranslation } from 'react-i18next' import { getDisposalOptions, getLabwareOptions, } from '../../../../../ui/labware/selectors' +import { hoverSelection } from '../../../../../ui/steps/actions/actions' import { DropdownStepFormField } from '../../../../../molecules' import type { FieldProps } from '../types' export function LabwareField(props: FieldProps): JSX.Element { const { name } = props - const { i18n, t } = useTranslation('protocol_steps') + const { i18n, t } = useTranslation(['protocol_steps', 'application']) const disposalOptions = useSelector(getDisposalOptions) const options = useSelector(getLabwareOptions) + const dispatch = useDispatch() const allOptions = name === 'dispense_labware' ? [...options, ...disposalOptions] @@ -23,6 +25,12 @@ export function LabwareField(props: FieldProps): JSX.Element { name={name} options={allOptions} title={i18n.format(t(`${name}`), 'capitalize')} + onEnter={(id: string) => { + dispatch(hoverSelection({ id, text: t('application:select') })) + }} + onExit={() => { + dispatch(hoverSelection({ id: null, text: null })) + }} /> ) } diff --git a/protocol-designer/src/pages/Designer/ProtocolSteps/StepForm/StepFormToolbox.tsx b/protocol-designer/src/pages/Designer/ProtocolSteps/StepForm/StepFormToolbox.tsx index 6ffbfafbcf2..d8ec3458985 100644 --- a/protocol-designer/src/pages/Designer/ProtocolSteps/StepForm/StepFormToolbox.tsx +++ b/protocol-designer/src/pages/Designer/ProtocolSteps/StepForm/StepFormToolbox.tsx @@ -65,6 +65,10 @@ import type { LiquidHandlingTab, StepFormProps, } from './types' +import { + hoverSelection, + selectDropdownItem, +} from '../../../../ui/steps/actions/actions' type StepFormMap = { [K in StepType]?: React.ComponentType | null @@ -239,6 +243,8 @@ export function StepFormToolbox(props: StepFormToolboxProps): JSX.Element { }) ) dispatch(analyticsEvent(stepDuration)) + dispatch(selectDropdownItem({ selection: null, mode: 'clear' })) + dispatch(hoverSelection({ id: null, text: null })) } else { setShowFormErrors(true) if (tab === 'aspirate' && isDispenseError && !isAspirateError) { @@ -300,7 +306,16 @@ export function StepFormToolbox(props: StepFormToolboxProps): JSX.Element { } childrenPadding="0" - onCloseClick={handleClose} + onCloseClick={() => { + handleClose() + dispatch( + selectDropdownItem({ + selection: null, + mode: 'clear', + }) + ) + dispatch(hoverSelection({ id: null, text: null })) + }} closeButton={} confirmButton={ diff --git a/protocol-designer/src/pages/Designer/ProtocolSteps/StepForm/StepTools/HeaterShakerTools/index.tsx b/protocol-designer/src/pages/Designer/ProtocolSteps/StepForm/StepTools/HeaterShakerTools/index.tsx index 1577db5da8c..392bba3d7cc 100644 --- a/protocol-designer/src/pages/Designer/ProtocolSteps/StepForm/StepTools/HeaterShakerTools/index.tsx +++ b/protocol-designer/src/pages/Designer/ProtocolSteps/StepForm/StepTools/HeaterShakerTools/index.tsx @@ -1,4 +1,4 @@ -import { useSelector } from 'react-redux' +import { useDispatch, useSelector } from 'react-redux' import { useTranslation } from 'react-i18next' import { Box, @@ -8,6 +8,7 @@ import { SPACING, StyledText, } from '@opentrons/components' +import { hoverSelection } from '../../../../../../ui/steps/actions/actions' import { getHeaterShakerLabwareOptions } from '../../../../../../ui/modules/selectors' import { DropdownStepFormField, @@ -21,7 +22,7 @@ export function HeaterShakerTools(props: StepFormProps): JSX.Element { const { propsForFields, formData, visibleFormErrors } = props const { t } = useTranslation(['application', 'form', 'protocol_steps']) const moduleLabwareOptions = useSelector(getHeaterShakerLabwareOptions) - + const dispatch = useDispatch() const mappedErrorsToField = getFormErrorsMappedToField(visibleFormErrors) return ( @@ -34,6 +35,12 @@ export function HeaterShakerTools(props: StepFormProps): JSX.Element { {...propsForFields.moduleId} options={moduleLabwareOptions} title={t('protocol_steps:module')} + onEnter={(id: string) => { + dispatch(hoverSelection({ id, text: t('select') })) + }} + onExit={() => { + dispatch(hoverSelection({ id: null, text: null })) + }} /> - - - {t('protocol_steps:module')} - - - - - {slotInfo[0]} - - - {slotInfo[1]} - - - } - description={ - - - - } - /> - - + { + dispatch( + hoverSelection({ + id, + text: t('application:location'), + }) + ) + }} + onExit={() => { + dispatch(hoverSelection({ id: null, text: null })) + }} /> ) } diff --git a/protocol-designer/src/pages/Designer/ProtocolSteps/StepForm/StepTools/MoveLabwareTools/MoveLabwareField.tsx b/protocol-designer/src/pages/Designer/ProtocolSteps/StepForm/StepTools/MoveLabwareTools/MoveLabwareField.tsx index 539905ec4c2..c27e95e1eb9 100644 --- a/protocol-designer/src/pages/Designer/ProtocolSteps/StepForm/StepTools/MoveLabwareTools/MoveLabwareField.tsx +++ b/protocol-designer/src/pages/Designer/ProtocolSteps/StepForm/StepTools/MoveLabwareTools/MoveLabwareField.tsx @@ -1,17 +1,25 @@ -import { useSelector } from 'react-redux' +import { useDispatch, useSelector } from 'react-redux' import { useTranslation } from 'react-i18next' import { getMoveLabwareOptions } from '../../../../../../ui/labware/selectors' import { DropdownStepFormField } from '../../../../../../molecules' +import { hoverSelection } from '../../../../../../ui/steps/actions/actions' import type { FieldProps } from '../../types' export function MoveLabwareField(props: FieldProps): JSX.Element { const options = useSelector(getMoveLabwareOptions) - const { t } = useTranslation('protocol_steps') + const dispatch = useDispatch() + const { t } = useTranslation(['protocol_steps', 'application']) return ( { + dispatch(hoverSelection({ id, text: t('application:select') })) + }} + onExit={() => { + dispatch(hoverSelection({ id: null, text: null })) + }} /> ) } diff --git a/protocol-designer/src/pages/Designer/ProtocolSteps/StepForm/StepTools/TemperatureTools/index.tsx b/protocol-designer/src/pages/Designer/ProtocolSteps/StepForm/StepTools/TemperatureTools/index.tsx index d384a6f6217..39bd3a77b8c 100644 --- a/protocol-designer/src/pages/Designer/ProtocolSteps/StepForm/StepTools/TemperatureTools/index.tsx +++ b/protocol-designer/src/pages/Designer/ProtocolSteps/StepForm/StepTools/TemperatureTools/index.tsx @@ -1,5 +1,5 @@ import { useTranslation } from 'react-i18next' -import { useSelector } from 'react-redux' +import { useDispatch, useSelector } from 'react-redux' import { Box, COLORS, @@ -7,6 +7,7 @@ import { Flex, SPACING, } from '@opentrons/components' +import { hoverSelection } from '../../../../../../ui/steps/actions/actions' import { getTemperatureLabwareOptions } from '../../../../../../ui/modules/selectors' import { DropdownStepFormField, @@ -20,7 +21,7 @@ export function TemperatureTools(props: StepFormProps): JSX.Element { const { propsForFields, formData, visibleFormErrors } = props const { t } = useTranslation(['application', 'form', 'protocol_steps']) const moduleLabwareOptions = useSelector(getTemperatureLabwareOptions) - + const dispatch = useDispatch() const mappedErrorsToField = getFormErrorsMappedToField(visibleFormErrors) return ( @@ -34,6 +35,12 @@ export function TemperatureTools(props: StepFormProps): JSX.Element { tooltipContent={null} options={moduleLabwareOptions} title={t('protocol_steps:module')} + onEnter={(id: string) => { + dispatch(hoverSelection({ id, text: t('select') })) + }} + onExit={() => { + dispatch(hoverSelection({ id: null, text: null })) + }} /> diff --git a/protocol-designer/src/pages/Designer/ProtocolSteps/StepForm/StepTools/ThermocyclerTools/index.tsx b/protocol-designer/src/pages/Designer/ProtocolSteps/StepForm/StepTools/ThermocyclerTools/index.tsx index 3e85004549e..335facdfd6c 100644 --- a/protocol-designer/src/pages/Designer/ProtocolSteps/StepForm/StepTools/ThermocyclerTools/index.tsx +++ b/protocol-designer/src/pages/Designer/ProtocolSteps/StepForm/StepTools/ThermocyclerTools/index.tsx @@ -8,7 +8,6 @@ import { RadioButton, SPACING, } from '@opentrons/components' - import { ProfileSettings } from './ProfileSettings' import { ProfileStepsSummary } from './ProfileStepsSummary' import { ThermocyclerState } from './ThermocyclerState' @@ -27,8 +26,7 @@ export function ThermocyclerTools(props: StepFormProps): JSX.Element { focusedField, setShowFormErrors, } = props - const { t } = useTranslation('form') - + const { t } = useTranslation(['form', 'application']) const [contentType, setContentType] = useState( formData.thermocyclerFormType as ThermocyclerContentType ) diff --git a/protocol-designer/src/pages/Designer/ProtocolSteps/StepForm/StepTools/__tests__/MagnetTools.test.tsx b/protocol-designer/src/pages/Designer/ProtocolSteps/StepForm/StepTools/__tests__/MagnetTools.test.tsx index 15b4adcd78b..6078835fbf6 100644 --- a/protocol-designer/src/pages/Designer/ProtocolSteps/StepForm/StepTools/__tests__/MagnetTools.test.tsx +++ b/protocol-designer/src/pages/Designer/ProtocolSteps/StepForm/StepTools/__tests__/MagnetTools.test.tsx @@ -103,9 +103,6 @@ describe('MagnetTools', () => { it('renders the text and a switch button for v2', () => { render(props) screen.getByText('Module') - screen.getByText('10') - screen.getByText('mock labware') - screen.getByText('mock module') screen.getByText('Magnet state') screen.getByLabelText('Engage') const toggleButton = screen.getByRole('switch') diff --git a/protocol-designer/src/pages/Designer/index.tsx b/protocol-designer/src/pages/Designer/index.tsx index 410faa44739..5e560fcf6e4 100644 --- a/protocol-designer/src/pages/Designer/index.tsx +++ b/protocol-designer/src/pages/Designer/index.tsx @@ -13,7 +13,10 @@ import { ToggleGroup, useOnClickOutside, } from '@opentrons/components' -import { selectTerminalItem } from '../../ui/steps/actions/actions' +import { + selectDropdownItem, + selectTerminalItem, +} from '../../ui/steps/actions/actions' import { useKitchen } from '../../organisms/Kitchen/hooks' import { getDeckSetupForActiveItem } from '../../top-selectors/labware-locations' import { generateNewProtocol } from '../../labware-ingred/actions' @@ -68,6 +71,12 @@ export function Designer(): JSX.Element { isActive: tab === 'startingDeck', onClick: () => { dispatch(selectDesignerTab({ tab: 'startingDeck' })) + dispatch( + selectDropdownItem({ + selection: null, + mode: 'clear', + }) + ) }, } const protocolStepTab = { diff --git a/protocol-designer/src/step-forms/test/createPresavedStepForm.test.ts b/protocol-designer/src/step-forms/test/createPresavedStepForm.test.ts index 9a78b49d0ed..77e1f25a229 100644 --- a/protocol-designer/src/step-forms/test/createPresavedStepForm.test.ts +++ b/protocol-designer/src/step-forms/test/createPresavedStepForm.test.ts @@ -37,6 +37,7 @@ beforeEach(() => { def: { parameters: { magneticModuleEngageHeight: EXAMPLE_ENGAGE_HEIGHT, + isTiprack: false, }, }, } diff --git a/protocol-designer/src/step-forms/utils/createPresavedStepForm.ts b/protocol-designer/src/step-forms/utils/createPresavedStepForm.ts index 387a3068a18..5951fbf0a04 100644 --- a/protocol-designer/src/step-forms/utils/createPresavedStepForm.ts +++ b/protocol-designer/src/step-forms/utils/createPresavedStepForm.ts @@ -2,6 +2,7 @@ import last from 'lodash/last' import { HEATERSHAKER_MODULE_TYPE, MAGNETIC_MODULE_TYPE, + TEMPERATURE_MODULE_TYPE, THERMOCYCLER_MODULE_TYPE, } from '@opentrons/shared-data' import { @@ -29,6 +30,7 @@ import type { FormData, StepType, StepIdType } from '../../form-types' import type { InitialDeckSetup } from '../types' import type { FormPatch } from '../../steplist/actions/types' import type { SavedStepFormState, OrderedStepIdsState } from '../reducers' + export interface CreatePresavedStepFormArgs { stepId: StepIdType stepType: StepType @@ -118,6 +120,64 @@ const _patchDefaultDropTipLocation = (args: { return null } +const _patchDefaultLabwareLocations = (args: { + labwareEntities: LabwareEntities + pipetteEntities: PipetteEntities + stepType: StepType +}): FormUpdater => formData => { + const { labwareEntities, pipetteEntities, stepType } = args + + const formHasMoveLabware = + formData && 'labware' in formData && stepType === 'moveLabware' + + const filteredLabware = Object.values(labwareEntities).filter( + lw => + // Filter out the tiprack, adapter, and lid entities + !lw.def?.parameters.isTiprack && + !lw.def?.allowedRoles?.includes('adapter') && + !lw.def?.allowedRoles?.includes('lid') + ) + + const filteredMoveLabware = Object.values(labwareEntities).filter( + lw => + // Filter out adapter entities + !lw.def?.allowedRoles?.includes('adapter') + ) + + const formHasAspirateLabware = formData && 'aspirate_labware' in formData + const formHasMixLabware = + formData && 'labware' in formData && stepType === 'mix' + + if (filteredLabware.length === 1 && formHasAspirateLabware) { + return handleFormChange( + { aspirate_labware: filteredLabware[0].id ?? null }, + formData, + pipetteEntities, + labwareEntities + ) + } + + if (filteredLabware.length === 1 && formHasMixLabware) { + return handleFormChange( + { labware: filteredLabware[0].id ?? null }, + formData, + pipetteEntities, + labwareEntities + ) + } + + if (filteredMoveLabware.length === 1 && formHasMoveLabware) { + return handleFormChange( + { labware: filteredMoveLabware[0].id }, + formData, + pipetteEntities, + labwareEntities + ) + } + + return null +} + const _patchDefaultMagnetFields = (args: { initialDeckSetup: InitialDeckSetup orderedStepIds: OrderedStepIdsState @@ -168,13 +228,17 @@ const _patchTemperatureModuleId = (args: { stepType: StepType }): FormUpdater => () => { const { initialDeckSetup, orderedStepIds, savedStepForms, stepType } = args + const numOfModules = + Object.values(initialDeckSetup.modules).filter( + module => module.type === TEMPERATURE_MODULE_TYPE + )?.length ?? 1 const hasTemperatureModuleId = stepType === 'pause' || stepType === 'temperature' // Auto-populate moduleId field of 'pause' and 'temperature' steps. // // Bypass dependent field changes, do not use handleFormChange - if (hasTemperatureModuleId) { + if (hasTemperatureModuleId && numOfModules === 1) { const moduleId = getNextDefaultTemperatureModuleId( savedStepForms, orderedStepIds, @@ -195,6 +259,10 @@ const _patchHeaterShakerModuleId = (args: { stepType: StepType }): FormUpdater => () => { const { initialDeckSetup, stepType } = args + const numOfModules = + Object.values(initialDeckSetup.modules).filter( + module => module.type === HEATERSHAKER_MODULE_TYPE + )?.length ?? 1 const hasHeaterShakerModuleId = stepType === 'pause' || stepType === 'heaterShaker' @@ -202,7 +270,7 @@ const _patchHeaterShakerModuleId = (args: { // Note, if both a temperature module and a heater shaker module are present, the pause form // will default to use the heater shaker // Bypass dependent field changes, do not use handleFormChange - if (hasHeaterShakerModuleId) { + if (hasHeaterShakerModuleId && numOfModules === 1) { const moduleId = getModuleOnDeckByType(initialDeckSetup, HEATERSHAKER_MODULE_TYPE)?.id ?? null @@ -273,6 +341,12 @@ export const createPresavedStepForm = ({ additionalEquipmentEntities, }) + const updateDefaultLabwareLocations = _patchDefaultLabwareLocations({ + labwareEntities, + pipetteEntities, + stepType, + }) + const updateDefaultPipette = _patchDefaultPipette({ initialDeckSetup, labwareEntities, @@ -317,6 +391,7 @@ export const createPresavedStepForm = ({ updateThermocyclerFields, updateHeaterShakerModuleId, updateMagneticModuleId, + updateDefaultLabwareLocations, ].reduce( (acc, updater: FormUpdater) => { const updates = updater(acc) diff --git a/protocol-designer/src/ui/steps/actions/__tests__/actions.test.ts b/protocol-designer/src/ui/steps/actions/__tests__/actions.test.ts index 7dbe2b12324..a4d62738ffe 100644 --- a/protocol-designer/src/ui/steps/actions/__tests__/actions.test.ts +++ b/protocol-designer/src/ui/steps/actions/__tests__/actions.test.ts @@ -7,7 +7,7 @@ import * as utils from '../../../../utils' import * as stepFormSelectors from '../../../../step-forms/selectors' import { getRobotStateTimeline } from '../../../../file-data/selectors' import { getMultiSelectLastSelected } from '../../selectors' -import { selectStep, selectAllSteps, deselectAllSteps } from '../actions' +import { selectAllSteps, deselectAllSteps } from '../actions' import { duplicateStep, duplicateMultipleSteps, @@ -52,38 +52,6 @@ const initialRobotState: RobotState = { } describe('steps actions', () => { - describe('selectStep', () => { - const stepId = 'stepId' - beforeEach(() => { - when(vi.mocked(stepFormSelectors.getSavedStepForms)) - .calledWith(expect.anything()) - .thenReturn({ - stepId: { - foo: 'getSavedStepFormsResult', - } as any, - }) - }) - afterEach(() => { - vi.resetAllMocks() - }) - // TODO(IL, 2020-04-17): also test scroll to top behavior - it('should select the step and populate the form', () => { - const store: any = mockStore() - store.dispatch(selectStep(stepId)) - expect(store.getActions()).toEqual([ - { - type: 'SELECT_STEP', - payload: stepId, - }, - { - type: 'POPULATE_FORM', - payload: { - foo: 'getSavedStepFormsResult', - }, - }, - ]) - }) - }) describe('selectAllSteps', () => { let ids: string[] beforeEach(() => { diff --git a/protocol-designer/src/ui/steps/actions/__tests__/addAndSelectStep.test.ts b/protocol-designer/src/ui/steps/actions/__tests__/addAndSelectStep.test.ts index 054133c6057..c19e9f56483 100644 --- a/protocol-designer/src/ui/steps/actions/__tests__/addAndSelectStep.test.ts +++ b/protocol-designer/src/ui/steps/actions/__tests__/addAndSelectStep.test.ts @@ -1,15 +1,19 @@ import { describe, expect, it, vi, beforeEach } from 'vitest' +import { fixture12Trough, fixtureTiprack1000ul } from '@opentrons/shared-data' import { addAndSelectStep } from '../thunks' import { PRESAVED_STEP_ID } from '../../../../steplist/types' import { addHint } from '../../../../tutorial/actions' import { selectors as labwareIngredSelectors } from '../../../../labware-ingred/selectors' import * as fileDataSelectors from '../../../../file-data/selectors' +import { getInitialDeckSetup } from '../../../../step-forms/selectors' +import type { LabwareDefinition2 } from '@opentrons/shared-data' import type { StepType } from '../../../../form-types' vi.mock('../../../../tutorial/actions') vi.mock('../../../../ui/modules/selectors') vi.mock('../../../../labware-ingred/selectors') vi.mock('../../../../file-data/selectors') +vi.mock('../../../../step-forms/selectors') const dispatch = vi.fn() const getState = vi.fn() @@ -20,6 +24,12 @@ beforeEach(() => { vi.mocked(fileDataSelectors.getRobotStateTimeline).mockReturnValue( 'mockGetRobotStateTimelineValue' as any ) + vi.mocked(getInitialDeckSetup).mockReturnValue({ + modules: {}, + labware: {}, + pipettes: {}, + additionalEquipmentOnDeck: {}, + }) }) describe('addAndSelectStep', () => { it('should dispatch addStep thunk, and no hints when no hints are applicable (eg pause step)', () => { @@ -43,4 +53,310 @@ describe('addAndSelectStep', () => { ], ]) }) + it('should dispatch a thermocycler selected action if the step type is thermocycler', () => { + vi.mocked(getInitialDeckSetup).mockReturnValue({ + modules: { + modId: { + type: 'thermocyclerModuleType', + id: 'modId', + slot: 'B2', + model: 'thermocyclerModuleV1', + moduleState: {} as any, + }, + }, + labware: {}, + pipettes: {}, + additionalEquipmentOnDeck: {}, + }) + const stepType: StepType = 'thermocycler' + const payload = { + stepType, + } + addAndSelectStep(payload)(dispatch, getState) + expect(dispatch.mock.calls).toEqual([ + [ + { + type: 'ADD_STEP', + payload: { + id: PRESAVED_STEP_ID, + stepType: 'thermocycler', + }, + meta: { + robotStateTimeline: 'mockGetRobotStateTimelineValue', + }, + }, + ], + [ + { + type: 'SELECT_DROPDOWN_ITEM', + payload: { + selection: { id: 'modId', text: 'Selected', field: '1' }, + mode: 'add', + }, + }, + ], + ]) + }) + it('should dispatch a magnet module selected action if the step type is magnet', () => { + vi.mocked(getInitialDeckSetup).mockReturnValue({ + modules: { + modId: { + type: 'magneticModuleType', + id: 'modId', + slot: '1', + model: 'magneticModuleV1', + moduleState: {} as any, + }, + }, + labware: {}, + pipettes: {}, + additionalEquipmentOnDeck: {}, + }) + const stepType: StepType = 'magnet' + const payload = { + stepType, + } + addAndSelectStep(payload)(dispatch, getState) + expect(dispatch.mock.calls).toEqual([ + [ + { + type: 'ADD_STEP', + payload: { + id: PRESAVED_STEP_ID, + stepType: 'magnet', + }, + meta: { + robotStateTimeline: 'mockGetRobotStateTimelineValue', + }, + }, + ], + [ + { + type: 'SELECT_DROPDOWN_ITEM', + payload: { + selection: { id: 'modId', text: 'Selected', field: '1' }, + mode: 'add', + }, + }, + ], + ]) + }) + it('should dispatch a temperature module selected action if the step type is temperature and only 1 temp mod', () => { + vi.mocked(getInitialDeckSetup).mockReturnValue({ + modules: { + modId: { + type: 'temperatureModuleType', + id: 'modId', + slot: 'B2', + model: 'temperatureModuleV1', + moduleState: {} as any, + }, + }, + labware: {}, + pipettes: {}, + additionalEquipmentOnDeck: {}, + }) + const stepType: StepType = 'temperature' + const payload = { + stepType, + } + addAndSelectStep(payload)(dispatch, getState) + expect(dispatch.mock.calls).toEqual([ + [ + { + type: 'ADD_STEP', + payload: { + id: PRESAVED_STEP_ID, + stepType: 'temperature', + }, + meta: { + robotStateTimeline: 'mockGetRobotStateTimelineValue', + }, + }, + ], + [ + { + type: 'SELECT_DROPDOWN_ITEM', + payload: { + selection: { id: 'modId', text: 'Selected', field: '1' }, + mode: 'add', + }, + }, + ], + ]) + }) + it('should not dispatch hs module selected action if the step type is hs and 2 mods', () => { + vi.mocked(getInitialDeckSetup).mockReturnValue({ + modules: { + modId: { + type: 'heaterShakerModuleType', + id: 'modId', + slot: 'B2', + model: 'heaterShakerModuleV1', + moduleState: {} as any, + }, + modId2: { + type: 'heaterShakerModuleType', + id: 'modId2', + slot: 'A1', + model: 'heaterShakerModuleV1', + moduleState: {} as any, + }, + }, + labware: {}, + pipettes: {}, + additionalEquipmentOnDeck: {}, + }) + const stepType: StepType = 'heaterShaker' + const payload = { + stepType, + } + addAndSelectStep(payload)(dispatch, getState) + expect(dispatch.mock.calls).toEqual([ + [ + { + type: 'ADD_STEP', + payload: { + id: PRESAVED_STEP_ID, + stepType: 'heaterShaker', + }, + meta: { + robotStateTimeline: 'mockGetRobotStateTimelineValue', + }, + }, + ], + ]) + }) + it('should dispatch labware selected action if the step type is mix and only 1 labware', () => { + vi.mocked(getInitialDeckSetup).mockReturnValue({ + modules: {}, + labware: { + labware: { + id: 'labware', + def: fixture12Trough as LabwareDefinition2, + labwareDefURI: 'mockDefUri', + slot: 'A1', + }, + labware2: { + id: 'labware2', + def: fixtureTiprack1000ul as LabwareDefinition2, + labwareDefURI: 'mockDefUri', + slot: 'B1', + }, + }, + pipettes: {}, + additionalEquipmentOnDeck: {}, + }) + const stepType: StepType = 'mix' + const payload = { + stepType, + } + addAndSelectStep(payload)(dispatch, getState) + expect(dispatch.mock.calls).toEqual([ + [ + { + type: 'ADD_STEP', + payload: { + id: PRESAVED_STEP_ID, + stepType: 'mix', + }, + meta: { + robotStateTimeline: 'mockGetRobotStateTimelineValue', + }, + }, + ], + [ + { + type: 'SELECT_DROPDOWN_ITEM', + payload: { + selection: { id: 'labware', text: 'Selected', field: '1' }, + mode: 'add', + }, + }, + ], + ]) + }) + it('should not dispatch labware selected action if the step type is moveLiquid and 2 labware', () => { + vi.mocked(getInitialDeckSetup).mockReturnValue({ + modules: {}, + labware: { + labware: { + id: 'labware', + def: fixture12Trough as LabwareDefinition2, + labwareDefURI: 'mockDefUri', + slot: 'A1', + }, + labware2: { + id: 'labware2', + def: fixture12Trough as LabwareDefinition2, + labwareDefURI: 'mockDefUri', + slot: 'B1', + }, + }, + pipettes: {}, + additionalEquipmentOnDeck: {}, + }) + const stepType: StepType = 'moveLiquid' + const payload = { + stepType, + } + addAndSelectStep(payload)(dispatch, getState) + expect(dispatch.mock.calls).toEqual([ + [ + { + type: 'ADD_STEP', + payload: { + id: PRESAVED_STEP_ID, + stepType: 'moveLiquid', + }, + meta: { + robotStateTimeline: 'mockGetRobotStateTimelineValue', + }, + }, + ], + ]) + }) + it('should dispatch move labware selected action if the step type is moveLabware and only 1 labware', () => { + vi.mocked(getInitialDeckSetup).mockReturnValue({ + modules: {}, + labware: { + labware2: { + id: 'labware2', + def: fixtureTiprack1000ul as LabwareDefinition2, + labwareDefURI: 'mockDefUri', + slot: 'B1', + }, + }, + pipettes: {}, + additionalEquipmentOnDeck: {}, + }) + const stepType: StepType = 'moveLabware' + const payload = { + stepType, + } + addAndSelectStep(payload)(dispatch, getState) + expect(dispatch.mock.calls).toEqual([ + [ + { + type: 'ADD_STEP', + payload: { + id: PRESAVED_STEP_ID, + stepType: 'moveLabware', + }, + meta: { + robotStateTimeline: 'mockGetRobotStateTimelineValue', + }, + }, + ], + [ + { + type: 'SELECT_DROPDOWN_ITEM', + payload: { + selection: { id: 'labware2', text: 'Selected', field: '1' }, + mode: 'add', + }, + }, + ], + ]) + }) }) diff --git a/protocol-designer/src/ui/steps/actions/actions.ts b/protocol-designer/src/ui/steps/actions/actions.ts index c85db48137d..4343760c455 100644 --- a/protocol-designer/src/ui/steps/actions/actions.ts +++ b/protocol-designer/src/ui/steps/actions/actions.ts @@ -17,15 +17,19 @@ import type { AnalyticsEventAction } from '../../../analytics/actions' import type { TerminalItemId, SubstepIdentifier } from '../../../steplist/types' import type { AddStepAction, + ClearWellSelectionLabwareKeyAction, HoverOnStepAction, HoverOnSubstepAction, - SelectTerminalItemAction, HoverOnTerminalItemAction, - SetWellSelectionLabwareKeyAction, - ClearWellSelectionLabwareKeyAction, - SelectStepAction, + hoverSelectionAction, + Mode, + selectDropdownItemAction, + Selection, SelectMultipleStepsAction, SelectMultipleStepsForGroupAction, + SelectStepAction, + SelectTerminalItemAction, + SetWellSelectionLabwareKeyAction, ToggleViewSubstepAction, ViewSubstep, } from './types' @@ -48,6 +52,28 @@ export const addStep = (args: { }, } } +export const hoverSelection = (args: Selection): hoverSelectionAction => ({ + type: 'HOVER_DROPDOWN_ITEM', + payload: { id: args.id, text: args.text }, +}) +export const selectDropdownItem = (args: { + selection: Selection | null + mode: Mode +}): selectDropdownItemAction => ({ + type: 'SELECT_DROPDOWN_ITEM', + payload: { + selection: + args.selection != null + ? { + id: args.selection.id, + text: args.selection.text, + field: args.selection.field, + } + : null, + mode: args.mode, + }, +}) + export const hoverOnSubstep = ( payload: SubstepIdentifier ): HoverOnSubstepAction => ({ @@ -95,9 +121,97 @@ export const resetSelectStep = (stepId: StepIdType): ThunkAction => ( type: 'POPULATE_FORM', payload: null, }) + dispatch({ + type: 'SELECT_DROPDOWN_ITEM', + payload: { + selection: { + id: null, + text: null, + }, + mode: 'clear', + }, + }) resetScrollElements() } +const setSelection = ( + formData: { + [x: string]: any + stepType: StepType + id: string + }, + dispatch: ThunkDispatch +): void => { + if (formData.stepType === 'moveLabware') { + dispatch({ + type: 'SELECT_DROPDOWN_ITEM', + payload: { + selection: { id: formData.labware, text: 'Selected', field: '1' }, + mode: 'add', + }, + }) + dispatch({ + type: 'SELECT_DROPDOWN_ITEM', + payload: { + selection: { id: formData.newLocation, text: 'Location', field: '2' }, + mode: 'add', + }, + }) + } else if (formData.stepType === 'moveLiquid') { + dispatch({ + type: 'SELECT_DROPDOWN_ITEM', + payload: { + selection: { + id: formData.aspirate_labware, + text: 'Source', + field: '1', + }, + mode: 'add', + }, + }) + dispatch({ + type: 'SELECT_DROPDOWN_ITEM', + payload: { + selection: { + id: formData.dispense_labware, + text: 'Destination', + field: '2', + }, + mode: 'add', + }, + }) + } else if (formData.stepType === 'mix') { + dispatch({ + type: 'SELECT_DROPDOWN_ITEM', + payload: { + selection: { + id: formData.labware, + text: 'Selected', + field: '1', + }, + mode: 'add', + }, + }) + } else if ( + formData.stepType === 'heaterShaker' || + formData.stepType === 'temperature' || + formData.stepType === 'thermocycler' || + formData.stepType === 'magnet' + ) { + dispatch({ + type: 'SELECT_DROPDOWN_ITEM', + payload: { + selection: { + id: formData.moduleId, + text: 'Selected', + field: '1', + }, + mode: 'add', + }, + }) + } +} + export const populateForm = (stepId: StepIdType): ThunkAction => ( dispatch: ThunkDispatch, getState: GetState @@ -108,9 +222,9 @@ export const populateForm = (stepId: StepIdType): ThunkAction => ( type: 'POPULATE_FORM', payload: formData, }) + setSelection(formData, dispatch) resetScrollElements() } - export const selectStep = (stepId: StepIdType): ThunkAction => ( dispatch: ThunkDispatch, getState: GetState @@ -126,9 +240,8 @@ export const selectStep = (stepId: StepIdType): ThunkAction => ( type: 'POPULATE_FORM', payload: formData, }) - resetScrollElements() + setSelection(formData, dispatch) } - // NOTE(sa, 2020-12-11): this is a thunk so that we can populate the batch edit form with things later export const selectMultipleSteps = ( stepIds: StepIdType[], diff --git a/protocol-designer/src/ui/steps/actions/thunks/index.ts b/protocol-designer/src/ui/steps/actions/thunks/index.ts index 47e0d846180..edb4bf08b7f 100644 --- a/protocol-designer/src/ui/steps/actions/thunks/index.ts +++ b/protocol-designer/src/ui/steps/actions/thunks/index.ts @@ -1,16 +1,23 @@ import last from 'lodash/last' +import { + HEATERSHAKER_MODULE_TYPE, + MAGNETIC_MODULE_TYPE, + TEMPERATURE_MODULE_TYPE, + THERMOCYCLER_MODULE_TYPE, +} from '@opentrons/shared-data' import { getUnsavedForm, getUnsavedFormIsPristineSetTempForm, getUnsavedFormIsPristineHeaterShakerForm, getOrderedStepIds, + getInitialDeckSetup, } from '../../../../step-forms/selectors' import { changeFormInput } from '../../../../steplist/actions/actions' import { PRESAVED_STEP_ID } from '../../../../steplist/types' import { PAUSE_UNTIL_TEMP } from '../../../../constants' import { uuid } from '../../../../utils' import { getMultiSelectLastSelected, getSelectedStepId } from '../../selectors' -import { addStep } from '../actions' +import { addStep, selectDropdownItem } from '../actions' import { actions as tutorialActions, selectors as tutorialSelectors, @@ -23,16 +30,128 @@ import type { DuplicateMultipleStepsAction, SelectMultipleStepsAction, } from '../types' + export const addAndSelectStep: (arg: { stepType: StepType }) => ThunkAction = payload => (dispatch, getState) => { const robotStateTimeline = fileDataSelectors.getRobotStateTimeline(getState()) + const initialDeckSetup = getInitialDeckSetup(getState()) + const { modules, labware } = initialDeckSetup dispatch( addStep({ stepType: payload.stepType, robotStateTimeline, }) ) + if (payload.stepType === 'thermocycler') { + const tcId = Object.entries(modules).find( + ([key, module]) => module.type === THERMOCYCLER_MODULE_TYPE + )?.[0] + if (tcId != null) { + dispatch( + selectDropdownItem({ + selection: { + id: tcId, + text: 'Selected', + field: '1', + }, + mode: 'add', + }) + ) + } + } else if (payload.stepType === 'magnet') { + const magId = Object.entries(modules).find( + ([key, module]) => module.type === MAGNETIC_MODULE_TYPE + )?.[0] + if (magId != null) { + dispatch( + selectDropdownItem({ + selection: { + id: magId, + text: 'Selected', + field: '1', + }, + mode: 'add', + }) + ) + } + } else if (payload.stepType === 'temperature') { + const temperatureModules = Object.entries(modules).filter( + ([key, module]) => module.type === TEMPERATURE_MODULE_TYPE + ) + // only set selected temperature module if only 1 type is on deck + const tempId = + temperatureModules.length === 1 ? temperatureModules[0][0] : null + if (tempId != null) { + dispatch( + selectDropdownItem({ + selection: { + id: tempId, + text: 'Selected', + field: '1', + }, + mode: 'add', + }) + ) + } + } else if (payload.stepType === 'heaterShaker') { + const hsModules = Object.entries(modules).filter( + ([key, module]) => module.type === HEATERSHAKER_MODULE_TYPE + ) + // only set selected h-s module if only 1 type is on deck + const hsId = hsModules.length === 1 ? hsModules[0][0] : null + if (hsId != null) { + dispatch( + selectDropdownItem({ + selection: { + id: hsId, + text: 'Selected', + field: '1', + }, + mode: 'add', + }) + ) + } + } else if (payload.stepType === 'mix' || payload.stepType === 'moveLiquid') { + const labwares = Object.entries(labware).filter( + ([key, lw]) => + !lw.def.parameters.isTiprack && + !lw.def.allowedRoles?.includes('adapter') && + !lw.def.allowedRoles?.includes('lid') + ) + // only set selected labware if only 1 available labware is on deck + const labwareId = labwares.length === 1 ? labwares[0][0] : null + if (labwareId != null) { + dispatch( + selectDropdownItem({ + selection: { + id: labwareId, + text: payload.stepType === 'moveLiquid' ? 'Source' : 'Selected', + field: '1', + }, + mode: 'add', + }) + ) + } + } else if (payload.stepType === 'moveLabware') { + const labwares = Object.entries(labware).filter( + ([key, lw]) => !lw.def.allowedRoles?.includes('adapter') + ) + // only set selected labware if only 1 available labware/tiprack/lid is on deck + const labwareId = labwares.length === 1 ? labwares[0][0] : null + if (labwareId != null) { + dispatch( + selectDropdownItem({ + selection: { + id: labwareId, + text: 'Selected', + field: '1', + }, + mode: 'add', + }) + ) + } + } } export interface ReorderSelectedStepAction { type: 'REORDER_SELECTED_STEP' diff --git a/protocol-designer/src/ui/steps/actions/types.ts b/protocol-designer/src/ui/steps/actions/types.ts index ebbf1e4dff2..52556930d1d 100644 --- a/protocol-designer/src/ui/steps/actions/types.ts +++ b/protocol-designer/src/ui/steps/actions/types.ts @@ -31,6 +31,24 @@ export interface DuplicateMultipleStepsAction { indexToInsert: number } } + +export type Mode = 'clear' | 'add' +export interface Selection { + id: string | null + text: string | null + field?: '1' | '2' +} +export interface selectDropdownItemAction { + type: 'SELECT_DROPDOWN_ITEM' + payload: { + selection: Selection | null + mode: 'add' | 'clear' + } +} +export interface hoverSelectionAction { + type: 'HOVER_DROPDOWN_ITEM' + payload: Selection +} export interface HoverOnSubstepAction { type: 'HOVER_ON_SUBSTEP' payload: SubstepIdentifier diff --git a/protocol-designer/src/ui/steps/reducers.ts b/protocol-designer/src/ui/steps/reducers.ts index 0a40cb6dbe2..e281cb01bc4 100644 --- a/protocol-designer/src/ui/steps/reducers.ts +++ b/protocol-designer/src/ui/steps/reducers.ts @@ -21,6 +21,7 @@ import type { SelectStepAction, SelectMultipleStepsAction, SelectTerminalItemAction, + Selection, } from './actions/types' export type CollapsedStepsState = Record @@ -188,6 +189,50 @@ const selectedSubstep: Reducer = handleActions( }, null ) +const hoveredDropdownItem: Reducer = handleActions( + { + HOVER_DROPDOWN_ITEM: ( + state, + action: { + payload: Selection + } + ) => action.payload, + }, + { id: null, text: null } +) +const selectedDropdownItem: Reducer = handleActions( + { + SELECT_DROPDOWN_ITEM: ( + state: Selection[], + action: { + payload: { + selection: Selection | null + mode: 'add' | 'clear' + } + } + ) => { + const { selection, mode } = action.payload + + switch (mode) { + case 'clear': + return [] + case 'add': { + if (!selection) { + return state + } + const updatedState = state.filter( + sel => sel.field !== selection.field + ) + + return [...updatedState, selection] + } + default: + return state + } + }, + }, + [] +) export interface StepsState { collapsedSteps: CollapsedStepsState selectedItem: SelectedItemState @@ -195,6 +240,8 @@ export interface StepsState { hoveredSubstep: SubstepIdentifier wellSelectionLabwareKey: string | null selectedSubstep: StepIdType | null + hoveredDropdownItem: Selection + selectedDropdownItem: Selection[] } export const _allReducers = { collapsedSteps, @@ -203,6 +250,8 @@ export const _allReducers = { hoveredSubstep, wellSelectionLabwareKey, selectedSubstep, + hoveredDropdownItem, + selectedDropdownItem, } export const rootReducer: Reducer = combineReducers( _allReducers diff --git a/protocol-designer/src/ui/steps/selectors.ts b/protocol-designer/src/ui/steps/selectors.ts index 53848c4a28a..c6f48ff2f2f 100644 --- a/protocol-designer/src/ui/steps/selectors.ts +++ b/protocol-designer/src/ui/steps/selectors.ts @@ -38,6 +38,7 @@ import type { CollapsedStepsState, HoverableItem, } from './reducers' +import type { Selection } from './actions/types' export const rootSelector = (state: BaseState): StepsState => state.ui.steps // ======= Selectors =============================================== @@ -102,6 +103,14 @@ export const getHoveredStepId: Selector = createSelector( item => item && item.selectionType === SINGLE_STEP_SELECTION_TYPE ? item.id : null ) +export const getHoveredDropdownItem: Selector = createSelector( + rootSelector, + (state: StepsState) => state.hoveredDropdownItem +) +export const getSelectedDropdownItem: Selector = createSelector( + rootSelector, + (state: StepsState) => state.selectedDropdownItem +) /** Array of labware (labwareId's) involved in hovered Step, or [] */ export const getHoveredStepLabware = createSelector( diff --git a/shared-data/js/fixtures.ts b/shared-data/js/fixtures.ts index 905429cd111..51f46fdf433 100644 --- a/shared-data/js/fixtures.ts +++ b/shared-data/js/fixtures.ts @@ -295,7 +295,7 @@ export function getFixtureDisplayName( } } -const STANDARD_OT2_SLOTS: AddressableAreaName[] = [ +export const STANDARD_OT2_SLOTS: AddressableAreaName[] = [ ADDRESSABLE_AREA_1, ADDRESSABLE_AREA_2, ADDRESSABLE_AREA_3, @@ -309,7 +309,7 @@ const STANDARD_OT2_SLOTS: AddressableAreaName[] = [ ADDRESSABLE_AREA_11, ] -const STANDARD_FLEX_SLOTS: AddressableAreaName[] = [ +export const STANDARD_FLEX_SLOTS: AddressableAreaName[] = [ A1_ADDRESSABLE_AREA, A2_ADDRESSABLE_AREA, A3_ADDRESSABLE_AREA,