From b3ec74b5431fcc47e6d62a7e6b9b7e9e2bffb33a Mon Sep 17 00:00:00 2001 From: Jethary Date: Tue, 23 Apr 2024 08:03:16 -0400 Subject: [PATCH 1/8] start to add some math --- .../js/helpers/flexAdjacentSlotGetters.ts | 0 .../src/utils/ninetySixChannelCollision.ts | 142 +++++++++++++++++- 2 files changed, 139 insertions(+), 3 deletions(-) create mode 100644 shared-data/js/helpers/flexAdjacentSlotGetters.ts diff --git a/shared-data/js/helpers/flexAdjacentSlotGetters.ts b/shared-data/js/helpers/flexAdjacentSlotGetters.ts new file mode 100644 index 00000000000..e69de29bb2d diff --git a/step-generation/src/utils/ninetySixChannelCollision.ts b/step-generation/src/utils/ninetySixChannelCollision.ts index 7a2b7f3e0c1..9a1ff1e935f 100644 --- a/step-generation/src/utils/ninetySixChannelCollision.ts +++ b/step-generation/src/utils/ninetySixChannelCollision.ts @@ -1,10 +1,141 @@ import toNumber from 'lodash/toNumber' -import { getModuleDef2 } from '@opentrons/shared-data' -import type { RobotState, InvariantContext } from '../types' +import { NozzleConfigurationStyle, getModuleDef2 } from '@opentrons/shared-data' +import type { RobotState, InvariantContext, PipetteEntity } from '../types' const SAFETY_MARGIN = 10 const targetNumbers = ['2', '3', '4'] +const A12_column_front_left_bound = { x: -11.03, y: 2 } +const A12_column_back_right_bound = { x: 526.77, y: 506.2 } +const PRIMARY_NOZZLE = 'A12' +type Point = { x: number; y: number; z?: number } + +const getIsWithinPipetteExtents = ( + pipetteEntity: PipetteEntity, + location: Point, + nozzleConfiguration: NozzleConfigurationStyle, + primaryNozzle: string +): boolean => { + const channels = pipetteEntity.spec.channels + switch (channels) { + case 96: { + if (nozzleConfiguration === 'COLUMN' && primaryNozzle === 'A12') { + return ( + A12_column_front_left_bound.x <= location.x && + location.x <= A12_column_back_right_bound.x && + A12_column_front_left_bound.y <= location.y && + location.y <= A12_column_back_right_bound.y + ) + } + } + case 8: + case 1: + // TODO(jr, 4/22/24): update this to support 8-channel partial tip + // and eventually all pipettes + return true + } +} + +const getPipetteBoundsAtSpecifiedMoveToPosition = ( + primaryNozzle: string, + pipetteEntity: PipetteEntity, + tipLength: number, + destinationPosition: Point +): Point[] => { + const primaryNozzleOffset = pipetteEntity.spec.nozzleMap[primaryNozzle] + const primaryNozzlePosition = { + x: destinationPosition.x, + y: destinationPosition.y + tipLength, + } + const pipetteBoundsOffsets = pipetteEntity.spec.pipetteBoundingBoxOffsets + const backLeftBound = { + x: + primaryNozzlePosition.x - + primaryNozzleOffset[0] + + pipetteBoundsOffsets.backLeftCorner[0], + y: + primaryNozzlePosition.y - + primaryNozzleOffset[1] + + pipetteBoundsOffsets.backLeftCorner[1], + z: primaryNozzleOffset[2] + pipetteBoundsOffsets.backLeftCorner[2], + } + const frontRightBound = { + x: + primaryNozzlePosition.x - + primaryNozzleOffset[0] + + pipetteBoundsOffsets.frontRightCorner[0], + y: + primaryNozzlePosition.y - + primaryNozzleOffset[1] + + pipetteBoundsOffsets.frontRightCorner[1], + z: primaryNozzleOffset[2] + pipetteBoundsOffsets.frontRightCorner[2], + } + + const backRightBound: Point = { + x: backLeftBound.x, + y: backLeftBound.y, + z: frontRightBound.z, + } + const frontLeftBound: Point = { + x: backLeftBound.x, + y: frontRightBound.y, + z: backLeftBound.z, + } + + return [backLeftBound, frontRightBound, backRightBound, frontLeftBound] +} + +export const getIsSafePipetteMovement = ( + robotState: RobotState, + invariantContext: InvariantContext, + pipetteId: string, + destLabwareId: string, + tipRackId: string, + destWellName: string, + destWellLocation: { + origin: string + offset: { x: number; y: number; z: number } + } +): boolean => { + const { pipetteEntities, labwareEntities } = invariantContext + const { labware: labwareState, pipettes, tipState } = robotState + const pipetteEntity = pipetteEntities[pipetteId] + const pipetteHasTip = tipState.pipettes[pipetteId] + const tipLength = pipetteHasTip + ? labwareEntities[tipRackId].def.parameters.tipLength ?? 0 + : 0 + const nozzleConfiguration = pipettes[pipetteId].nozzles + const location = { + x: destWellLocation.offset.x, + y: destWellLocation.offset.y, + } + + // early exit for now if nozzle configuration is not partial tip + if (nozzleConfiguration !== 'COLUMN') { + return true + } + + const isWithinPipetteExtents = getIsWithinPipetteExtents( + pipetteEntity, + location, + nozzleConfiguration, + // TODO(jr, 4/22/24): PD only supports A12 as a primary nozzle for now + PRIMARY_NOZZLE + ) + + if (!isWithinPipetteExtents) { + return false + } else { + const labwareSlot = labwareState[destLabwareId].slot + const pipetteBoundsAtWellLocation = getPipetteBoundsAtSpecifiedMoveToPosition( + PRIMARY_NOZZLE, + pipetteEntity, + tipLength, + destWellLocation.offset + ) + const surroundingSlots = [] + } +} export const getIsTallLabwareWestOf96Channel = ( robotState: RobotState, invariantContext: InvariantContext, @@ -12,8 +143,13 @@ export const getIsTallLabwareWestOf96Channel = ( pipetteId: string, tipRackId: string ): boolean => { - const { labwareEntities, additionalEquipmentEntities } = invariantContext + const { + labwareEntities, + additionalEquipmentEntities, + pipetteEntities, + } = invariantContext const { labware: labwareState, tipState } = robotState + const test = pipetteEntities[pipetteId].spec.pipetteBoundingBoxOffsets const pipetteHasTip = tipState.pipettes[pipetteId] const tipLength = pipetteHasTip ? labwareEntities[tipRackId].def.parameters.tipLength ?? 0 From 83621084640672405cb95a932c53fc5e80d89fc4 Mon Sep 17 00:00:00 2001 From: Jethary Date: Tue, 23 Apr 2024 11:30:53 -0400 Subject: [PATCH 2/8] add more logic --- .../js/helpers/flexAdjacentSlotGetters.ts | 0 .../js/helpers/getFlexSurroundingSlots.ts | 54 ++++ shared-data/js/helpers/index.ts | 1 + shared-data/js/types.ts | 2 +- .../src/utils/ninetySixChannelCollision.ts | 277 ++++++++++++------ 5 files changed, 250 insertions(+), 84 deletions(-) delete mode 100644 shared-data/js/helpers/flexAdjacentSlotGetters.ts create mode 100644 shared-data/js/helpers/getFlexSurroundingSlots.ts diff --git a/shared-data/js/helpers/flexAdjacentSlotGetters.ts b/shared-data/js/helpers/flexAdjacentSlotGetters.ts deleted file mode 100644 index e69de29bb2d..00000000000 diff --git a/shared-data/js/helpers/getFlexSurroundingSlots.ts b/shared-data/js/helpers/getFlexSurroundingSlots.ts new file mode 100644 index 00000000000..1f3d97ba512 --- /dev/null +++ b/shared-data/js/helpers/getFlexSurroundingSlots.ts @@ -0,0 +1,54 @@ +import type { DeckSlotId } from '../types' + +const FLEX_GRID = [ + ['A1', 'A2', 'A3'], + ['B1', 'B2', 'B3'], + ['C1', 'C2', 'C3'], + ['D1', 'D2', 'D3'], +] +const LETTER_TO_ROW_MAP: Record = { + A: 1, + B: 2, + C: 3, + D: 4, +} +const ROWS = 4 +const COLS = 4 +const DIRECTIONS = [ + [-1, -1], + [-1, 0], + [-1, 1], // NW, N, NE + [0, -1], + [0, 1], // W, E + [1, -1], + [1, 0], + [1, 1], // SW, S, SE +] + +// get all surrounding slots for a Flex +export const getFlexSurroundingSlots = ( + slot: DeckSlotId, + stagingAreaSlots: DeckSlotId[] +): DeckSlotId[] => { + // account for staging area slots + for (let i = 0; i < FLEX_GRID.length; i++) { + FLEX_GRID[i].push(stagingAreaSlots[i]) + } + + const col = parseInt(slot.charAt(1)) - 1 + const letter = slot.charAt(0) + const row = LETTER_TO_ROW_MAP[letter] + + const surroungingSlots = [] + + for (let [dRow, dCol] of DIRECTIONS) { + const newRow = row + dRow + const newCol = col + dCol + + if (newRow >= 0 && newRow < ROWS && newCol >= 0 && newCol < COLS) { + surroungingSlots.push(FLEX_GRID[newRow][newCol]) + } + } + + return surroungingSlots +} diff --git a/shared-data/js/helpers/index.ts b/shared-data/js/helpers/index.ts index a07d10472f6..791fa1f5db1 100644 --- a/shared-data/js/helpers/index.ts +++ b/shared-data/js/helpers/index.ts @@ -27,6 +27,7 @@ export * from './getLoadedLabwareDefinitionsByUri' export * from './getOccludedSlotCountForModule' export * from './labwareInference' export * from './getAddressableAreasInProtocol' +export * from './getFlexSurroundingSlots' export * from './getSimplestFlexDeckConfig' export * from './formatRunTimeParameterDefaultValue' export * from './formatRunTimeParameterValue' diff --git a/shared-data/js/types.ts b/shared-data/js/types.ts index 4d51f992f22..9aa960f9ad5 100644 --- a/shared-data/js/types.ts +++ b/shared-data/js/types.ts @@ -284,7 +284,7 @@ export interface CutoutFixture { height: number } -type AreaType = +export type AreaType = | 'slot' | 'movableTrash' | 'wasteChute' diff --git a/step-generation/src/utils/ninetySixChannelCollision.ts b/step-generation/src/utils/ninetySixChannelCollision.ts index 9a1ff1e935f..ad0317b2542 100644 --- a/step-generation/src/utils/ninetySixChannelCollision.ts +++ b/step-generation/src/utils/ninetySixChannelCollision.ts @@ -1,15 +1,30 @@ -import toNumber from 'lodash/toNumber' -import { NozzleConfigurationStyle, getModuleDef2 } from '@opentrons/shared-data' -import type { RobotState, InvariantContext, PipetteEntity } from '../types' +import { + FLEX_ROBOT_TYPE, + THERMOCYCLER_MODULE_TYPE, + getAddressableAreaFromSlotId, + getDeckDefFromRobotType, + getFlexSurroundingSlots, + getModuleDef2, + getPositionFromSlotId, +} from '@opentrons/shared-data' +import type { + AddressableArea, + CoordinateTuple, + NozzleConfigurationStyle, +} from '@opentrons/shared-data' +import type { RobotState, InvariantContext, PipetteEntity, ModuleEntities } from '../types' -const SAFETY_MARGIN = 10 -const targetNumbers = ['2', '3', '4'] const A12_column_front_left_bound = { x: -11.03, y: 2 } const A12_column_back_right_bound = { x: 526.77, y: 506.2 } const PRIMARY_NOZZLE = 'A12' +interface SlotInfo { + addressableArea: AddressableArea | null + position: CoordinateTuple | null +} type Point = { x: number; y: number; z?: number } +// check if nozzle(s) are inbounds const getIsWithinPipetteExtents = ( pipetteEntity: PipetteEntity, location: Point, @@ -36,6 +51,7 @@ const getIsWithinPipetteExtents = ( } } +// return pipette bounds at a sepcific position const getPipetteBoundsAtSpecifiedMoveToPosition = ( primaryNozzle: string, pipetteEntity: PipetteEntity, @@ -85,20 +101,157 @@ const getPipetteBoundsAtSpecifiedMoveToPosition = ( return [backLeftBound, frontRightBound, backRightBound, frontLeftBound] } +// return whether the two provided rectangles are overlapping in the 2d space. +const hasOverlappingRectangles = ( + rectangle1: Point[], + rectangle2: Point[] +): boolean => { + const xCoords = [ + rectangle1[0].x, + rectangle1[1].x, + rectangle2[0].x, + rectangle2[1].x, + ] + const xLengthRect1 = Math.abs(rectangle1[1].x - rectangle1[0].x) + const xLengthRect2 = Math.abs((rectangle2[1].x = rectangle2[0].x)) + const overlappingInX = + Math.abs(Math.max(...xCoords) - Math.min(...xCoords)) < + xLengthRect1 + xLengthRect2 + const yCoordinates = [ + rectangle1[0].y, + rectangle1[1].y, + rectangle2[0].y, + rectangle2[1].y, + ] + const yLengthRect1 = Math.abs(rectangle1[1].y - rectangle1[0].y) + const yLengthRect2 = Math.abs(rectangle2[1].y - rectangle2[0].y) + const overlappingInY = + Math.abs(Math.max(...yCoordinates) - Math.min(...yCoordinates)) < + yLengthRect1 + yLengthRect2 + return overlappingInX && overlappingInY +} + +// check the highest Z-point of all items stacked given a deck slot (including modules, +// adapters, and modules on adapters) +const getHighestZInSlot = ( + robotState: RobotState, + invariantContext: InvariantContext, + // the labware on the top spot + labwareSlot: string +): number => { + const { modules, labware } = robotState + const { moduleEntities, labwareEntities } = invariantContext + if (modules[labwareSlot] != null) { + const moduleDimensions = getModuleDef2(moduleEntities[labwareSlot].model) + .dimensions + return ( + // labware + module + labwareEntities[labwareSlot].def.dimensions.zDimension + + moduleDimensions.bareOverallHeight + + (moduleDimensions.lidHeight ?? 0) + ) + } else if (labware[labwareSlot] != null) { + const adapterSlot = labware[labwareSlot].slot + if (modules[adapterSlot] != null) { + const moduleDimensions = getModuleDef2(moduleEntities[adapterSlot].model) + .dimensions + return ( + // labware + adapter + module + labwareEntities[labwareSlot].def.dimensions.zDimension + + labwareEntities[adapterSlot].def.dimensions.zDimension + + moduleDimensions.bareOverallHeight + + (moduleDimensions.lidHeight ?? 0) + ) + } else { + return ( + // labware + adapter + labwareEntities[labwareSlot].def.dimensions.zDimension + + labwareEntities[adapterSlot].def.dimensions.zDimension + ) + } + } else { + // labware + return labwareEntities[labwareSlot].def.dimensions.zDimension + } +} + +// check if the slot overlaps with the pipette position +const slotHasPotentialCollidingObject = ( + pipetteBounds: Point[], + slotInfo: SlotInfo[], + robotState: RobotState, + invariantContext: InvariantContext, + labwareSlot: string +): boolean => { + for (let slot of slotInfo) { + const slotBounds = slot.addressableArea?.boundingBox + const slotPosition = slot.position + + // If slotPosition or slotBounds is null, continue to the next iteration + if (slotPosition == null || slotBounds == null) { + continue + } + + const backLeftCoords = { + x: slotBounds.xDimension, + y: slotBounds.yDimension, + z: slotBounds.zDimension, + } + const frontRightCoords = { + x: slotPosition[0], + y: slotPosition[1], + z: slotPosition[2], + } + + // Check for overlapping rectangles and pipette z-coordinate if slot overlaps with pipette bounds + if ( + hasOverlappingRectangles( + [pipetteBounds[0], pipetteBounds[1]], + [backLeftCoords, frontRightCoords] + ) && + pipetteBounds[0].z != null + ) { + const highestZInSlot = getHighestZInSlot( + robotState, + invariantContext, + labwareSlot + ) + + if (highestZInSlot >= pipetteBounds[0]?.z) { + return true + } + } + } + return false +} + +const getWillCollideWithThermocyclerLid = (pipetteBounds: Point[], slotInfos: SlotInfo[], moduleEntities: ModuleEntities): boolean => { + const slotIds = slotInfos.map(slot => slot.addressableArea?.id) +if (slotIds.includes('A1') && Object.values(moduleEntities).find(module => module.type === THERMOCYCLER_MODULE_TYPE)) +} + +// util to use in step-generation for if the pipette movement is safe export const getIsSafePipetteMovement = ( robotState: RobotState, invariantContext: InvariantContext, pipetteId: string, destLabwareId: string, tipRackId: string, - destWellName: string, destWellLocation: { origin: string offset: { x: number; y: number; z: number } } ): boolean => { - const { pipetteEntities, labwareEntities } = invariantContext - const { labware: labwareState, pipettes, tipState } = robotState + const deckDefinition = getDeckDefFromRobotType(FLEX_ROBOT_TYPE) + const { + pipetteEntities, + labwareEntities, + additionalEquipmentEntities, + } = invariantContext + const { labware: labwareState, pipettes, tipState, modules } = robotState + const stagingAreaSlots = Object.values(additionalEquipmentEntities) + .filter(ae => ae.name === 'stagingArea') + .map(stagingArea => stagingArea.location as string) const pipetteEntity = pipetteEntities[pipetteId] const pipetteHasTip = tipState.pipettes[pipetteId] const tipLength = pipetteHasTip @@ -127,88 +280,46 @@ export const getIsSafePipetteMovement = ( return false } else { const labwareSlot = labwareState[destLabwareId].slot + let deckSlot = labwareSlot + if (modules[labwareSlot] != null) { + deckSlot = modules[labwareSlot].slot + } else if (labwareState[labwareSlot] != null) { + const adapterSlot = labwareState[labwareSlot].slot + const adapterInModuleSlot = + modules[adapterSlot] != null ? modules[adapterSlot].slot : null + if (adapterInModuleSlot != null) { + deckSlot = adapterInModuleSlot + } else { + deckSlot = adapterSlot + } + } + const pipetteBoundsAtWellLocation = getPipetteBoundsAtSpecifiedMoveToPosition( PRIMARY_NOZZLE, pipetteEntity, tipLength, destWellLocation.offset ) - const surroundingSlots = [] - } -} -export const getIsTallLabwareWestOf96Channel = ( - robotState: RobotState, - invariantContext: InvariantContext, - sourceLabwareId: string, - pipetteId: string, - tipRackId: string -): boolean => { - const { - labwareEntities, - additionalEquipmentEntities, - pipetteEntities, - } = invariantContext - const { labware: labwareState, tipState } = robotState - const test = pipetteEntities[pipetteId].spec.pipetteBoundingBoxOffsets - const pipetteHasTip = tipState.pipettes[pipetteId] - const tipLength = pipetteHasTip - ? labwareEntities[tipRackId].def.parameters.tipLength ?? 0 - : 0 - // early exit if source labware is the waste chute or trash bin - if (additionalEquipmentEntities[sourceLabwareId] != null) { - return false - } - - const labwareSlot = labwareState[sourceLabwareId].slot - const letter = labwareSlot.charAt(0) - const number = labwareSlot.charAt(1) - - if (targetNumbers.includes(number)) { - const westNumber = toNumber(number) - 1 - const westSlot = letter + westNumber - - const westLabwareState = Object.entries(labwareState).find( - ([id, labware]) => labware.slot === westSlot + const surroundingSlots = getFlexSurroundingSlots( + labwareSlot, + stagingAreaSlots ) - if (westLabwareState != null) { - const westLabwareId = westLabwareState[0] - if (labwareEntities[westLabwareId] == null) { - console.error( - `expected to find labware west of source labware but could not, with labware id ${westLabwareId}` - ) - } - if (labwareEntities[westLabwareId] != null) { - const westLabwareHeight = - labwareEntities[westLabwareId].def.dimensions.zDimension - const westLabwareSlot = robotState.labware[westLabwareId].slot - let adapterHeight: number = 0 - let moduleHeight: number = 0 - // if labware is on an adapter + or on an adapter + module - if (robotState.labware[westLabwareSlot] != null) { - const adapterSlot = robotState.labware[westLabwareSlot]?.slot - adapterHeight = - invariantContext.labwareEntities[westLabwareSlot]?.def.dimensions - .zDimension - const moduleModel = - invariantContext.moduleEntities[adapterSlot]?.model - const moduleDimensions = - moduleModel != null ? getModuleDef2(moduleModel)?.dimensions : null - moduleHeight = - moduleDimensions != null ? moduleDimensions.bareOverallHeight : 0 - // if labware is on a module - } else if (invariantContext.moduleEntities[westLabwareSlot] != null) { - const moduleModel = - invariantContext.moduleEntities[westLabwareSlot].model - moduleHeight = getModuleDef2(moduleModel).dimensions.bareOverallHeight - } - const totalHighestZ = westLabwareHeight + adapterHeight + moduleHeight - const sourceLabwareHeight = - labwareEntities[sourceLabwareId].def.dimensions.zDimension - - return totalHighestZ + SAFETY_MARGIN > sourceLabwareHeight + tipLength + const slotInfos: SlotInfo[] = surroundingSlots.map(slot => { + const addressableArea = getAddressableAreaFromSlotId(slot, deckDefinition) + const position = getPositionFromSlotId(slot, deckDefinition) + + return { + addressableArea, + position, } - } + }) + // TODO - still need todo the thermocycler collision stuff + return slotHasPotentialCollidingObject( + pipetteBoundsAtWellLocation, + slotInfos, + robotState, + invariantContext, + deckSlot + ) } - - return false } From 659f0674602cef0665167b5fb1e08bee4a64ee8e Mon Sep 17 00:00:00 2001 From: Jethary Date: Tue, 23 Apr 2024 16:21:02 -0400 Subject: [PATCH 3/8] feat(step-generation, shared-data): pipette collision warnings closes AUTH-19 --- .../__tests__/getFlexSurroundingSlots.test.ts | 30 +++ .../js/helpers/getFlexSurroundingSlots.ts | 55 ++--- .../getIsSafePipetteMovement.test.ts | 123 +++++++++++ .../ninetySixChannelCollision.test.ts | 146 ------------- .../src/commandCreators/atomic/replaceTip.ts | 17 +- .../commandCreators/compound/consolidate.ts | 82 ++++---- .../commandCreators/compound/distribute.ts | 73 +++---- .../src/commandCreators/compound/mix.ts | 34 +-- .../src/commandCreators/compound/transfer.ts | 75 +++---- step-generation/src/errorCreators.ts | 10 +- step-generation/src/types.ts | 2 +- step-generation/src/utils/index.ts | 2 +- ...elCollision.ts => safePipetteMovements.ts} | 193 ++++++++++++------ 13 files changed, 445 insertions(+), 397 deletions(-) create mode 100644 shared-data/js/helpers/__tests__/getFlexSurroundingSlots.test.ts create mode 100644 step-generation/src/__tests__/getIsSafePipetteMovement.test.ts delete mode 100644 step-generation/src/__tests__/ninetySixChannelCollision.test.ts rename step-generation/src/utils/{ninetySixChannelCollision.ts => safePipetteMovements.ts} (67%) diff --git a/shared-data/js/helpers/__tests__/getFlexSurroundingSlots.test.ts b/shared-data/js/helpers/__tests__/getFlexSurroundingSlots.test.ts new file mode 100644 index 00000000000..a91d2f737c5 --- /dev/null +++ b/shared-data/js/helpers/__tests__/getFlexSurroundingSlots.test.ts @@ -0,0 +1,30 @@ +import { describe, it, expect } from 'vitest' +import { getFlexSurroundingSlots } from '../getFlexSurroundingSlots' + +describe('getFlexSurroundingSlots', () => { + it('returns slots when slot is D2', () => { + const results = getFlexSurroundingSlots('D2', []) + expect(results).toStrictEqual(['C1', 'C2', 'C3', 'D1', 'D3']) + }) + it('returns slots when selected is a center slot', () => { + const results = getFlexSurroundingSlots('C2', []) + expect(results).toStrictEqual([ + 'B1', + 'B2', + 'B3', + 'C1', + 'C3', + 'D1', + 'D2', + 'D3', + ]) + }) + it('returns slots when selected is a column 3 with staging areas present', () => { + const results = getFlexSurroundingSlots('B3', ['A4']) + expect(results).toStrictEqual(['A2', 'A3', 'A4', 'B2', 'C2', 'C3']) + }) + it('returns slots when selected is a corner, A1', () => { + const results = getFlexSurroundingSlots('A1', ['A4']) + expect(results).toStrictEqual(['A2', 'B1', 'B2']) + }) +}) diff --git a/shared-data/js/helpers/getFlexSurroundingSlots.ts b/shared-data/js/helpers/getFlexSurroundingSlots.ts index 1f3d97ba512..9900cee9880 100644 --- a/shared-data/js/helpers/getFlexSurroundingSlots.ts +++ b/shared-data/js/helpers/getFlexSurroundingSlots.ts @@ -6,49 +6,58 @@ const FLEX_GRID = [ ['C1', 'C2', 'C3'], ['D1', 'D2', 'D3'], ] + const LETTER_TO_ROW_MAP: Record = { - A: 1, - B: 2, - C: 3, - D: 4, + A: 0, + B: 1, + C: 2, + D: 3, } + +let COLS = 3 // Initial number of columns in each row const ROWS = 4 -const COLS = 4 + const DIRECTIONS = [ - [-1, -1], - [-1, 0], - [-1, 1], // NW, N, NE - [0, -1], - [0, 1], // W, E - [1, -1], - [1, 0], - [1, 1], // SW, S, SE + [-1, -1], // NW + [-1, 0], // N + [-1, 1], // NE + [0, -1], // W + [0, 1], // E + [1, -1], // SW + [1, 0], // S + [1, 1], // SE ] -// get all surrounding slots for a Flex export const getFlexSurroundingSlots = ( slot: DeckSlotId, stagingAreaSlots: DeckSlotId[] ): DeckSlotId[] => { - // account for staging area slots - for (let i = 0; i < FLEX_GRID.length; i++) { - FLEX_GRID[i].push(stagingAreaSlots[i]) + // Handle staging area slots + if (stagingAreaSlots.length > 0) { + stagingAreaSlots.forEach((stagingSlot, index) => { + if (stagingSlot) { + FLEX_GRID[index].push(stagingSlot) + } + }) + COLS = Math.max(COLS, FLEX_GRID[0].length) // Update COLS to the maximum row length } - const col = parseInt(slot.charAt(1)) - 1 const letter = slot.charAt(0) + const col = parseInt(slot.charAt(1)) - 1 // Convert the column to a 0-based index const row = LETTER_TO_ROW_MAP[letter] - const surroungingSlots = [] + const surroundingSlots: DeckSlotId[] = [] - for (let [dRow, dCol] of DIRECTIONS) { + // Iterate through both directions + DIRECTIONS.forEach(([dRow, dCol]) => { const newRow = row + dRow const newCol = col + dCol if (newRow >= 0 && newRow < ROWS && newCol >= 0 && newCol < COLS) { - surroungingSlots.push(FLEX_GRID[newRow][newCol]) + surroundingSlots.push(FLEX_GRID[newRow][newCol]) } - } + }) - return surroungingSlots + // Filter out any undefined values from the staging area slots that are not added + return surroundingSlots.filter(slot => slot !== undefined) } diff --git a/step-generation/src/__tests__/getIsSafePipetteMovement.test.ts b/step-generation/src/__tests__/getIsSafePipetteMovement.test.ts new file mode 100644 index 00000000000..5d6459d3487 --- /dev/null +++ b/step-generation/src/__tests__/getIsSafePipetteMovement.test.ts @@ -0,0 +1,123 @@ +import { expect, describe, it } from 'vitest' +import { getIsSafePipetteMovement } from '../utils' +import { + LabwareDefinition2, + TEMPERATURE_MODULE_TYPE, + TEMPERATURE_MODULE_V2, + fixture96Plate, + fixtureP100096V2Specs, + fixtureTiprack1000ul, +} from '@opentrons/shared-data' +import { InvariantContext, RobotState } from '../types' + +const mockLabwareId = 'labwareId' +const mockPipId = 'pip' +const mockTiprackId = 'tiprackId' +const mockModule = 'moduleId' +const mockInvariantProperties: InvariantContext = { + pipetteEntities: { + pip: { + name: 'p1000_96', + id: 'pip', + tiprackDefURI: ['mockDefUri'], + tiprackLabwareDef: [fixtureTiprack1000ul as LabwareDefinition2], + spec: fixtureP100096V2Specs, + }, + }, + labwareEntities: { + [mockLabwareId]: { + id: mockLabwareId, + labwareDefURI: 'mockDefUri', + def: fixture96Plate as LabwareDefinition2, + }, + [mockTiprackId]: { + id: mockTiprackId, + labwareDefURI: 'mockTipUri', + def: fixtureTiprack1000ul as LabwareDefinition2, + }, + }, + moduleEntities: {}, + additionalEquipmentEntities: {}, + config: { + OT_PD_DISABLE_MODULE_RESTRICTIONS: false, + }, +} + +const mockRobotState: RobotState = { + pipettes: { pip: { mount: 'left' } }, + labware: { [mockLabwareId]: { slot: 'D2' }, [mockTiprackId]: { slot: 'A2' } }, + modules: {}, + tipState: { tipracks: {}, pipettes: {} }, + liquidState: { pipettes: {}, labware: {}, additionalEquipment: {} }, +} +describe('getIsSafePipetteMovement', () => { + it('returns true when the labware id is a trash bin', () => { + const result = getIsSafePipetteMovement( + { + labware: {}, + pipettes: {}, + modules: {}, + tipState: {}, + liquidState: {}, + } as any, + { + labwareEntities: {}, + pipetteEntities: {}, + moduleEntities: {}, + additionalEquipmentEntities: { + trashBin: { name: 'trashBin', location: 'A3', id: 'trashBin' }, + }, + config: {} as any, + }, + 'mockId', + 'mockTrashBin', + 'mockTiprackId', + { x: 0, y: 0, z: 0 } + ) + expect(result).toEqual(true) + }) + it('returns false when within pipette extents is false', () => { + const result = getIsSafePipetteMovement( + mockRobotState, + mockInvariantProperties, + mockPipId, + mockLabwareId, + mockTiprackId, + { x: -12, y: -100, z: 20 } + ) + expect(result).toEqual(false) + }) + it('returns false when slot has a module near it', () => { + mockRobotState.modules = { + [mockModule]: { slot: 'D1', moduleState: {} as any }, + } + mockInvariantProperties.moduleEntities = { + [mockModule]: { + id: mockModule, + type: TEMPERATURE_MODULE_TYPE, + model: TEMPERATURE_MODULE_V2, + }, + } + const result = getIsSafePipetteMovement( + mockRobotState, + mockInvariantProperties, + mockPipId, + mockLabwareId, + mockTiprackId, + { x: -1, y: 5, z: 20 } + ) + expect(result).toEqual(false) + }) + // todo(jr, 4/23/24): add more test cases, test thermocycler collision, collision with tip attached + // it.only('returns true when there are no collisions!', () => { + // const result = getIsSafePipetteMovement( + // mockRobotState, + // mockInvariantProperties, + // mockPipId, + // mockLabwareId, + // mockTiprackId, + // { x: 0, y: 0, z: 0 } + // ) + // expect(result).toEqual(true) + // }) +}) diff --git a/step-generation/src/__tests__/ninetySixChannelCollision.test.ts b/step-generation/src/__tests__/ninetySixChannelCollision.test.ts deleted file mode 100644 index aae8c8acab9..00000000000 --- a/step-generation/src/__tests__/ninetySixChannelCollision.test.ts +++ /dev/null @@ -1,146 +0,0 @@ -import { beforeEach, describe, it, expect } from 'vitest' -import { getIsTallLabwareWestOf96Channel } from '../utils/ninetySixChannelCollision' -import type { LabwareDefinition2 } from '@opentrons/shared-data' -import type { RobotState, InvariantContext } from '../types' - -let invariantContext: InvariantContext -let robotState: RobotState - -const mockSourceId = 'sourceId' -const mockWestId = 'westId' -const mockPipetteId = 'pipetteId' -const mockTiprackId = 'tiprackId' -const mockSourceDef: LabwareDefinition2 = { - dimensions: { zDimension: 100 }, -} as any -const mockWestDef: LabwareDefinition2 = { - dimensions: { zDimension: 90 }, -} as any -const mockWestDefTall: LabwareDefinition2 = { - dimensions: { zDimension: 101 }, -} as any -const mockTiprackDefinition: LabwareDefinition2 = { - parameters: { tipLength: 10 }, -} as any -describe('getIsTallLabwareWestOf96Channel ', () => { - beforeEach(() => { - invariantContext = { - labwareEntities: { - [mockSourceId]: { - id: mockSourceId, - labwareDefURI: 'mockDefUri', - def: mockSourceDef, - }, - }, - additionalEquipmentEntities: {}, - moduleEntities: {}, - config: {} as any, - pipetteEntities: { - [mockPipetteId]: { - name: 'p1000_96', - id: mockPipetteId, - tiprackDefURI: ['mockUri'], - tiprackLabwareDef: [mockTiprackDefinition], - spec: {} as any, - }, - }, - } - robotState = { - labware: { [mockSourceId]: { slot: 'A1' } }, - pipettes: {}, - modules: {}, - tipState: { pipettes: { [mockPipetteId]: false } } as any, - liquidState: {} as any, - } - }) - it('should return false when the slot is in column is 1', () => { - expect( - getIsTallLabwareWestOf96Channel( - robotState, - invariantContext, - mockSourceId, - mockPipetteId, - mockTiprackId - ) - ).toBe(false) - }) - it('should return false when source id is a waste chute', () => { - invariantContext = { - ...invariantContext, - additionalEquipmentEntities: { - [mockSourceId]: { - id: mockSourceId, - name: 'wasteChute', - location: 'D3', - }, - }, - } - expect( - getIsTallLabwareWestOf96Channel( - robotState, - invariantContext, - mockSourceId, - mockPipetteId, - mockTiprackId - ) - ).toBe(false) - }) - it('should return false when there is no labware west of source labware', () => { - robotState.labware = { [mockSourceId]: { slot: 'A2' } } - expect( - getIsTallLabwareWestOf96Channel( - robotState, - invariantContext, - mockSourceId, - mockPipetteId, - mockTiprackId - ) - ).toBe(false) - }) - it('should return false when the west labware height is not tall enough', () => { - invariantContext.labwareEntities = { - ...invariantContext.labwareEntities, - [mockWestId]: { - id: mockWestId, - labwareDefURI: 'mockDefUri', - def: mockWestDef, - }, - } - robotState.labware = { - [mockSourceId]: { slot: 'A2' }, - [mockWestId]: { slot: 'A1' }, - } - expect( - getIsTallLabwareWestOf96Channel( - robotState, - invariantContext, - mockSourceId, - mockPipetteId, - mockTiprackId - ) - ).toBe(false) - }) - it('should return true when the west labware height is tall enough', () => { - invariantContext.labwareEntities = { - ...invariantContext.labwareEntities, - [mockWestId]: { - id: mockWestId, - labwareDefURI: 'mockDefUri', - def: mockWestDefTall, - }, - } - robotState.labware = { - [mockSourceId]: { slot: 'A2' }, - [mockWestId]: { slot: 'A1' }, - } - expect( - getIsTallLabwareWestOf96Channel( - robotState, - invariantContext, - mockSourceId, - mockPipetteId, - mockTiprackId - ) - ).toBe(true) - }) -}) diff --git a/step-generation/src/commandCreators/atomic/replaceTip.ts b/step-generation/src/commandCreators/atomic/replaceTip.ts index 85160be713c..7aae3b98be1 100644 --- a/step-generation/src/commandCreators/atomic/replaceTip.ts +++ b/step-generation/src/commandCreators/atomic/replaceTip.ts @@ -7,7 +7,7 @@ import { curryCommandCreator, getIsHeaterShakerEastWestMultiChannelPipette, getIsHeaterShakerEastWestWithLatchOpen, - getIsTallLabwareWestOf96Channel, + getIsSafePipetteMovement, getLabwareSlot, modulePipetteCollision, pipetteAdjacentHeaterShakerWhileShaking, @@ -160,23 +160,18 @@ export const replaceTip: CommandCreator = ( if ( channels === 96 && nozzles === COLUMN && - getIsTallLabwareWestOf96Channel( + !getIsSafePipetteMovement( prevRobotState, invariantContext, nextTiprack.tiprackId, pipette, - tipRack + tipRack, + // we don't adjust the offset when moving to the tiprack + { x: 0, y: 0 } ) ) { return { - errors: [ - errorCreators.tallLabwareWestOf96ChannelPipetteLabware({ - source: 'tiprack', - labware: - invariantContext.labwareEntities[nextTiprack.tiprackId].def.metadata - .displayName, - }), - ], + errors: [errorCreators.possiblePipetteCollision()], } } diff --git a/step-generation/src/commandCreators/compound/consolidate.ts b/step-generation/src/commandCreators/compound/consolidate.ts index b37f2ede1b0..f7fc4c85f9d 100644 --- a/step-generation/src/commandCreators/compound/consolidate.ts +++ b/step-generation/src/commandCreators/compound/consolidate.ts @@ -18,7 +18,7 @@ import { airGapHelper, dispenseLocationHelper, moveHelper, - getIsTallLabwareWestOf96Channel, + getIsSafePipetteMovement, getWasteChuteAddressableAreaNamePip, } from '../../utils' import { @@ -56,6 +56,31 @@ export const consolidate: CommandCreator = ( * 'once': get a new tip at the beginning of the consolidate step, and use it throughout * 'never': reuse the tip from the last step */ + + // TODO: BC 2019-07-08 these argument names are a bit misleading, instead of being values bound + // to the action of aspiration of dispensing in a given command, they are actually values bound + // to a given labware associated with a command (e.g. Source, Destination). For this reason we + // currently remapping the inner mix values. Those calls to mixUtil should become easier to read + // when we decide to rename these fields/args... probably all the way up to the UI level. + const { + aspirateDelay, + aspirateFlowRateUlSec, + aspirateOffsetFromBottomMm, + blowoutFlowRateUlSec, + blowoutOffsetFromTopMm, + dispenseAirGapVolume, + dispenseDelay, + dispenseFlowRateUlSec, + dispenseOffsetFromBottomMm, + mixFirstAspirate, + mixInDestination, + dropTipLocation, + aspirateXOffset, + aspirateYOffset, + dispenseXOffset, + dispenseYOffset, + } = args + const actionName = 'consolidate' const pipetteData = prevRobotState.pipettes[args.pipette] const is96Channel = @@ -91,72 +116,37 @@ export const consolidate: CommandCreator = ( if ( is96Channel && args.nozzles === COLUMN && - getIsTallLabwareWestOf96Channel( + !getIsSafePipetteMovement( prevRobotState, invariantContext, - args.sourceLabware, args.pipette, - args.tipRack + args.sourceLabware, + args.tipRack, + { x: aspirateXOffset, y: aspirateYOffset } ) ) { return { - errors: [ - errorCreators.tallLabwareWestOf96ChannelPipetteLabware({ - source: 'aspirate', - labware: - invariantContext.labwareEntities[args.sourceLabware].def.metadata - .displayName, - }), - ], + errors: [errorCreators.possiblePipetteCollision()], } } if ( is96Channel && args.nozzles === COLUMN && - getIsTallLabwareWestOf96Channel( + !getIsSafePipetteMovement( prevRobotState, invariantContext, - args.destLabware, args.pipette, - args.tipRack + args.destLabware, + args.tipRack, + { x: dispenseXOffset, y: dispenseYOffset } ) ) { return { - errors: [ - errorCreators.tallLabwareWestOf96ChannelPipetteLabware({ - source: 'dispense', - labware: - invariantContext.labwareEntities[args.destLabware].def.metadata - .displayName, - }), - ], + errors: [errorCreators.possiblePipetteCollision()], } } - // TODO: BC 2019-07-08 these argument names are a bit misleading, instead of being values bound - // to the action of aspiration of dispensing in a given command, they are actually values bound - // to a given labware associated with a command (e.g. Source, Destination). For this reason we - // currently remapping the inner mix values. Those calls to mixUtil should become easier to read - // when we decide to rename these fields/args... probably all the way up to the UI level. - const { - aspirateDelay, - aspirateFlowRateUlSec, - aspirateOffsetFromBottomMm, - blowoutFlowRateUlSec, - blowoutOffsetFromTopMm, - dispenseAirGapVolume, - dispenseDelay, - dispenseFlowRateUlSec, - dispenseOffsetFromBottomMm, - mixFirstAspirate, - mixInDestination, - dropTipLocation, - aspirateXOffset, - aspirateYOffset, - dispenseXOffset, - dispenseYOffset, - } = args const aspirateAirGapVolume = args.aspirateAirGapVolume || 0 const maxWellsPerChunk = Math.floor( getPipetteWithTipMaxVol(args.pipette, invariantContext, args.tipRack) / diff --git a/step-generation/src/commandCreators/compound/distribute.ts b/step-generation/src/commandCreators/compound/distribute.ts index 520ce06aeb4..eae11c1452f 100644 --- a/step-generation/src/commandCreators/compound/distribute.ts +++ b/step-generation/src/commandCreators/compound/distribute.ts @@ -16,7 +16,7 @@ import { blowoutUtil, wasteChuteCommandsUtil, getDispenseAirGapLocation, - getIsTallLabwareWestOf96Channel, + getIsSafePipetteMovement, getWasteChuteAddressableAreaNamePip, } from '../../utils' import { @@ -53,6 +53,26 @@ export const distribute: CommandCreator = ( * 'once': get a new tip at the beginning of the distribute step, and use it throughout * 'never': reuse the tip from the last step */ + + // TODO: BC 2019-07-08 these argument names are a bit misleading, instead of being values bound + // to the action of aspiration of dispensing in a given command, they are actually values bound + // to a given labware associated with a command (e.g. Source, Destination). For this reason we + // currently remapping the inner mix values. Those calls to mixUtil should become easier to read + // when we decide to rename these fields/args... probably all the way up to the UI level. + const { + aspirateDelay, + aspirateFlowRateUlSec, + aspirateOffsetFromBottomMm, + dispenseDelay, + dispenseFlowRateUlSec, + dispenseOffsetFromBottomMm, + blowoutLocation, + aspirateXOffset, + aspirateYOffset, + dispenseXOffset, + dispenseYOffset, + } = args + // TODO Ian 2018-05-03 next ~20 lines match consolidate.js const actionName = 'distribute' const errors: CommandCreatorError[] = [] @@ -91,67 +111,38 @@ export const distribute: CommandCreator = ( if ( is96Channel && args.nozzles === COLUMN && - getIsTallLabwareWestOf96Channel( + !getIsSafePipetteMovement( prevRobotState, invariantContext, - args.sourceLabware, args.pipette, - args.tipRack + args.sourceLabware, + args.tipRack, + { x: aspirateXOffset, y: aspirateYOffset } ) ) { - errors.push( - errorCreators.tallLabwareWestOf96ChannelPipetteLabware({ - source: 'aspirate', - labware: - invariantContext.labwareEntities[args.sourceLabware].def.metadata - .displayName, - }) - ) + errors.push(errorCreators.possiblePipetteCollision()) } if ( is96Channel && args.nozzles === COLUMN && - getIsTallLabwareWestOf96Channel( + !getIsSafePipetteMovement( prevRobotState, invariantContext, - args.destLabware, args.pipette, - args.tipRack + args.destLabware, + args.tipRack, + { x: dispenseXOffset, y: dispenseYOffset } ) ) { - errors.push( - errorCreators.tallLabwareWestOf96ChannelPipetteLabware({ - source: 'dispense', - labware: - invariantContext.labwareEntities[args.destLabware].def.metadata - .displayName, - }) - ) + errors.push(errorCreators.possiblePipetteCollision()) } if (errors.length > 0) return { errors, } - // TODO: BC 2019-07-08 these argument names are a bit misleading, instead of being values bound - // to the action of aspiration of dispensing in a given command, they are actually values bound - // to a given labware associated with a command (e.g. Source, Destination). For this reason we - // currently remapping the inner mix values. Those calls to mixUtil should become easier to read - // when we decide to rename these fields/args... probably all the way up to the UI level. - const { - aspirateDelay, - aspirateFlowRateUlSec, - aspirateOffsetFromBottomMm, - dispenseDelay, - dispenseFlowRateUlSec, - dispenseOffsetFromBottomMm, - blowoutLocation, - aspirateXOffset, - aspirateYOffset, - dispenseXOffset, - dispenseYOffset, - } = args + const aspirateAirGapVolume = args.aspirateAirGapVolume || 0 const dispenseAirGapVolume = args.dispenseAirGapVolume || 0 // TODO error on negative args.disposalVolume? diff --git a/step-generation/src/commandCreators/compound/mix.ts b/step-generation/src/commandCreators/compound/mix.ts index 284529c7c1f..734be8c1a39 100644 --- a/step-generation/src/commandCreators/compound/mix.ts +++ b/step-generation/src/commandCreators/compound/mix.ts @@ -5,7 +5,7 @@ import { blowoutUtil, curryCommandCreator, reduceCommandCreators, - getIsTallLabwareWestOf96Channel, + getIsSafePipetteMovement, } from '../../utils' import * as errorCreators from '../../errorCreators' import { @@ -178,25 +178,29 @@ export const mix: CommandCreator = ( return { errors: [errorCreators.dropTipLocationDoesNotExist()] } } - if ( - is96Channel && - data.nozzles === COLUMN && - getIsTallLabwareWestOf96Channel( + console.log(invariantContext.pipetteEntities[pipette]) + + if (is96Channel && data.nozzles === COLUMN) { + const isAspirateSafePipetteMovement = getIsSafePipetteMovement( prevRobotState, invariantContext, + pipette, labware, + tipRack, + { x: aspirateXOffset, y: aspirateYOffset } + ) + const isDispenseSafePipetteMovement = getIsSafePipetteMovement( + prevRobotState, + invariantContext, pipette, - tipRack + labware, + tipRack, + { x: dispenseXOffset, y: dispenseYOffset } ) - ) { - return { - errors: [ - errorCreators.tallLabwareWestOf96ChannelPipetteLabware({ - source: 'mix', - labware: - invariantContext.labwareEntities[labware].def.metadata.displayName, - }), - ], + if (!isAspirateSafePipetteMovement && !isDispenseSafePipetteMovement) { + return { + errors: [errorCreators.possiblePipetteCollision()], + } } } const stateNozzles = prevRobotState.pipettes[pipette].nozzles diff --git a/step-generation/src/commandCreators/compound/transfer.ts b/step-generation/src/commandCreators/compound/transfer.ts index 2d16c8064bf..9c59d301aa4 100644 --- a/step-generation/src/commandCreators/compound/transfer.ts +++ b/step-generation/src/commandCreators/compound/transfer.ts @@ -18,7 +18,7 @@ import { getTrashOrLabware, dispenseLocationHelper, moveHelper, - getIsTallLabwareWestOf96Channel, + getIsSafePipetteMovement, getWasteChuteAddressableAreaNamePip, } from '../../utils' import { @@ -63,6 +63,27 @@ export const transfer: CommandCreator = ( NOTE: In some situations, different changeTip options have equivalent outcomes. That's OK. */ + // TODO: BC 2019-07-08 these argument names are a bit misleading, instead of being values bound + // to the action of aspiration of dispensing in a given command, they are actually values bound + // to a given labware associated with a command (e.g. Source, Destination). For this reason we + // currently remapping the inner mix values. Those calls to mixUtil should become easier to read + // when we decide to rename these fields/args... probably all the way up to the UI level. + const { + aspirateDelay, + dispenseDelay, + aspirateFlowRateUlSec, + aspirateOffsetFromBottomMm, + blowoutFlowRateUlSec, + blowoutOffsetFromTopMm, + dispenseFlowRateUlSec, + dispenseOffsetFromBottomMm, + tipRack, + aspirateXOffset, + aspirateYOffset, + dispenseXOffset, + dispenseYOffset, + } = args + const trashOrLabware = getTrashOrLabware( invariantContext.labwareEntities, invariantContext.additionalEquipmentEntities, @@ -130,43 +151,31 @@ export const transfer: CommandCreator = ( if ( is96Channel && args.nozzles === COLUMN && - getIsTallLabwareWestOf96Channel( + !getIsSafePipetteMovement( prevRobotState, invariantContext, - args.sourceLabware, args.pipette, - args.tipRack + args.sourceLabware, + args.tipRack, + { x: aspirateXOffset, y: aspirateYOffset, z: aspirateOffsetFromBottomMm } ) ) { - errors.push( - errorCreators.tallLabwareWestOf96ChannelPipetteLabware({ - source: 'aspirate', - labware: - invariantContext.labwareEntities[args.sourceLabware].def.metadata - .displayName, - }) - ) + errors.push(errorCreators.possiblePipetteCollision()) } if ( is96Channel && args.nozzles === COLUMN && - getIsTallLabwareWestOf96Channel( + !getIsSafePipetteMovement( prevRobotState, invariantContext, - args.destLabware, args.pipette, - args.tipRack + args.destLabware, + args.tipRack, + { x: dispenseXOffset, y: dispenseYOffset, z: dispenseOffsetFromBottomMm } ) ) { - errors.push( - errorCreators.tallLabwareWestOf96ChannelPipetteLabware({ - source: 'dispense', - labware: - invariantContext.labwareEntities[args.destLabware].def.metadata - .displayName, - }) - ) + errors.push(errorCreators.possiblePipetteCollision()) } if (errors.length > 0) @@ -190,26 +199,6 @@ export const transfer: CommandCreator = ( pipetteSpec.channels ) - // TODO: BC 2019-07-08 these argument names are a bit misleading, instead of being values bound - // to the action of aspiration of dispensing in a given command, they are actually values bound - // to a given labware associated with a command (e.g. Source, Destination). For this reason we - // currently remapping the inner mix values. Those calls to mixUtil should become easier to read - // when we decide to rename these fields/args... probably all the way up to the UI level. - const { - aspirateDelay, - dispenseDelay, - aspirateFlowRateUlSec, - aspirateOffsetFromBottomMm, - blowoutFlowRateUlSec, - blowoutOffsetFromTopMm, - dispenseFlowRateUlSec, - dispenseOffsetFromBottomMm, - tipRack, - aspirateXOffset, - aspirateYOffset, - dispenseXOffset, - dispenseYOffset, - } = args const aspirateAirGapVolume = args.aspirateAirGapVolume || 0 const dispenseAirGapVolume = args.dispenseAirGapVolume || 0 const effectiveTransferVol = diff --git a/step-generation/src/errorCreators.ts b/step-generation/src/errorCreators.ts index 50a271effe0..581b04d72f9 100644 --- a/step-generation/src/errorCreators.ts +++ b/step-generation/src/errorCreators.ts @@ -175,13 +175,11 @@ export const tallLabwareEastWestOfHeaterShaker = ( } } -export const tallLabwareWestOf96ChannelPipetteLabware = (args: { - source: string - labware: string -}): CommandCreatorError => { +export const possiblePipetteCollision = (): CommandCreatorError => { return { - type: 'TALL_LABWARE_WEST_OF_96_CHANNEL_LABWARE', - message: `Labware to the left of the ${args.source} ${args.labware} is too tall and will collide with the 96-channel.`, + type: 'POSSIBLE_PIPETTE_COLLISION', + message: + 'There is a possibility that the Pipette will collide with the a labware or module on the deck', } } diff --git a/step-generation/src/types.ts b/step-generation/src/types.ts index 6cef80c43ed..e63360a3f27 100644 --- a/step-generation/src/types.ts +++ b/step-generation/src/types.ts @@ -539,9 +539,9 @@ export type ErrorType = | 'PIPETTE_HAS_TIP' | 'PIPETTE_VOLUME_EXCEEDED' | 'PIPETTING_INTO_COLUMN_4' + | 'POSSIBLE_PIPETTE_COLLISION' | 'REMOVE_96_CHANNEL_TIPRACK_ADAPTER' | 'TALL_LABWARE_EAST_WEST_OF_HEATER_SHAKER' - | 'TALL_LABWARE_WEST_OF_96_CHANNEL_LABWARE' | 'THERMOCYCLER_LID_CLOSED' | 'TIP_VOLUME_EXCEEDED' diff --git a/step-generation/src/utils/index.ts b/step-generation/src/utils/index.ts index ac363cbcd97..9c8ab222c57 100644 --- a/step-generation/src/utils/index.ts +++ b/step-generation/src/utils/index.ts @@ -20,6 +20,6 @@ export * from './commandCreatorArgsGetters' export * from './heaterShakerCollision' export * from './misc' export * from './movableTrashCommandsUtil' -export * from './ninetySixChannelCollision' +export * from './safePipetteMovements' export * from './wasteChuteCommandsUtil' export const uuid: () => string = uuidv4 diff --git a/step-generation/src/utils/ninetySixChannelCollision.ts b/step-generation/src/utils/safePipetteMovements.ts similarity index 67% rename from step-generation/src/utils/ninetySixChannelCollision.ts rename to step-generation/src/utils/safePipetteMovements.ts index ad0317b2542..7d4431671e0 100644 --- a/step-generation/src/utils/ninetySixChannelCollision.ts +++ b/step-generation/src/utils/safePipetteMovements.ts @@ -12,17 +12,42 @@ import type { CoordinateTuple, NozzleConfigurationStyle, } from '@opentrons/shared-data' -import type { RobotState, InvariantContext, PipetteEntity, ModuleEntities } from '../types' +import type { + RobotState, + InvariantContext, + PipetteEntity, + ModuleEntities, + LabwareEntity, +} from '../types' const A12_column_front_left_bound = { x: -11.03, y: 2 } const A12_column_back_right_bound = { x: 526.77, y: 506.2 } const PRIMARY_NOZZLE = 'A12' +const FLEX_TC_LID_COLLISION_ZONE = { + back_left: { x: -43.25, y: 454.9, z: 211.91 }, + front_right: { x: 128.75, y: 402, z: 211.91 }, +} +const FLEX_TC_LID_BACK_LEFT_PT = { + x: FLEX_TC_LID_COLLISION_ZONE.back_left.x, + y: FLEX_TC_LID_COLLISION_ZONE.back_left.y, + z: FLEX_TC_LID_COLLISION_ZONE.back_left.z, +} + +const FLEX_TC_LID_FRONT_RIGHT_PT = { + x: FLEX_TC_LID_COLLISION_ZONE.front_right.x, + y: FLEX_TC_LID_COLLISION_ZONE.front_right.y, + z: FLEX_TC_LID_COLLISION_ZONE.front_right.z, +} interface SlotInfo { addressableArea: AddressableArea | null position: CoordinateTuple | null } -type Point = { x: number; y: number; z?: number } +interface Point { + x: number + y: number + z?: number +} // check if nozzle(s) are inbounds const getIsWithinPipetteExtents = ( @@ -42,6 +67,7 @@ const getIsWithinPipetteExtents = ( location.y <= A12_column_back_right_bound.y ) } + break } case 8: case 1: @@ -136,42 +162,47 @@ const hasOverlappingRectangles = ( const getHighestZInSlot = ( robotState: RobotState, invariantContext: InvariantContext, - // the labware on the top spot - labwareSlot: string + labwareId: string ): number => { const { modules, labware } = robotState const { moduleEntities, labwareEntities } = invariantContext - if (modules[labwareSlot] != null) { - const moduleDimensions = getModuleDef2(moduleEntities[labwareSlot].model) + if (modules[labwareId] != null) { + const moduleDimensions = getModuleDef2(moduleEntities[labwareId].model) .dimensions return ( // labware + module - labwareEntities[labwareSlot].def.dimensions.zDimension + + labwareEntities[labwareId].def.dimensions.zDimension + moduleDimensions.bareOverallHeight + (moduleDimensions.lidHeight ?? 0) ) - } else if (labware[labwareSlot] != null) { - const adapterSlot = labware[labwareSlot].slot - if (modules[adapterSlot] != null) { - const moduleDimensions = getModuleDef2(moduleEntities[adapterSlot].model) - .dimensions - return ( - // labware + adapter + module - labwareEntities[labwareSlot].def.dimensions.zDimension + - labwareEntities[adapterSlot].def.dimensions.zDimension + - moduleDimensions.bareOverallHeight + - (moduleDimensions.lidHeight ?? 0) - ) + } else if (labware[labwareId] != null) { + const adapterId = labware[labwareId].slot + if (labwareEntities[adapterId] != null) { + if (modules[adapterId] != null) { + const moduleDimensions = getModuleDef2(moduleEntities[adapterId].model) + .dimensions + return ( + // labware + adapter + module + labwareEntities[labwareId].def.dimensions.zDimension + + labwareEntities[adapterId].def.dimensions.zDimension + + moduleDimensions.bareOverallHeight + + (moduleDimensions.lidHeight ?? 0) + ) + } else { + return ( + // labware + adapter + labwareEntities[labwareId].def.dimensions.zDimension + + labwareEntities[adapterId].def.dimensions.zDimension + ) + } } else { - return ( - // labware + adapter - labwareEntities[labwareSlot].def.dimensions.zDimension + - labwareEntities[adapterSlot].def.dimensions.zDimension - ) + // labware + return labwareEntities[labwareId].def.dimensions.zDimension } + // shouldn't hit here! } else { - // labware - return labwareEntities[labwareSlot].def.dimensions.zDimension + console.error('something went wrong, this shoud not be hit') + return 0 } } @@ -181,9 +212,9 @@ const slotHasPotentialCollidingObject = ( slotInfo: SlotInfo[], robotState: RobotState, invariantContext: InvariantContext, - labwareSlot: string + labwareId: string ): boolean => { - for (let slot of slotInfo) { + for (const slot of slotInfo) { const slotBounds = slot.addressableArea?.boundingBox const slotPosition = slot.position @@ -211,23 +242,54 @@ const slotHasPotentialCollidingObject = ( ) && pipetteBounds[0].z != null ) { + console.log('hit here') const highestZInSlot = getHighestZInSlot( robotState, invariantContext, - labwareSlot + labwareId ) - - if (highestZInSlot >= pipetteBounds[0]?.z) { - return true - } + console.log('highestZInSlot', highestZInSlot, pipetteBounds) + return highestZInSlot >= pipetteBounds[0]?.z } } return false } -const getWillCollideWithThermocyclerLid = (pipetteBounds: Point[], slotInfos: SlotInfo[], moduleEntities: ModuleEntities): boolean => { +const getWillCollideWithThermocyclerLid = ( + pipetteBounds: Point[], + slotInfos: SlotInfo[], + moduleEntities: ModuleEntities +): boolean => { const slotIds = slotInfos.map(slot => slot.addressableArea?.id) -if (slotIds.includes('A1') && Object.values(moduleEntities).find(module => module.type === THERMOCYCLER_MODULE_TYPE)) + if ( + slotIds.includes('A1') && + Object.values(moduleEntities).find( + module => module.type === THERMOCYCLER_MODULE_TYPE + ) + ) { + return ( + hasOverlappingRectangles( + [FLEX_TC_LID_BACK_LEFT_PT, FLEX_TC_LID_FRONT_RIGHT_PT], + [pipetteBounds[0], pipetteBounds[1]] + ) && pipetteBounds[0].x <= FLEX_TC_LID_BACK_LEFT_PT.z + ) + } else { + return false + } +} + +const getWellPosition = ( + labwareEntity: LabwareEntity, + wellLocationOffset: Point +): Point => { + const wellDimensions = labwareEntity.def.dimensions + + // getting location from the bottom of the well since PD + return { + x: wellLocationOffset.x + wellDimensions.xDimension, + y: wellLocationOffset.y + wellDimensions.yDimension, + z: (wellLocationOffset.z ?? 0) + wellDimensions.zDimension, + } } // util to use in step-generation for if the pipette movement is safe @@ -235,20 +297,25 @@ export const getIsSafePipetteMovement = ( robotState: RobotState, invariantContext: InvariantContext, pipetteId: string, - destLabwareId: string, + labwareId: string, tipRackId: string, - destWellLocation: { - origin: string - offset: { x: number; y: number; z: number } - } + wellLocationOffset: Point ): boolean => { const deckDefinition = getDeckDefFromRobotType(FLEX_ROBOT_TYPE) const { pipetteEntities, labwareEntities, additionalEquipmentEntities, + moduleEntities, } = invariantContext - const { labware: labwareState, pipettes, tipState, modules } = robotState + const { labware: labwareState, tipState, modules } = robotState + const nozzleConfiguration = 'COLUMN' + + // early exit if labwareId is a trashBin or wasteChute + if (labwareEntities[labwareId] == null) { + return true + } + const stagingAreaSlots = Object.values(additionalEquipmentEntities) .filter(ae => ae.name === 'stagingArea') .map(stagingArea => stagingArea.location as string) @@ -257,29 +324,22 @@ export const getIsSafePipetteMovement = ( const tipLength = pipetteHasTip ? labwareEntities[tipRackId].def.parameters.tipLength ?? 0 : 0 - const nozzleConfiguration = pipettes[pipetteId].nozzles - const location = { - x: destWellLocation.offset.x, - y: destWellLocation.offset.y, - } - - // early exit for now if nozzle configuration is not partial tip - if (nozzleConfiguration !== 'COLUMN') { - return true - } + const wellLocationPoint = getWellPosition( + labwareEntities[labwareId], + wellLocationOffset + ) const isWithinPipetteExtents = getIsWithinPipetteExtents( pipetteEntity, - location, + wellLocationPoint, nozzleConfiguration, // TODO(jr, 4/22/24): PD only supports A12 as a primary nozzle for now PRIMARY_NOZZLE ) - if (!isWithinPipetteExtents) { return false } else { - const labwareSlot = labwareState[destLabwareId].slot + const labwareSlot = labwareState[labwareId].slot let deckSlot = labwareSlot if (modules[labwareSlot] != null) { deckSlot = modules[labwareSlot].slot @@ -293,12 +353,11 @@ export const getIsSafePipetteMovement = ( deckSlot = adapterSlot } } - const pipetteBoundsAtWellLocation = getPipetteBoundsAtSpecifiedMoveToPosition( PRIMARY_NOZZLE, pipetteEntity, tipLength, - destWellLocation.offset + wellLocationOffset ) const surroundingSlots = getFlexSurroundingSlots( labwareSlot, @@ -307,19 +366,25 @@ export const getIsSafePipetteMovement = ( const slotInfos: SlotInfo[] = surroundingSlots.map(slot => { const addressableArea = getAddressableAreaFromSlotId(slot, deckDefinition) const position = getPositionFromSlotId(slot, deckDefinition) - return { addressableArea, position, } }) - // TODO - still need todo the thermocycler collision stuff - return slotHasPotentialCollidingObject( - pipetteBoundsAtWellLocation, - slotInfos, - robotState, - invariantContext, - deckSlot + + return ( + !getWillCollideWithThermocyclerLid( + pipetteBoundsAtWellLocation, + slotInfos, + moduleEntities + ) && + !slotHasPotentialCollidingObject( + pipetteBoundsAtWellLocation, + slotInfos, + robotState, + invariantContext, + labwareId + ) ) } } From ba9eb4803db917f23a4789d1ad285b92f8363c9d Mon Sep 17 00:00:00 2001 From: Jethary Date: Wed, 24 Apr 2024 07:32:01 -0400 Subject: [PATCH 4/8] remove exporting type --- shared-data/js/types.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/shared-data/js/types.ts b/shared-data/js/types.ts index 9aa960f9ad5..4d51f992f22 100644 --- a/shared-data/js/types.ts +++ b/shared-data/js/types.ts @@ -284,7 +284,7 @@ export interface CutoutFixture { height: number } -export type AreaType = +type AreaType = | 'slot' | 'movableTrash' | 'wasteChute' From 6e8627613e256f60277e0c76f0a0dac4b536ced0 Mon Sep 17 00:00:00 2001 From: Jethary Date: Wed, 24 Apr 2024 10:09:39 -0400 Subject: [PATCH 5/8] add test coverage --- .../getIsSafePipetteMovement.test.ts | 70 +++++++++-- .../src/utils/safePipetteMovements.ts | 113 +++++++++--------- 2 files changed, 115 insertions(+), 68 deletions(-) diff --git a/step-generation/src/__tests__/getIsSafePipetteMovement.test.ts b/step-generation/src/__tests__/getIsSafePipetteMovement.test.ts index 5d6459d3487..b0d40489178 100644 --- a/step-generation/src/__tests__/getIsSafePipetteMovement.test.ts +++ b/step-generation/src/__tests__/getIsSafePipetteMovement.test.ts @@ -7,6 +7,7 @@ import { fixture96Plate, fixtureP100096V2Specs, fixtureTiprack1000ul, + fixtureTiprackAdapter, } from '@opentrons/shared-data' import { InvariantContext, RobotState } from '../types' @@ -14,6 +15,8 @@ const mockLabwareId = 'labwareId' const mockPipId = 'pip' const mockTiprackId = 'tiprackId' const mockModule = 'moduleId' +const mockLabware2 = 'labwareId2' +const mockAdapter = 'adapterId' const mockInvariantProperties: InvariantContext = { pipetteEntities: { pip: { @@ -35,6 +38,16 @@ const mockInvariantProperties: InvariantContext = { labwareDefURI: 'mockTipUri', def: fixtureTiprack1000ul as LabwareDefinition2, }, + [mockAdapter]: { + id: mockAdapter, + labwareDefURI: 'mockAdapterUri', + def: fixtureTiprackAdapter as LabwareDefinition2, + }, + [mockLabware2]: { + id: mockLabware2, + labwareDefURI: 'mockDefUri', + def: fixture96Plate as LabwareDefinition2, + }, }, moduleEntities: {}, additionalEquipmentEntities: {}, @@ -87,7 +100,7 @@ describe('getIsSafePipetteMovement', () => { ) expect(result).toEqual(false) }) - it('returns false when slot has a module near it', () => { + it('returns true when there are no collisions and a module near it', () => { mockRobotState.modules = { [mockModule]: { slot: 'D1', moduleState: {} as any }, } @@ -106,18 +119,49 @@ describe('getIsSafePipetteMovement', () => { mockTiprackId, { x: -1, y: 5, z: 20 } ) + expect(result).toEqual(true) + }) + it('returns false when there is a tip that collides', () => { + mockRobotState.tipState.tipracks = { mockTiprackId: { A1: true } } + const result = getIsSafePipetteMovement( + mockRobotState, + mockInvariantProperties, + mockPipId, + mockLabwareId, + mockTiprackId, + { x: -1, y: 5, z: 0 } + ) + expect(result).toEqual(false) + }) + it('returns false when there is a tall module nearby in a diagonal slot with adapter and labware', () => { + mockRobotState.modules = { + [mockModule]: { slot: 'C1', moduleState: {} as any }, + } + mockRobotState.labware = { + [mockLabwareId]: { slot: 'D2' }, + [mockAdapter]: { + slot: mockModule, + }, + [mockLabware2]: { + slot: mockAdapter, + }, + } + mockInvariantProperties.moduleEntities = { + [mockModule]: { + id: mockModule, + type: TEMPERATURE_MODULE_TYPE, + model: TEMPERATURE_MODULE_V2, + }, + } + const result = getIsSafePipetteMovement( + mockRobotState, + mockInvariantProperties, + mockPipId, + mockLabwareId, + mockTiprackId, + { x: 0, y: 0, z: 0 } + ) expect(result).toEqual(false) }) - // todo(jr, 4/23/24): add more test cases, test thermocycler collision, collision with tip attached - // it.only('returns true when there are no collisions!', () => { - // const result = getIsSafePipetteMovement( - // mockRobotState, - // mockInvariantProperties, - // mockPipId, - // mockLabwareId, - // mockTiprackId, - // { x: 0, y: 0, z: 0 } - // ) - // expect(result).toEqual(true) - // }) + // todo(jr, 4/23/24): add more test cases, test thermocycler collision - i'll do this in a follow up }) diff --git a/step-generation/src/utils/safePipetteMovements.ts b/step-generation/src/utils/safePipetteMovements.ts index 7d4431671e0..74c87b15f8c 100644 --- a/step-generation/src/utils/safePipetteMovements.ts +++ b/step-generation/src/utils/safePipetteMovements.ts @@ -10,6 +10,7 @@ import { import type { AddressableArea, CoordinateTuple, + DeckSlotId, NozzleConfigurationStyle, } from '@opentrons/shared-data' import type { @@ -23,6 +24,7 @@ import type { const A12_column_front_left_bound = { x: -11.03, y: 2 } const A12_column_back_right_bound = { x: 526.77, y: 506.2 } const PRIMARY_NOZZLE = 'A12' +const NOZZLE_CONFIGURATION = 'COLUMN' const FLEX_TC_LID_COLLISION_ZONE = { back_left: { x: -43.25, y: 454.9, z: 211.91 }, front_right: { x: 128.75, y: 402, z: 211.91 }, @@ -51,43 +53,39 @@ interface Point { // check if nozzle(s) are inbounds const getIsWithinPipetteExtents = ( - pipetteEntity: PipetteEntity, location: Point, nozzleConfiguration: NozzleConfigurationStyle, primaryNozzle: string ): boolean => { - const channels = pipetteEntity.spec.channels - switch (channels) { - case 96: { - if (nozzleConfiguration === 'COLUMN' && primaryNozzle === 'A12') { - return ( - A12_column_front_left_bound.x <= location.x && - location.x <= A12_column_back_right_bound.x && - A12_column_front_left_bound.y <= location.y && - location.y <= A12_column_back_right_bound.y - ) - } - break - } - case 8: - case 1: - // TODO(jr, 4/22/24): update this to support 8-channel partial tip - // and eventually all pipettes - return true + if (nozzleConfiguration === 'COLUMN' && primaryNozzle === 'A12') { + const isWithinBounds = + A12_column_front_left_bound.x <= location.x && + location.x <= A12_column_back_right_bound.x && + A12_column_front_left_bound.y <= location.y && + location.y <= A12_column_back_right_bound.y + + return isWithinBounds + } else { + // TODO: Handle other configurations such as 8-channel partial tip, and eventually all pipettes. + return true } } // return pipette bounds at a sepcific position const getPipetteBoundsAtSpecifiedMoveToPosition = ( - primaryNozzle: string, pipetteEntity: PipetteEntity, tipLength: number, destinationPosition: Point ): Point[] => { - const primaryNozzleOffset = pipetteEntity.spec.nozzleMap[primaryNozzle] + // ask sanniti about how to get primary nozzle? is it + const primaryNozzleOffset = + pipetteEntity.spec.nozzleMap != null + ? pipetteEntity.spec.nozzleMap.A1 + : pipetteEntity.spec.nozzleOffset const primaryNozzlePosition = { x: destinationPosition.x, - y: destinationPosition.y + tipLength, + y: destinationPosition.y, + z: (destinationPosition.z ?? 0) + tipLength, } const pipetteBoundsOffsets = pipetteEntity.spec.pipetteBoundingBoxOffsets const backLeftBound = { @@ -99,7 +97,10 @@ const getPipetteBoundsAtSpecifiedMoveToPosition = ( primaryNozzlePosition.y - primaryNozzleOffset[1] + pipetteBoundsOffsets.backLeftCorner[1], - z: primaryNozzleOffset[2] + pipetteBoundsOffsets.backLeftCorner[2], + z: + primaryNozzlePosition.z - + primaryNozzleOffset[2] + + pipetteBoundsOffsets.backLeftCorner[2], } const frontRightBound = { x: @@ -110,11 +111,14 @@ const getPipetteBoundsAtSpecifiedMoveToPosition = ( primaryNozzlePosition.y - primaryNozzleOffset[1] + pipetteBoundsOffsets.frontRightCorner[1], - z: primaryNozzleOffset[2] + pipetteBoundsOffsets.frontRightCorner[2], + z: + primaryNozzlePosition.z - + primaryNozzleOffset[2] + + pipetteBoundsOffsets.frontRightCorner[2], } const backRightBound: Point = { - x: backLeftBound.x, + x: frontRightBound.x, y: backLeftBound.y, z: frontRightBound.z, } @@ -128,7 +132,7 @@ const getPipetteBoundsAtSpecifiedMoveToPosition = ( } // return whether the two provided rectangles are overlapping in the 2d space. -const hasOverlappingRectangles = ( +const getHasOverlappingRectangles = ( rectangle1: Point[], rectangle2: Point[] ): boolean => { @@ -154,6 +158,7 @@ const hasOverlappingRectangles = ( const overlappingInY = Math.abs(Math.max(...yCoordinates) - Math.min(...yCoordinates)) < yLengthRect1 + yLengthRect2 + return overlappingInX && overlappingInY } @@ -207,7 +212,7 @@ const getHighestZInSlot = ( } // check if the slot overlaps with the pipette position -const slotHasPotentialCollidingObject = ( +const getSlotHasPotentialCollidingObject = ( pipetteBounds: Point[], slotInfo: SlotInfo[], robotState: RobotState, @@ -224,31 +229,28 @@ const slotHasPotentialCollidingObject = ( } const backLeftCoords = { - x: slotBounds.xDimension, - y: slotBounds.yDimension, - z: slotBounds.zDimension, + x: slotPosition[0], + y: slotBounds.yDimension + slotPosition[1], + z: slotPosition[2], } const frontRightCoords = { - x: slotPosition[0], + x: slotPosition[0] + slotBounds.xDimension, y: slotPosition[1], z: slotPosition[2], } - // Check for overlapping rectangles and pipette z-coordinate if slot overlaps with pipette bounds if ( - hasOverlappingRectangles( + getHasOverlappingRectangles( [pipetteBounds[0], pipetteBounds[1]], [backLeftCoords, frontRightCoords] ) && pipetteBounds[0].z != null ) { - console.log('hit here') const highestZInSlot = getHighestZInSlot( robotState, invariantContext, labwareId ) - console.log('highestZInSlot', highestZInSlot, pipetteBounds) return highestZInSlot >= pipetteBounds[0]?.z } } @@ -257,18 +259,15 @@ const slotHasPotentialCollidingObject = ( const getWillCollideWithThermocyclerLid = ( pipetteBounds: Point[], - slotInfos: SlotInfo[], moduleEntities: ModuleEntities ): boolean => { - const slotIds = slotInfos.map(slot => slot.addressableArea?.id) if ( - slotIds.includes('A1') && Object.values(moduleEntities).find( module => module.type === THERMOCYCLER_MODULE_TYPE ) ) { return ( - hasOverlappingRectangles( + getHasOverlappingRectangles( [FLEX_TC_LID_BACK_LEFT_PT, FLEX_TC_LID_FRONT_RIGHT_PT], [pipetteBounds[0], pipetteBounds[1]] ) && pipetteBounds[0].x <= FLEX_TC_LID_BACK_LEFT_PT.z @@ -282,13 +281,18 @@ const getWellPosition = ( labwareEntity: LabwareEntity, wellLocationOffset: Point ): Point => { - const wellDimensions = labwareEntity.def.dimensions + const { dimensions: wellDimensions, cornerOffsetFromSlot } = labwareEntity.def // getting location from the bottom of the well since PD return { - x: wellLocationOffset.x + wellDimensions.xDimension, - y: wellLocationOffset.y + wellDimensions.yDimension, - z: (wellLocationOffset.z ?? 0) + wellDimensions.zDimension, + x: + cornerOffsetFromSlot.x + wellLocationOffset.x + wellDimensions.xDimension, + y: + cornerOffsetFromSlot.y + wellLocationOffset.y + wellDimensions.yDimension, + z: + cornerOffsetFromSlot.z + + (wellLocationOffset.z ?? 0) + + wellDimensions.zDimension, } } @@ -309,7 +313,6 @@ export const getIsSafePipetteMovement = ( moduleEntities, } = invariantContext const { labware: labwareState, tipState, modules } = robotState - const nozzleConfiguration = 'COLUMN' // early exit if labwareId is a trashBin or wasteChute if (labwareEntities[labwareId] == null) { @@ -330,31 +333,33 @@ export const getIsSafePipetteMovement = ( ) const isWithinPipetteExtents = getIsWithinPipetteExtents( - pipetteEntity, wellLocationPoint, - nozzleConfiguration, // TODO(jr, 4/22/24): PD only supports A12 as a primary nozzle for now + // and only for 96-channel column pick up + NOZZLE_CONFIGURATION, PRIMARY_NOZZLE ) if (!isWithinPipetteExtents) { return false } else { const labwareSlot = labwareState[labwareId].slot - let deckSlot = labwareSlot + // labware on deck + let deckSlot: DeckSlotId = labwareSlot if (modules[labwareSlot] != null) { + // labware on module deckSlot = modules[labwareSlot].slot } else if (labwareState[labwareSlot] != null) { const adapterSlot = labwareState[labwareSlot].slot - const adapterInModuleSlot = - modules[adapterSlot] != null ? modules[adapterSlot].slot : null - if (adapterInModuleSlot != null) { - deckSlot = adapterInModuleSlot + if (modules[adapterSlot] != null) { + // labware on adapter on module + deckSlot = modules[adapterSlot].slot } else { + // labware on adapter on deck + // eslint-disable-next-line no-unused-vars deckSlot = adapterSlot } } const pipetteBoundsAtWellLocation = getPipetteBoundsAtSpecifiedMoveToPosition( - PRIMARY_NOZZLE, pipetteEntity, tipLength, wellLocationOffset @@ -371,14 +376,12 @@ export const getIsSafePipetteMovement = ( position, } }) - return ( !getWillCollideWithThermocyclerLid( pipetteBoundsAtWellLocation, - slotInfos, moduleEntities ) && - !slotHasPotentialCollidingObject( + !getSlotHasPotentialCollidingObject( pipetteBoundsAtWellLocation, slotInfos, robotState, From 0fe582606dd83c4110ff8a5eb5d49c0a71de1836 Mon Sep 17 00:00:00 2001 From: Jethary Date: Wed, 24 Apr 2024 10:22:00 -0400 Subject: [PATCH 6/8] clean up some comments --- step-generation/src/utils/safePipetteMovements.ts | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/step-generation/src/utils/safePipetteMovements.ts b/step-generation/src/utils/safePipetteMovements.ts index 74c87b15f8c..885768a2dff 100644 --- a/step-generation/src/utils/safePipetteMovements.ts +++ b/step-generation/src/utils/safePipetteMovements.ts @@ -77,7 +77,6 @@ const getPipetteBoundsAtSpecifiedMoveToPosition = ( tipLength: number, destinationPosition: Point ): Point[] => { - // ask sanniti about how to get primary nozzle? is it const primaryNozzleOffset = pipetteEntity.spec.nozzleMap != null ? pipetteEntity.spec.nozzleMap.A1 @@ -283,7 +282,8 @@ const getWellPosition = ( ): Point => { const { dimensions: wellDimensions, cornerOffsetFromSlot } = labwareEntity.def - // getting location from the bottom of the well since PD + // getting location from the bottom of the well since PD only supports aspirate/dispense from bottom + // note: api includes calibration data here which PD does not have knowledge of at the moment return { x: cornerOffsetFromSlot.x + wellLocationOffset.x + wellDimensions.xDimension, @@ -355,7 +355,6 @@ export const getIsSafePipetteMovement = ( deckSlot = modules[adapterSlot].slot } else { // labware on adapter on deck - // eslint-disable-next-line no-unused-vars deckSlot = adapterSlot } } From e0690ac97bef14f2fbfc23f6e54d187cd7c4b9de Mon Sep 17 00:00:00 2001 From: Jethary Date: Wed, 24 Apr 2024 17:32:34 -0400 Subject: [PATCH 7/8] fix lint --- step-generation/src/utils/safePipetteMovements.ts | 15 --------------- 1 file changed, 15 deletions(-) diff --git a/step-generation/src/utils/safePipetteMovements.ts b/step-generation/src/utils/safePipetteMovements.ts index 885768a2dff..713dd7bf4d1 100644 --- a/step-generation/src/utils/safePipetteMovements.ts +++ b/step-generation/src/utils/safePipetteMovements.ts @@ -343,21 +343,6 @@ export const getIsSafePipetteMovement = ( return false } else { const labwareSlot = labwareState[labwareId].slot - // labware on deck - let deckSlot: DeckSlotId = labwareSlot - if (modules[labwareSlot] != null) { - // labware on module - deckSlot = modules[labwareSlot].slot - } else if (labwareState[labwareSlot] != null) { - const adapterSlot = labwareState[labwareSlot].slot - if (modules[adapterSlot] != null) { - // labware on adapter on module - deckSlot = modules[adapterSlot].slot - } else { - // labware on adapter on deck - deckSlot = adapterSlot - } - } const pipetteBoundsAtWellLocation = getPipetteBoundsAtSpecifiedMoveToPosition( pipetteEntity, tipLength, From 736fc66fb54aeffa298cc7b5cbab9c1b81b00284 Mon Sep 17 00:00:00 2001 From: Jethary Date: Wed, 24 Apr 2024 21:59:40 -0400 Subject: [PATCH 8/8] fix lint --- step-generation/src/utils/safePipetteMovements.ts | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/step-generation/src/utils/safePipetteMovements.ts b/step-generation/src/utils/safePipetteMovements.ts index 713dd7bf4d1..ea1d7d0cadc 100644 --- a/step-generation/src/utils/safePipetteMovements.ts +++ b/step-generation/src/utils/safePipetteMovements.ts @@ -10,7 +10,6 @@ import { import type { AddressableArea, CoordinateTuple, - DeckSlotId, NozzleConfigurationStyle, } from '@opentrons/shared-data' import type { @@ -312,7 +311,7 @@ export const getIsSafePipetteMovement = ( additionalEquipmentEntities, moduleEntities, } = invariantContext - const { labware: labwareState, tipState, modules } = robotState + const { labware: labwareState, tipState } = robotState // early exit if labwareId is a trashBin or wasteChute if (labwareEntities[labwareId] == null) {