From 1382fcc1dd7f829a9e52766caf29a95a83a990ba Mon Sep 17 00:00:00 2001 From: ncdiehl11 Date: Thu, 8 Aug 2024 14:58:54 -0400 Subject: [PATCH 1/3] feat(app, components): add stacking to non-module stacks Add stack icon, highlight, and click handling to non-module stacks (example: slot + adapter + labware) --- .../SetupLabware/LabwareStackModal.tsx | 21 +++++--- .../SetupLabware/SetupLabwareMap.tsx | 48 +++++++++++++++---- .../ProtocolSetupLabware/LabwareMapView.tsx | 5 ++ .../Labware/LabwareAdapter/index.tsx | 22 ++++++++- .../hardware-sim/Labware/LabwareRender.tsx | 2 + .../labwareInternals/LabwareOutline.tsx | 1 + 6 files changed, 82 insertions(+), 17 deletions(-) diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLabware/LabwareStackModal.tsx b/app/src/organisms/Devices/ProtocolRun/SetupLabware/LabwareStackModal.tsx index 80bd38a3255..b0501a84944 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLabware/LabwareStackModal.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupLabware/LabwareStackModal.tsx @@ -210,14 +210,21 @@ export const LabwareStackModal = ( justifyContent={JUSTIFY_SPACE_BETWEEN} > - + {adapterDef.parameters.loadName === + 'opentrons_flex_96_tiprack_adapter' ? ( + tiprackAdapterImg + ) : ( + + )} - + {moduleModel != null ? ( + + ) : null} ) : null} {moduleModel != null ? ( diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLabware/SetupLabwareMap.tsx b/app/src/organisms/Devices/ProtocolRun/SetupLabware/SetupLabwareMap.tsx index 8a35d8d203e..c7b219ca085 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLabware/SetupLabwareMap.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupLabware/SetupLabwareMap.tsx @@ -25,6 +25,7 @@ import { OffDeckLabwareList } from './OffDeckLabwareList' import type { CompletedProtocolAnalysis, + LoadLabwareRunTimeCommand, ProtocolAnalysisOutput, } from '@opentrons/shared-data' import { LabwareStackModal } from './LabwareStackModal' @@ -137,20 +138,51 @@ export function SetupLabwareMap({ const topLabwareId = labwareInAdapter?.result?.labwareId ?? labwareId const topLabwareDisplayName = labwareInAdapter?.params.displayName ?? displayName - + const isLabwareInStack = + topLabwareDefinition != null && + topLabwareId != null && + labwareInAdapter != null + + const loadLabwareCommand = commands.find( + command => + command.commandType === 'loadLabware' && + command.result?.labwareId === labwareId + ) as LoadLabwareRunTimeCommand + const labwareLocation = loadLabwareCommand?.params.location return { - labwareLocation: { slotName }, + labwareLocation, definition: topLabwareDefinition, topLabwareId, topLabwareDisplayName, + // TODO (nd: 08/08/2024) fix passing of onLabwareClick to actually perform click handling. + // Here, I set to an empty function to produce pointer cursor style. + onLabwareClick: isLabwareInStack ? () => {} : undefined, + highlight: isLabwareInStack && hoverLabwareId === topLabwareId, labwareChildren: ( - + { + if (isLabwareInStack) { + setLabwareStackDetailsLabwareId(topLabwareId) + } + }} + onMouseEnter={() => { + if (topLabwareDefinition != null && topLabwareId != null) { + setHoverLabwareId(() => topLabwareId) + } + }} + onMouseLeave={() => { + setHoverLabwareId(null) + }} + > + + ), + stacked: isLabwareInStack, } } ) diff --git a/app/src/organisms/ProtocolSetupLabware/LabwareMapView.tsx b/app/src/organisms/ProtocolSetupLabware/LabwareMapView.tsx index 1e00d266008..49c6fd38d74 100644 --- a/app/src/organisms/ProtocolSetupLabware/LabwareMapView.tsx +++ b/app/src/organisms/ProtocolSetupLabware/LabwareMapView.tsx @@ -85,6 +85,10 @@ export function LabwareMapView(props: LabwareMapViewProps): JSX.Element { const topLabwareDefinition = labwareInAdapter?.result?.definition ?? labwareDef const topLabwareId = labwareInAdapter?.result?.labwareId ?? labwareId + const isLabwareInStack = + topLabwareDefinition != null && + topLabwareId != null && + labwareInAdapter != null return { labwareLocation: { slotName }, @@ -95,6 +99,7 @@ export function LabwareMapView(props: LabwareMapViewProps): JSX.Element { }, labwareChildren: null, highlight: true, + stacked: isLabwareInStack, } } ) diff --git a/components/src/hardware-sim/Labware/LabwareAdapter/index.tsx b/components/src/hardware-sim/Labware/LabwareAdapter/index.tsx index 978fb7cbea3..fc05d8b5621 100644 --- a/components/src/hardware-sim/Labware/LabwareAdapter/index.tsx +++ b/components/src/hardware-sim/Labware/LabwareAdapter/index.tsx @@ -4,6 +4,9 @@ import { Opentrons96FlatBottomAdapter } from './Opentrons96FlatBottomAdapter' import { OpentronsUniversalFlatAdapter } from './OpentronsUniversalFlatAdapter' import { OpentronsAluminumFlatBottomPlate } from './OpentronsAluminumFlatBottomPlate' import { OpentronsFlex96TiprackAdapter } from './OpentronsFlex96TiprackAdapter' +import { COLORS } from '../../../helix-design-system' +import { LabwareOutline } from '../labwareInternals' +import type { LabwareDefinition2 } from '@opentrons/shared-data' const LABWARE_ADAPTER_LOADNAME_PATHS = { opentrons_96_deep_well_adapter: Opentrons96DeepWellAdapter, @@ -20,13 +23,28 @@ export const labwareAdapterLoadNames = Object.keys( export interface LabwareAdapterProps { labwareLoadName: LabwareAdapterLoadName + definition?: LabwareDefinition2 + highlight?: boolean } export const LabwareAdapter = ( props: LabwareAdapterProps ): JSX.Element | null => { - const { labwareLoadName } = props + const { labwareLoadName, definition, highlight = false } = props + const highlightOutline = + highlight && definition != null ? ( + + ) : null const SVGElement = LABWARE_ADAPTER_LOADNAME_PATHS[labwareLoadName] - return + return ( + + + {highlightOutline} + + ) } diff --git a/components/src/hardware-sim/Labware/LabwareRender.tsx b/components/src/hardware-sim/Labware/LabwareRender.tsx index 41c2537a7d9..9137a2d2f15 100644 --- a/components/src/hardware-sim/Labware/LabwareRender.tsx +++ b/components/src/hardware-sim/Labware/LabwareRender.tsx @@ -88,6 +88,8 @@ export const LabwareRender = (props: LabwareRenderProps): JSX.Element => { > diff --git a/components/src/hardware-sim/Labware/labwareInternals/LabwareOutline.tsx b/components/src/hardware-sim/Labware/labwareInternals/LabwareOutline.tsx index 7478c671114..743743bd6c0 100644 --- a/components/src/hardware-sim/Labware/labwareInternals/LabwareOutline.tsx +++ b/components/src/hardware-sim/Labware/labwareInternals/LabwareOutline.tsx @@ -65,6 +65,7 @@ export function LabwareOutline(props: LabwareOutlineProps): JSX.Element { rx="8" ry="8" showRadius={showRadius} + fill={backgroundFill} /> Date: Fri, 9 Aug 2024 10:01:33 -0400 Subject: [PATCH 2/3] refactor SetupLabwareMap and getLocationInfoNames util to check addressable areas --- .../SetupLabware/SetupLabwareMap.tsx | 13 +----- .../__tests__/getLocationInfoNames.test.ts | 40 +++++++++++++++++++ .../ProtocolRun/utils/getLocationInfoNames.ts | 12 ++++++ 3 files changed, 54 insertions(+), 11 deletions(-) diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLabware/SetupLabwareMap.tsx b/app/src/organisms/Devices/ProtocolRun/SetupLabware/SetupLabwareMap.tsx index c7b219ca085..71609edb0ff 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLabware/SetupLabwareMap.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupLabware/SetupLabwareMap.tsx @@ -25,7 +25,6 @@ import { OffDeckLabwareList } from './OffDeckLabwareList' import type { CompletedProtocolAnalysis, - LoadLabwareRunTimeCommand, ProtocolAnalysisOutput, } from '@opentrons/shared-data' import { LabwareStackModal } from './LabwareStackModal' @@ -143,23 +142,15 @@ export function SetupLabwareMap({ topLabwareId != null && labwareInAdapter != null - const loadLabwareCommand = commands.find( - command => - command.commandType === 'loadLabware' && - command.result?.labwareId === labwareId - ) as LoadLabwareRunTimeCommand - const labwareLocation = loadLabwareCommand?.params.location return { - labwareLocation, + labwareLocation: { slotName }, definition: topLabwareDefinition, topLabwareId, topLabwareDisplayName, - // TODO (nd: 08/08/2024) fix passing of onLabwareClick to actually perform click handling. - // Here, I set to an empty function to produce pointer cursor style. - onLabwareClick: isLabwareInStack ? () => {} : undefined, highlight: isLabwareInStack && hoverLabwareId === topLabwareId, labwareChildren: ( { if (isLabwareInStack) { setLabwareStackDetailsLabwareId(topLabwareId) diff --git a/app/src/organisms/Devices/ProtocolRun/utils/__tests__/getLocationInfoNames.test.ts b/app/src/organisms/Devices/ProtocolRun/utils/__tests__/getLocationInfoNames.test.ts index f917f64035f..d0b3551972f 100644 --- a/app/src/organisms/Devices/ProtocolRun/utils/__tests__/getLocationInfoNames.test.ts +++ b/app/src/organisms/Devices/ProtocolRun/utils/__tests__/getLocationInfoNames.test.ts @@ -6,6 +6,7 @@ import type { ModuleModel } from '@opentrons/shared-data' const ADAPTER_DISPLAY_NAME = 'Opentrons 96 Flat Bottom Adapter' const LABWARE_DISPLAY_NAME = 'Corning 24 Well Plate 3.4 mL Flat' const SLOT = '5' +const SLOT_EXTENSION = 'C4' const MOCK_MODEL = 'heaterShakerModuleV1' as ModuleModel const ADAPTER_ID = 'd9a85adf-d272-4edd-9aae-426ef5756fef:opentrons/opentrons_96_flat_bottom_adapter/1' @@ -119,6 +120,34 @@ const MOCK_ADAPTER_COMMANDS = [ }, }, ] +const MOCK_ADAPTER_EXTENSION_COMMANDS = [ + { + commandType: 'loadLabware', + params: { + location: { + addressableAreaName: SLOT_EXTENSION, + }, + }, + result: { + labwareId: ADAPTER_ID, + definition: { + metadata: { displayName: ADAPTER_DISPLAY_NAME }, + }, + }, + }, + { + commandType: 'loadLabware', + params: { + location: { + labwareId: ADAPTER_ID, + }, + }, + result: { + labwareId: LABWARE_ID, + definition: {}, + }, + }, +] vi.mock('@opentrons/shared-data') @@ -168,4 +197,15 @@ describe('getLocationInfoNames', () => { getLocationInfoNames(LABWARE_ID, MOCK_ADAPTER_COMMANDS as any) ).toEqual(expected) }) + it('returns the adapter, slot number if the labware is on an adapter on the deck extension slot', () => { + const expected = { + slotName: SLOT_EXTENSION, + labwareName: LABWARE_DISPLAY_NAME, + adapterName: ADAPTER_DISPLAY_NAME, + adapterId: ADAPTER_ID, + } + expect( + getLocationInfoNames(LABWARE_ID, MOCK_ADAPTER_EXTENSION_COMMANDS as any) + ).toEqual(expected) + }) }) diff --git a/app/src/organisms/Devices/ProtocolRun/utils/getLocationInfoNames.ts b/app/src/organisms/Devices/ProtocolRun/utils/getLocationInfoNames.ts index c3404945dcb..26d618859f9 100644 --- a/app/src/organisms/Devices/ProtocolRun/utils/getLocationInfoNames.ts +++ b/app/src/organisms/Devices/ProtocolRun/utils/getLocationInfoNames.ts @@ -87,6 +87,18 @@ export function getLocationInfoNames( loadedAdapterCommand?.result?.definition.metadata.displayName, adapterId: loadedAdapterCommand?.result?.labwareId, } + } else if ( + loadedAdapterCommand?.params.location !== 'offDeck' && + 'addressableAreaName' in loadedAdapterCommand?.params.location + ) { + return { + slotName: loadedAdapterCommand?.params.location.addressableAreaName, + labwareName, + labwareNickname, + adapterName: + loadedAdapterCommand?.result?.definition.metadata.displayName, + adapterId: loadedAdapterCommand?.result?.labwareId, + } } else if ( loadedAdapterCommand?.params.location !== 'offDeck' && 'moduleId' in loadedAdapterCommand?.params.location From baed74ffd2ebd8627732259237de15d0bc484e4b Mon Sep 17 00:00:00 2001 From: ncdiehl11 Date: Fri, 9 Aug 2024 10:36:16 -0400 Subject: [PATCH 3/3] use const instead of string literal --- .../Devices/ProtocolRun/SetupLabware/LabwareStackModal.tsx | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLabware/LabwareStackModal.tsx b/app/src/organisms/Devices/ProtocolRun/SetupLabware/LabwareStackModal.tsx index b0501a84944..962f5c579c2 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLabware/LabwareStackModal.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupLabware/LabwareStackModal.tsx @@ -29,6 +29,7 @@ import { getModuleType, TC_MODULE_LOCATION_OT2, TC_MODULE_LOCATION_OT3, + THERMOCYCLER_MODULE_TYPE, } from '@opentrons/shared-data' import tiprackAdapter from '../../../../assets/images/labware/opentrons_flex_96_tiprack_adapter.png' @@ -72,7 +73,7 @@ export const LabwareStackModal = ( const isModuleThermocycler = moduleModel == null ? false - : getModuleType(moduleModel) === 'thermocyclerModuleType' + : getModuleType(moduleModel) === THERMOCYCLER_MODULE_TYPE const thermocyclerLocation = robotType === FLEX_ROBOT_TYPE ? TC_MODULE_LOCATION_OT3