From d1db655f3597ee4e5322d2646dcf1fdeb3df4d70 Mon Sep 17 00:00:00 2001 From: Brian Arthur Cooper Date: Mon, 3 Apr 2023 20:29:45 -0400 Subject: [PATCH] fix(app): fix cal check step numbers and return tip, labware offset data help copy (#12423) Fix logic that calculates the step number within the newly redesigned calibration health check flow based on how many pipettes are being checked. Modernize contents of return tip step in cal check. fix help copy in learn more about labware position check offsets. Closes RQA-611, Closes RQA-609, Closes RAUT-393, Closes RQA-613 --- .../localization/en/device_settings.json | 3 +- .../en/labware_position_check.json | 4 +- .../localization/en/robot_calibration.json | 6 ++- .../__tests__/ApplyHistoricOffsets.test.tsx | 10 ++-- .../organisms/ApplyHistoricOffsets/index.tsx | 35 +++++-------- .../CalibrationPanels/SaveZPoint.tsx | 13 ++--- .../__tests__/SaveZPoint.test.tsx | 8 +-- .../__tests__/CalibrationTaskList.test.tsx | 14 +++++- .../organisms/CalibrationTaskList/index.tsx | 26 +++++++--- .../CheckCalibration/ConfirmExitModal.tsx | 31 ------------ .../organisms/CheckCalibration/ReturnTip.tsx | 50 ++++++++++--------- .../__tests__/CheckCalibration.test.tsx | 2 +- .../__tests__/ReturnTip.test.tsx | 2 +- app/src/organisms/CheckCalibration/index.tsx | 24 ++++++--- .../__tests__/CalibrationDashboard.test.tsx | 13 ++++- 15 files changed, 124 insertions(+), 117 deletions(-) delete mode 100644 app/src/organisms/CheckCalibration/ConfirmExitModal.tsx diff --git a/app/src/assets/localization/en/device_settings.json b/app/src/assets/localization/en/device_settings.json index 319bd47743d..d8c3703230d 100644 --- a/app/src/assets/localization/en/device_settings.json +++ b/app/src/assets/localization/en/device_settings.json @@ -226,5 +226,6 @@ "clear_all_stored_data": "Clear all stored data", "returns_your_device_to_new_state": "This returns your device to a new state.", "not_connected": "Not connected", - "connected_via": "Connected via {{networkInterface}}" + "connected_via": "Connected via {{networkInterface}}", + "attach_a_pipette_before_calibrating": "Attach a pipette in order to perform calibration" } diff --git a/app/src/assets/localization/en/labware_position_check.json b/app/src/assets/localization/en/labware_position_check.json index cc4c7f50ee2..fae11072453 100644 --- a/app/src/assets/localization/en/labware_position_check.json +++ b/app/src/assets/localization/en/labware_position_check.json @@ -61,8 +61,8 @@ "slot": "Slot {{slotName}}", "robot_has_offsets_from_previous_runs": "This robot has offsets for labware used in this protocol. If you apply these offsets, you can still adjust them with Labware Position Check.", "no_offset_data_on_robot": "This robot has no useable labware offset data for this run.", - "what_labware_offset": "What is labware offset?", - "robot_has_no_offsets_from_previous_runs": "Labware offset data references previous protocol run labware locations to save you time. Labware that has been used in the same slot as a protocol you are running will not need additional calibration. If all the labware in a new protocol have been checked in previous runs, you do not have to run Labware Position Check. You can add new offsets with Labware Position Check in later steps if you desire.", + "what_is_labware_offset_data": "What is labware offset data?", + "robot_has_no_offsets_from_previous_runs": "Labware offset data references previous protocol run labware locations to save you time. If all the labware in this protocol have been checked in previous runs, that data will be applied to this run. You can add new offsets with Labware Position Check in later steps.", "see_how_offsets_work": "See how labware offsets work", "stored_offset_data": "Stored Labware Offset data", "applied_offset_data": "Applied Labware Offset data", diff --git a/app/src/assets/localization/en/robot_calibration.json b/app/src/assets/localization/en/robot_calibration.json index 00023e5b6fc..ad22fe185ef 100644 --- a/app/src/assets/localization/en/robot_calibration.json +++ b/app/src/assets/localization/en/robot_calibration.json @@ -21,6 +21,7 @@ "check_tip_on_block": "Check tip on block", "check_tip_on_trash": "Check tip on trash bin", "check_xy_axes": "Check x- and y-axis in slot {{slotName}}", + "check_z_axis_on_slot": "Check z-axis on slot 5", "check_z_axis_on_block": "Check z-axis on block", "check_z_axis_on_trash": "Check z-axis on trash bin", "clear_other_slots": "Clear all other deck slots", @@ -124,5 +125,8 @@ "opentrons": "opentrons", "custom": "custom", "confirm_tip_rack": "Confirm tip rack", - "recalibrate_pipette": "Recalibrate pipette" + "recalibrate_pipette": "Recalibrate pipette", + "return_tip_and_continue": "Return tip and continue to next pipette", + "return_tip_and_exit": "Return tip and see calibration health check results", + "return_tip": "Return tip" } diff --git a/app/src/organisms/ApplyHistoricOffsets/__tests__/ApplyHistoricOffsets.test.tsx b/app/src/organisms/ApplyHistoricOffsets/__tests__/ApplyHistoricOffsets.test.tsx index e61068af66b..15e96980048 100644 --- a/app/src/organisms/ApplyHistoricOffsets/__tests__/ApplyHistoricOffsets.test.tsx +++ b/app/src/organisms/ApplyHistoricOffsets/__tests__/ApplyHistoricOffsets.test.tsx @@ -115,15 +115,15 @@ describe('ApplyHistoricOffsets', () => { getByText('No offset data available') getByText('Learn more').click() - getByRole('heading', { name: 'No labware offset data available' }) - getByText('This robot has no useable labware offset data for this run.') - getByText('What is labware offset?') + getByRole('heading', { name: 'What is labware offset data?' }) + getByText( - 'Labware offset data references previous protocol run labware locations to save you time. Labware that has been used in the same slot as a protocol you are running will not need additional calibration. If all the labware in a new protocol have been checked in previous runs, you do not have to run Labware Position Check.' + 'Labware offset data references previous protocol run labware locations to save you time. If all the labware in this protocol have been checked in previous runs, that data will be applied to this run.' ) getByText( - 'You can add new offsets with Labware Position Check in later steps if you desire.' + 'You can add new offsets with Labware Position Check in later steps.' ) + expect( getByRole('link', { name: 'See how labware offsets work' }) ).toHaveAttribute( diff --git a/app/src/organisms/ApplyHistoricOffsets/index.tsx b/app/src/organisms/ApplyHistoricOffsets/index.tsx index d70795378ee..9c3a9bd6e8e 100644 --- a/app/src/organisms/ApplyHistoricOffsets/index.tsx +++ b/app/src/organisms/ApplyHistoricOffsets/index.tsx @@ -27,6 +27,7 @@ import type { LoadedModule, RunTimeCommand, } from '@opentrons/shared-data' +import { ExternalLink } from '../../atoms/Link/ExternalLink' const HOW_OFFSETS_WORK_SUPPORT_URL = 'https://support.opentrons.com/s/article/How-Labware-Offsets-work-on-the-OT-2' @@ -110,7 +111,7 @@ export function ApplyHistoricOffsets( setShowOffsetDataModal(false)} @@ -126,36 +127,26 @@ export function ApplyHistoricOffsets( } > {noOffsetData ? ( - <> - - {t('no_offset_data_on_robot')} - - - {t('what_labware_offset')} - - - ), - }} - /> - + + ), + }} + /> ) : ( {t('robot_has_offsets_from_previous_runs')} )} - {t('see_how_offsets_work')} - + {!noOffsetData ? ( isLabwareOffsetCodeSnippetsOn ? ( mount && assetMap[mount][isMulti ? 'multi' : 'single'], [mount, isMulti] @@ -79,14 +79,9 @@ export function SaveZPoint(props: CalibrationPanelProps): JSX.Element { ) let title = t('calibrate_z_axis_on_slot') - let bodyTranlsationKey = 'jog_pipette_to_touch_slot' + const bodyTranslationKey = 'jog_pipette_to_touch_slot' if (isHealthCheck) { - title = - calBlock != null ? t('check_z_axis_on_block') : t('check_z_axis_on_trash') - bodyTranlsationKey = - calBlock != null - ? 'jog_pipette_to_touch_block' - : 'jog_pipette_to_touch_trash' + title = t('check_z_axis_on_slot') } return ( @@ -108,7 +103,7 @@ export function SaveZPoint(props: CalibrationPanelProps): JSX.Element { , }} diff --git a/app/src/organisms/CalibrationPanels/__tests__/SaveZPoint.test.tsx b/app/src/organisms/CalibrationPanels/__tests__/SaveZPoint.test.tsx index 1e08d3f8fb0..ef106ce75bf 100644 --- a/app/src/organisms/CalibrationPanels/__tests__/SaveZPoint.test.tsx +++ b/app/src/organisms/CalibrationPanels/__tests__/SaveZPoint.test.tsx @@ -168,9 +168,9 @@ describe('SaveZPoint', () => { sessionType: Sessions.SESSION_TYPE_CALIBRATION_HEALTH_CHECK, calBlock: mockTipLengthCalBlock, })[0] - getByRole('heading', { name: 'Check z-axis on block' }) + getByRole('heading', { name: 'Check z-axis on slot 5' }) getByText( - 'Jog the pipette until the tip is barely touching (less than 0.1 mm) the block in slot 6.' + 'Jog the pipette until the tip is barely touching (less than 0.1 mm) the deck in slot 5.' ) }) @@ -178,9 +178,9 @@ describe('SaveZPoint', () => { const { getByText, getByRole } = render({ sessionType: Sessions.SESSION_TYPE_CALIBRATION_HEALTH_CHECK, })[0] - getByRole('heading', { name: 'Check z-axis on trash bin' }) + getByRole('heading', { name: 'Check z-axis on slot 5' }) getByText( - 'Jog the pipette until the tip is barely touching (less than 0.1 mm) the flat surface of the trash bin.' + 'Jog the pipette until the tip is barely touching (less than 0.1 mm) the deck in slot 5.' ) }) }) diff --git a/app/src/organisms/CalibrationTaskList/__tests__/CalibrationTaskList.test.tsx b/app/src/organisms/CalibrationTaskList/__tests__/CalibrationTaskList.test.tsx index be0f487ad0c..798a798f1a8 100644 --- a/app/src/organisms/CalibrationTaskList/__tests__/CalibrationTaskList.test.tsx +++ b/app/src/organisms/CalibrationTaskList/__tests__/CalibrationTaskList.test.tsx @@ -15,7 +15,12 @@ import { expectedIncompleteRightMountTaskList, expectedIncompleteLeftMountTaskList, } from '../../Devices/hooks/__fixtures__/taskListFixtures' -import { useCalibrationTaskList, useRunHasStarted } from '../../Devices/hooks' +import { + useCalibrationTaskList, + useRunHasStarted, + useAttachedPipettes, +} from '../../Devices/hooks' +import { mockLeftProtoPipette } from '../../../redux/pipettes/__fixtures__' jest.mock('../../Devices/hooks') jest.mock('../../ProtocolUpload/hooks') @@ -26,6 +31,9 @@ const mockUseCalibrationTaskList = useCalibrationTaskList as jest.MockedFunction const mockUseRunHasStarted = useRunHasStarted as jest.MockedFunction< typeof useRunHasStarted > +const mockUseAttachedPipettes = useAttachedPipettes as jest.MockedFunction< + typeof useAttachedPipettes +> const render = (robotName: string = 'otie') => { return renderWithProviders( @@ -47,6 +55,10 @@ describe('CalibrationTaskList', () => { beforeEach(() => { mockUseCalibrationTaskList.mockReturnValue(expectedTaskList) mockUseRunHasStarted.mockReturnValue(false) + mockUseAttachedPipettes.mockReturnValue({ + left: mockLeftProtoPipette, + right: null, + }) }) afterEach(() => { diff --git a/app/src/organisms/CalibrationTaskList/index.tsx b/app/src/organisms/CalibrationTaskList/index.tsx index 4b707018cb8..c961026801a 100644 --- a/app/src/organisms/CalibrationTaskList/index.tsx +++ b/app/src/organisms/CalibrationTaskList/index.tsx @@ -19,7 +19,11 @@ import { StyledText } from '../../atoms/text' import { Modal } from '../../molecules/Modal' import { TaskList } from '../TaskList' -import { useCalibrationTaskList, useRunHasStarted } from '../Devices/hooks' +import { + useAttachedPipettes, + useCalibrationTaskList, + useRunHasStarted, +} from '../Devices/hooks' import { useCurrentRunId } from '../ProtocolUpload/hooks' import type { DashboardCalOffsetInvoker } from '../../pages/Devices/CalibrationDashboard/hooks/useDashboardCalibratePipOffset' @@ -55,7 +59,21 @@ export function CalibrationTaskList({ deckCalLauncher ) const runId = useCurrentRunId() + + let generalTaskDisabledReason = null + + const attachedPipettes = useAttachedPipettes() + if (attachedPipettes.left == null && attachedPipettes.right == null) { + generalTaskDisabledReason = t( + 'device_settings:attach_a_pipette_before_calibrating' + ) + } + const runHasStarted = useRunHasStarted(runId) + if (runHasStarted) + generalTaskDisabledReason = t( + 'device_settings:some_robot_controls_are_not_available' + ) React.useEffect(() => { if ( @@ -148,11 +166,7 @@ export function CalibrationTaskList({ taskList={taskList} taskListStatus={taskListStatus} generalTaskClickHandler={() => setHasLaunchedWizard(true)} - generalTaskDisabledReason={ - runHasStarted - ? t('device_settings:some_robot_controls_are_not_available') - : null - } + generalTaskDisabledReason={generalTaskDisabledReason} /> )} diff --git a/app/src/organisms/CheckCalibration/ConfirmExitModal.tsx b/app/src/organisms/CheckCalibration/ConfirmExitModal.tsx deleted file mode 100644 index e5effc40c9c..00000000000 --- a/app/src/organisms/CheckCalibration/ConfirmExitModal.tsx +++ /dev/null @@ -1,31 +0,0 @@ -import * as React from 'react' - -import { AlertModal } from '@opentrons/components' - -export interface ConfirmExitModalProps { - back: () => unknown - exit: () => unknown -} - -const HEADING = 'Are you sure you want to exit?' -const GO_BACK = 'go back' -const EXIT = 'continue' -const WARNING = - 'Doing so will take you to the summary page and prompt you to drop the tip.' - -export function ConfirmExitModal(props: ConfirmExitModalProps): JSX.Element { - const { back, exit } = props - - return ( - - {WARNING} - - ) -} diff --git a/app/src/organisms/CheckCalibration/ReturnTip.tsx b/app/src/organisms/CheckCalibration/ReturnTip.tsx index 71dd3e5d060..0c907ea4a8c 100644 --- a/app/src/organisms/CheckCalibration/ReturnTip.tsx +++ b/app/src/organisms/CheckCalibration/ReturnTip.tsx @@ -1,11 +1,10 @@ import * as React from 'react' import { Flex, - PrimaryBtn, - ALIGN_CENTER, DIRECTION_COLUMN, - JUSTIFY_CENTER, - SPACING_3, + JUSTIFY_SPACE_BETWEEN, + SPACING, + ALIGN_STRETCH, } from '@opentrons/components' import uniq from 'lodash/uniq' @@ -13,14 +12,12 @@ import * as Sessions from '../../redux/sessions' import { StyledText } from '../../atoms/text' import type { CalibrationPanelProps } from '../../organisms/CalibrationPanels/types' import type { SessionCommandString } from '../../redux/sessions/types' - -const CONFIRM_RETURN_BODY = 'Return tip and ' -const CONTINUE_TO_NEXT = 'continue to next pipette' -const EXIT_PROGRAM = 'see calibration health check results' -const CONTINUE = 'continue to the next tiprack' -const EXIT = 'continue to the result summary' +import { useTranslation } from 'react-i18next' +import { PrimaryButton } from '../../atoms/buttons' +import { NeedHelpLink } from '../CalibrationPanels' export function ReturnTip(props: CalibrationPanelProps): JSX.Element { + const { t } = useTranslation('robot_calibration') const { sendCommands, checkBothPipettes, activePipette, instruments } = props const onFinalPipette = !checkBothPipettes || @@ -55,23 +52,28 @@ export function ReturnTip(props: CalibrationPanelProps): JSX.Element { } return ( - - {`${CONFIRM_RETURN_BODY} - ${onFinalPipette ? EXIT_PROGRAM : CONTINUE_TO_NEXT}`} - - + + {onFinalPipette + ? t('return_tip_and_exit') + : t('return_tip_and_continue')} + + + - {onFinalPipette ? EXIT : CONTINUE} - + + + {t('return_tip')} + + ) } diff --git a/app/src/organisms/CheckCalibration/__tests__/CheckCalibration.test.tsx b/app/src/organisms/CheckCalibration/__tests__/CheckCalibration.test.tsx index e28370f9588..8330d215205 100644 --- a/app/src/organisms/CheckCalibration/__tests__/CheckCalibration.test.tsx +++ b/app/src/organisms/CheckCalibration/__tests__/CheckCalibration.test.tsx @@ -41,7 +41,7 @@ describe('CheckCalibration', () => { heading: 'Did pipette pick up tip successfully?', currentStep: 'inspectingTip', }, - { heading: 'Check z-axis on trash bin', currentStep: 'comparingHeight' }, + { heading: 'Check z-axis on slot 5', currentStep: 'comparingHeight' }, { heading: 'Check x- and y-axis in slot 1', currentStep: 'comparingPointOne', diff --git a/app/src/organisms/CheckCalibration/__tests__/ReturnTip.test.tsx b/app/src/organisms/CheckCalibration/__tests__/ReturnTip.test.tsx index 700faa8fca2..15ccb375f46 100644 --- a/app/src/organisms/CheckCalibration/__tests__/ReturnTip.test.tsx +++ b/app/src/organisms/CheckCalibration/__tests__/ReturnTip.test.tsx @@ -19,7 +19,7 @@ describe('ReturnTip', () => { const getContinueButton = ( wrapper: ReactWrapper> - ) => wrapper.find('button[title="confirmReturnTip"]') + ) => wrapper.find('button[aria-label="return tip"]') beforeEach(() => { mockSendCommands = jest.fn() diff --git a/app/src/organisms/CheckCalibration/index.tsx b/app/src/organisms/CheckCalibration/index.tsx index 9fd318fafb3..a6b1192d1d1 100644 --- a/app/src/organisms/CheckCalibration/index.tsx +++ b/app/src/organisms/CheckCalibration/index.tsx @@ -68,7 +68,16 @@ const STEPS_IN_ORDER_ONE_PIPETTE: RobotCalibrationCheckStep[] = [ Sessions.CHECK_STEP_RETURNING_TIP, Sessions.CHECK_STEP_RESULTS_SUMMARY, ] -const STEPS_IN_ORDER_SECOND_PIPETTE: RobotCalibrationCheckStep[] = [ +const STEPS_IN_ORDER_BOTH_PIPETTES: RobotCalibrationCheckStep[] = [ + Sessions.CHECK_STEP_SESSION_STARTED, + Sessions.CHECK_STEP_LABWARE_LOADED, + Sessions.CHECK_STEP_COMPARING_NOZZLE, + Sessions.CHECK_STEP_PREPARING_PIPETTE, + Sessions.CHECK_STEP_INSPECTING_TIP, + Sessions.CHECK_STEP_COMPARING_TIP, + Sessions.CHECK_STEP_COMPARING_HEIGHT, + Sessions.CHECK_STEP_COMPARING_POINT_ONE, + Sessions.CHECK_STEP_RETURNING_TIP, Sessions.CHECK_STEP_LABWARE_LOADED, Sessions.CHECK_STEP_COMPARING_NOZZLE, Sessions.CHECK_STEP_PREPARING_PIPETTE, @@ -86,12 +95,11 @@ function getStepIndexCheckingBothPipettes( rank: RobotCalibrationCheckPipetteRank | null ): number { if (currentStep == null || rank == null) return 0 - return ( - (rank === CHECK_PIPETTE_RANK_FIRST - ? STEPS_IN_ORDER_ONE_PIPETTE.findIndex(step => step === currentStep) - : STEPS_IN_ORDER_SECOND_PIPETTE.findIndex(step => step === currentStep) + - 7) ?? 0 - ) + return rank === CHECK_PIPETTE_RANK_FIRST + ? STEPS_IN_ORDER_BOTH_PIPETTES.findIndex(step => step === currentStep) + : STEPS_IN_ORDER_BOTH_PIPETTES.slice(9).findIndex( + step => step === currentStep + ) + 9 } export function CheckCalibration( @@ -175,7 +183,7 @@ export function CheckCalibration( currentStep={stepIndex} totalSteps={ checkBothPipettes - ? STEPS_IN_ORDER_SECOND_PIPETTE.length - 1 + 7 + ? STEPS_IN_ORDER_BOTH_PIPETTES.length - 1 : STEPS_IN_ORDER_ONE_PIPETTE.length - 1 } onExit={confirmExit} diff --git a/app/src/pages/Devices/CalibrationDashboard/__tests__/CalibrationDashboard.test.tsx b/app/src/pages/Devices/CalibrationDashboard/__tests__/CalibrationDashboard.test.tsx index 52758c36cf8..ff0cf5cd1e2 100644 --- a/app/src/pages/Devices/CalibrationDashboard/__tests__/CalibrationDashboard.test.tsx +++ b/app/src/pages/Devices/CalibrationDashboard/__tests__/CalibrationDashboard.test.tsx @@ -6,11 +6,15 @@ import { renderWithProviders } from '@opentrons/components' import { i18n } from '../../../../i18n' import { CalibrationDashboard } from '..' -import { useCalibrationTaskList } from '../../../../organisms/Devices/hooks' +import { + useCalibrationTaskList, + useAttachedPipettes, +} from '../../../../organisms/Devices/hooks' import { useDashboardCalibratePipOffset } from '../hooks/useDashboardCalibratePipOffset' import { useDashboardCalibrateTipLength } from '../hooks/useDashboardCalibrateTipLength' import { useDashboardCalibrateDeck } from '../hooks/useDashboardCalibrateDeck' import { expectedTaskList } from '../../../../organisms/Devices/hooks/__fixtures__/taskListFixtures' +import { mockLeftProtoPipette } from '../../../../redux/pipettes/__fixtures__' jest.mock('../../../../organisms/Devices/hooks') jest.mock('../hooks/useDashboardCalibratePipOffset') @@ -29,6 +33,9 @@ const mockUseDashboardCalibrateTipLength = useDashboardCalibrateTipLength as jes const mockUseDashboardCalibrateDeck = useDashboardCalibrateDeck as jest.MockedFunction< typeof useDashboardCalibrateDeck > +const mockUseAttachedPipettes = useAttachedPipettes as jest.MockedFunction< + typeof useAttachedPipettes +> const render = (path = '/') => { return renderWithProviders( @@ -49,6 +56,10 @@ describe('CalibrationDashboard', () => { mockUseDashboardCalibratePipOffset.mockReturnValue([() => {}, null]) mockUseDashboardCalibrateTipLength.mockReturnValue([() => {}, null]) mockUseDashboardCalibrateDeck.mockReturnValue([() => {}, null]) + mockUseAttachedPipettes.mockReturnValue({ + left: mockLeftProtoPipette, + right: null, + }) }) it('renders a robot calibration dashboard title', () => {