diff --git a/app/src/assets/localization/en/error_recovery.json b/app/src/assets/localization/en/error_recovery.json index b80e6a04121..3b070cd3cac 100644 --- a/app/src/assets/localization/en/error_recovery.json +++ b/app/src/assets/localization/en/error_recovery.json @@ -21,5 +21,6 @@ "stand_back_resuming": "Stand back, resuming current step", "stand_back_retrying": "Stand back, retrying current command", "tip_not_detected": "Tip not detected", + "view_error_details": "View error details", "view_recovery_options": "View recovery options" } diff --git a/app/src/molecules/InterventionModal/__tests__/InterventionModal.test.tsx b/app/src/molecules/InterventionModal/__tests__/InterventionModal.test.tsx index 79512ce47d0..060a96eb401 100644 --- a/app/src/molecules/InterventionModal/__tests__/InterventionModal.test.tsx +++ b/app/src/molecules/InterventionModal/__tests__/InterventionModal.test.tsx @@ -1,14 +1,33 @@ import * as React from 'react' -import { describe, it, expect, beforeEach } from 'vitest' +import { vi, describe, it, expect, beforeEach } from 'vitest' +import { when } from 'vitest-when' import '@testing-library/jest-dom/vitest' -import { screen } from '@testing-library/react' + +import { screen, fireEvent } from '@testing-library/react' import { COLORS, BORDERS } from '@opentrons/components' + +import { i18n } from '../../../i18n' import { renderWithProviders } from '../../../__testing-utils__' +import { getIsOnDevice } from '../../../redux/config' + import { InterventionModal } from '../' + import type { ModalType } from '../' +import type { State } from '../../../redux/types' + +vi.mock('../../../redux/config') + +const MOCK_STATE: State = { + config: { + isOnDevice: false, + }, +} as any const render = (props: React.ComponentProps) => { - return renderWithProviders()[0] + return renderWithProviders(, { + i18nInstance: i18n, + initialState: MOCK_STATE, + })[0] } describe('InterventionModal', () => { @@ -16,11 +35,12 @@ describe('InterventionModal', () => { beforeEach(() => { props = { - heading: 'mock intervention heading', + iconHeading: 'mock intervention icon heading', children: 'mock intervention children', iconName: 'alert-circle', type: 'intervention-required', } + when(vi.mocked(getIsOnDevice)).calledWith(MOCK_STATE).thenReturn(false) }) ;(['intervention-required', 'error'] as ModalType[]).forEach(type => { const color = @@ -45,7 +65,7 @@ describe('InterventionModal', () => { it('renders passed elements', () => { render(props) screen.getByText('mock intervention children') - screen.getByText('mock intervention heading') + screen.getByText('mock intervention icon heading') }) it('renders an icon if an icon is specified', () => { const { container } = render(props) @@ -63,4 +83,39 @@ describe('InterventionModal', () => { ) expect(icon).toBeNull() }) + + it('renders title heading text if passed', () => { + props = { ...props, titleHeading: 'mock intervention title heading' } + render(props) + screen.getByText('mock intervention title heading') + }) + + it('fires an onClick when clicking on the iconHeading if passed', () => { + const mockOnClick = vi.fn() + props = { ...props, iconHeadingOnClick: mockOnClick } + render(props) + + fireEvent.click(screen.getByText('mock intervention icon heading')) + + expect(mockOnClick).toHaveBeenCalled() + }) + + it('renders the alternative desktop style when isOnDevice is false', () => { + render(props) + + expect(screen.getByTestId('__otInterventionModal')).toHaveStyle({ + width: '47rem', + maxHeight: '100%', + }) + }) + + it('renders the alternative ODD style when isOnDevice is true', () => { + when(vi.mocked(getIsOnDevice)).calledWith(MOCK_STATE).thenReturn(true) + render(props) + + expect(screen.getByTestId('__otInterventionModal')).toHaveStyle({ + width: '62rem', + height: '35.5rem', + }) + }) }) diff --git a/app/src/molecules/InterventionModal/index.tsx b/app/src/molecules/InterventionModal/index.tsx index c02adc18b64..f43ce12b16c 100644 --- a/app/src/molecules/InterventionModal/index.tsx +++ b/app/src/molecules/InterventionModal/index.tsx @@ -1,4 +1,6 @@ import * as React from 'react' +import { useSelector } from 'react-redux' + import { ALIGN_CENTER, BORDERS, @@ -7,6 +9,7 @@ import { Flex, Icon, JUSTIFY_CENTER, + JUSTIFY_SPACE_BETWEEN, OVERFLOW_AUTO, POSITION_ABSOLUTE, POSITION_RELATIVE, @@ -15,6 +18,8 @@ import { } from '@opentrons/components' import type { IconName } from '@opentrons/components' +import { getIsOnDevice } from '../../redux/config' + export type ModalType = 'intervention-required' | 'error' const BASE_STYLE = { @@ -32,20 +37,29 @@ const BASE_STYLE = { const BORDER_STYLE_BASE = `6px ${BORDERS.styleSolid}` -const MODAL_STYLE = { +const MODAL_BASE_STYLE = { backgroundColor: COLORS.white, position: POSITION_RELATIVE, overflowY: OVERFLOW_AUTO, - maxHeight: '100%', - width: '47rem', borderRadius: BORDERS.borderRadius8, boxShadow: BORDERS.smallDropShadow, 'data-testid': '__otInterventionModal', } as const +const MODAL_DESKTOP_STYLE = { + ...MODAL_BASE_STYLE, + maxHeight: '100%', + width: '47rem', +} as const + +const MODAL_ODD_STYLE = { + ...MODAL_BASE_STYLE, + width: '62rem', + height: '35.5rem', +} as const + const HEADER_STYLE = { alignItems: ALIGN_CENTER, - gridGap: SPACING.spacing12, padding: `${SPACING.spacing20} ${SPACING.spacing32}`, color: COLORS.white, position: POSITION_STICKY, @@ -69,8 +83,12 @@ const INTERVENTION_REQUIRED_COLOR = COLORS.blue50 const ERROR_COLOR = COLORS.red50 export interface InterventionModalProps { - /** optional modal heading **/ - heading?: React.ReactNode + /** Optional modal title heading. Aligned to the left. */ + titleHeading?: React.ReactNode + /** Optional modal heading right of the icon. Aligned right if titleHeading is supplied, otherwise aligned left. **/ + iconHeading?: React.ReactNode + /** Optional onClick for the icon heading and icon. */ + iconHeadingOnClick?: () => void /** overall style hint */ type?: ModalType /** optional icon name */ @@ -86,21 +104,35 @@ export function InterventionModal(props: InterventionModalProps): JSX.Element { const border = `${BORDER_STYLE_BASE} ${ modalType === 'error' ? ERROR_COLOR : INTERVENTION_REQUIRED_COLOR }` + const headerJustifyContent = + props.titleHeading != null ? JUSTIFY_SPACE_BETWEEN : undefined + + const isOnDevice = useSelector(getIsOnDevice) + const modalStyle = isOnDevice ? MODAL_ODD_STYLE : MODAL_DESKTOP_STYLE + return ( { e.stopPropagation() }} > - - {props.iconName != null ? ( - - ) : null} - {props.heading != null ? props.heading : null} + + {props.titleHeading} + + {props.iconName != null ? ( + + ) : null} + {props.iconHeading != null ? props.iconHeading : null} + {props.children} diff --git a/app/src/organisms/ErrorRecoveryFlows/BeforeBeginning.tsx b/app/src/organisms/ErrorRecoveryFlows/BeforeBeginning.tsx index 02c8ecc4ba7..0f51169032c 100644 --- a/app/src/organisms/ErrorRecoveryFlows/BeforeBeginning.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/BeforeBeginning.tsx @@ -4,8 +4,6 @@ import { Trans, useTranslation } from 'react-i18next' import { DIRECTION_COLUMN, Flex, - JUSTIFY_SPACE_BETWEEN, - SPACING, JUSTIFY_CENTER, StyledText, } from '@opentrons/components' @@ -16,6 +14,7 @@ import { BODY_TEXT_STYLE, ODD_SECTION_TITLE_STYLE, } from './constants' +import { RecoverySingleColumnContent } from './shared' import type { RecoveryContentProps } from './types' @@ -28,12 +27,7 @@ export function BeforeBeginning({ if (isOnDevice) { return ( - + {t('before_you_begin')} @@ -52,7 +46,7 @@ export function BeforeBeginning({ marginTop="auto" /> - + ) } else { return null diff --git a/app/src/organisms/ErrorRecoveryFlows/ErrorRecoveryHeader.tsx b/app/src/organisms/ErrorRecoveryFlows/ErrorRecoveryHeader.tsx deleted file mode 100644 index 0c501149b82..00000000000 --- a/app/src/organisms/ErrorRecoveryFlows/ErrorRecoveryHeader.tsx +++ /dev/null @@ -1,90 +0,0 @@ -import * as React from 'react' -import { useTranslation } from 'react-i18next' -import { css } from 'styled-components' - -import { - Box, - DIRECTION_ROW, - BORDERS, - ALIGN_CENTER, - Flex, - JUSTIFY_SPACE_BETWEEN, - TYPOGRAPHY, - COLORS, - SPACING, - RESPONSIVENESS, - StyledText, - Icon, -} from '@opentrons/components' - -import { useErrorName } from './utils' -import { NON_DESIGN_SANCTIONED_COLOR_1 } from './constants' - -import type { ErrorKind } from './types' - -interface ErrorRecoveryHeaderProps { - errorKind: ErrorKind -} -export function ErrorRecoveryHeader({ - errorKind, -}: ErrorRecoveryHeaderProps): JSX.Element { - const { t } = useTranslation('error_recovery') - const errorName = useErrorName(errorKind) - - return ( - - - - {t('recovery_mode')} - - - {errorName} - - - - - ) -} - -function AlertHeaderIcon(): JSX.Element { - return ( - - ) -} - -const BOX_STYLE = css` - background-color: ${NON_DESIGN_SANCTIONED_COLOR_1}; - @media ${RESPONSIVENESS.touchscreenMediaQuerySpecs} { - border-radius: ${BORDERS.borderRadius12} ${BORDERS.borderRadius12} 0 0; - } -` -const HEADER_CONTAINER_STYLE = css` - flex-direction: ${DIRECTION_ROW}; - justify-content: ${JUSTIFY_SPACE_BETWEEN}; - padding: ${SPACING.spacing16} ${SPACING.spacing32}; - @media ${RESPONSIVENESS.touchscreenMediaQuerySpecs} { - padding: 1.75rem ${SPACING.spacing32}; - } -` -const HEADER_TEXT_STYLE = css` - ${TYPOGRAPHY.pSemiBold} - color: ${COLORS.white}; - cursor: default; - - @media ${RESPONSIVENESS.touchscreenMediaQuerySpecs} { - font-size: ${TYPOGRAPHY.fontSize22}; - font-weight: ${TYPOGRAPHY.fontWeightBold}; - line-height: ${TYPOGRAPHY.lineHeight28}; - } -` diff --git a/app/src/organisms/ErrorRecoveryFlows/ErrorRecoveryWizard.tsx b/app/src/organisms/ErrorRecoveryFlows/ErrorRecoveryWizard.tsx index 3a721903985..4dfdc7af784 100644 --- a/app/src/organisms/ErrorRecoveryFlows/ErrorRecoveryWizard.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/ErrorRecoveryWizard.tsx @@ -2,19 +2,13 @@ import * as React from 'react' import { createPortal } from 'react-dom' import { useSelector } from 'react-redux' -import { - BORDERS, - COLORS, - DIRECTION_COLUMN, - Flex, - POSITION_ABSOLUTE, -} from '@opentrons/components' +import { StyledText } from '@opentrons/components' import { getIsOnDevice } from '../../redux/config' import { getTopPortalEl } from '../../App/portal' +import { InterventionModal } from '../../molecules/InterventionModal' import { BeforeBeginning } from './BeforeBeginning' import { SelectRecoveryOption, RetryStep, CancelRun } from './RecoveryOptions' -import { ErrorRecoveryHeader } from './ErrorRecoveryHeader' import { RecoveryInProgress } from './RecoveryInProgress' import { getErrorKind } from './utils' import { RECOVERY_MAP } from './constants' @@ -28,6 +22,7 @@ import type { useRecoveryCommands, UseRecoveryCommandsResult, } from './useRecoveryCommands' +import { useTranslation } from 'react-i18next' interface UseERWizardResult { hasLaunchedRecovery: boolean @@ -86,21 +81,31 @@ export function ErrorRecoveryWizard( ) } -function ErrorRecoveryComponent(props: RecoveryContentProps): JSX.Element { +export function ErrorRecoveryComponent( + props: RecoveryContentProps +): JSX.Element { + const { t } = useTranslation('error_recovery') + + const buildTitleHeading = (): JSX.Element => { + const titleText = props.hasLaunchedRecovery + ? t('recovery_mode') + : t('cancel_run') + return {titleText} + } + + const buildIconHeading = (): JSX.Element => ( + {t('view_error_details')} + ) + return createPortal( - - - , + , getTopPortalEl() ) } diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/CancelRun.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/CancelRun.tsx index 91ff0f7985a..da9980a21fd 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/CancelRun.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/CancelRun.tsx @@ -7,13 +7,12 @@ import { COLORS, Flex, Icon, - JUSTIFY_SPACE_BETWEEN, SPACING, StyledText, } from '@opentrons/components' import { RECOVERY_MAP } from '../constants' -import { RecoveryFooterButtons } from './shared' +import { RecoveryFooterButtons, RecoverySingleColumnContent } from '../shared' import type { RecoveryContentProps } from '../types' @@ -34,13 +33,9 @@ export function CancelRun({ if (isOnDevice) { return ( - - + ) } else { return null diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/RetryStep.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/RetryStep.tsx index 1f750a7df1c..283da580c9a 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/RetryStep.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/RetryStep.tsx @@ -6,13 +6,12 @@ import { DIRECTION_COLUMN, Flex, Icon, - JUSTIFY_SPACE_BETWEEN, SPACING, StyledText, } from '@opentrons/components' import { RECOVERY_MAP } from '../constants' -import { RecoveryFooterButtons } from './shared' +import { RecoveryFooterButtons, RecoverySingleColumnContent } from '../shared' import type { RecoveryContentProps } from '../types' @@ -35,13 +34,9 @@ export function RetryStep({ if (isOnDevice) { return ( - - + ) } else { return null diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/SelectRecoveryOption.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/SelectRecoveryOption.tsx index 8acf05cedd3..88206c3eab2 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/SelectRecoveryOption.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/SelectRecoveryOption.tsx @@ -5,7 +5,6 @@ import { useTranslation } from 'react-i18next' import { DIRECTION_COLUMN, Flex, - JUSTIFY_SPACE_BETWEEN, SPACING, StyledText, } from '@opentrons/components' @@ -16,7 +15,7 @@ import { ODD_SECTION_TITLE_STYLE, } from '../constants' import { RadioButton } from '../../../atoms/buttons' -import { RecoveryFooterButtons } from './shared' +import { RecoveryFooterButtons, RecoverySingleColumnContent } from '../shared' import type { ErrorKind, RecoveryContentProps, RecoveryRoute } from '../types' @@ -36,12 +35,7 @@ export function SelectRecoveryOption({ if (isOnDevice) { return ( - + {t('how_do_you_want_to_proceed')} @@ -58,7 +52,7 @@ export function SelectRecoveryOption({ proceedToRoute(selectedRoute as RecoveryRoute) } /> - + ) } else { return null diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/shared/index.ts b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/shared/index.ts deleted file mode 100644 index 82d6cdb7120..00000000000 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/shared/index.ts +++ /dev/null @@ -1 +0,0 @@ -export { RecoveryFooterButtons } from './RecoveryFooterButtons' diff --git a/app/src/organisms/ErrorRecoveryFlows/__tests__/ErrorRecoveryHeader.test.tsx b/app/src/organisms/ErrorRecoveryFlows/__tests__/ErrorRecoveryHeader.test.tsx deleted file mode 100644 index 31e9f596728..00000000000 --- a/app/src/organisms/ErrorRecoveryFlows/__tests__/ErrorRecoveryHeader.test.tsx +++ /dev/null @@ -1,36 +0,0 @@ -import * as React from 'react' -import { screen } from '@testing-library/react' -import { beforeEach, describe, it } from 'vitest' - -import { renderWithProviders } from '../../../__testing-utils__' -import { i18n } from '../../../i18n' -import { ErrorRecoveryHeader } from '../ErrorRecoveryHeader' -import { ERROR_KINDS } from '../constants' - -const render = (props: React.ComponentProps) => { - return renderWithProviders(, { - i18nInstance: i18n, - })[0] -} - -describe('ErrorRecoveryHeader', () => { - let props: React.ComponentProps - - beforeEach(() => { - props = { - errorKind: ERROR_KINDS.GENERAL_ERROR, - } - }) - - it('renders appropriate copy independent of errorKind', () => { - render(props) - - screen.getByText('Recovery Mode') - }) - - it('renders the appropriate header for a general error kind', () => { - render(props) - - screen.getByText('General error') - }) -}) diff --git a/app/src/organisms/ErrorRecoveryFlows/__tests__/ErrorRecoveryWizard.test.tsx b/app/src/organisms/ErrorRecoveryFlows/__tests__/ErrorRecoveryWizard.test.tsx index 3111bff953a..5c7ce7cef75 100644 --- a/app/src/organisms/ErrorRecoveryFlows/__tests__/ErrorRecoveryWizard.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/__tests__/ErrorRecoveryWizard.test.tsx @@ -7,6 +7,7 @@ import { i18n } from '../../../i18n' import { mockRecoveryContentProps } from '../__fixtures__' import { ErrorRecoveryContent, + ErrorRecoveryComponent, useInitialPipetteHome, useERWizard, } from '../ErrorRecoveryWizard' @@ -47,7 +48,9 @@ describe('useERWizard', () => { }) }) -const render = (props: React.ComponentProps) => { +const renderRecoveryContent = ( + props: React.ComponentProps +) => { return renderWithProviders(, { i18nInstance: i18n, })[0] @@ -78,7 +81,7 @@ describe('ErrorRecoveryContent', () => { }) it(`returns SelectRecoveryOption when the route is ${OPTION_SELECTION.ROUTE}`, () => { - render(props) + renderRecoveryContent(props) screen.getByText('MOCK_SELECT_RECOVERY_OPTION') }) @@ -91,7 +94,7 @@ describe('ErrorRecoveryContent', () => { route: BEFORE_BEGINNING.ROUTE, }, } - render(props) + renderRecoveryContent(props) screen.getByText('MOCK_BEFORE_BEGINNING') }) @@ -104,7 +107,7 @@ describe('ErrorRecoveryContent', () => { route: RETRY_FAILED_COMMAND.ROUTE, }, } - render(props) + renderRecoveryContent(props) screen.getByText('MOCK_RESUME_RUN') }) @@ -117,7 +120,7 @@ describe('ErrorRecoveryContent', () => { route: ROBOT_CANCELING.ROUTE, }, } - render(props) + renderRecoveryContent(props) screen.getByText('MOCK_IN_PROGRESS') }) @@ -130,7 +133,7 @@ describe('ErrorRecoveryContent', () => { route: ROBOT_IN_MOTION.ROUTE, }, } - render(props) + renderRecoveryContent(props) screen.getByText('MOCK_IN_PROGRESS') }) @@ -143,7 +146,7 @@ describe('ErrorRecoveryContent', () => { route: ROBOT_IN_MOTION.ROUTE, }, } - render(props) + renderRecoveryContent(props) screen.getByText('MOCK_IN_PROGRESS') }) @@ -156,7 +159,7 @@ describe('ErrorRecoveryContent', () => { route: ROBOT_IN_MOTION.ROUTE, }, } - render(props) + renderRecoveryContent(props) screen.getByText('MOCK_IN_PROGRESS') }) @@ -231,3 +234,32 @@ describe('useInitialPipetteHome', () => { }) }) }) + +const renderRecoveryComponent = ( + props: React.ComponentProps +) => { + return renderWithProviders(, { + i18nInstance: i18n, + })[0] +} + +describe('ErrorRecoveryComponent', () => { + let props: React.ComponentProps + + beforeEach(() => { + props = mockRecoveryContentProps + }) + + it('renders an intervention modal with appropriate text', () => { + renderRecoveryComponent(props) + screen.getByTestId('__otInterventionModal') + screen.getByText('Recovery Mode') + screen.getByText('View error details') + }) + + it('renders alternative header text if the recovery mode has not been launched', () => { + props = { ...props, hasLaunchedRecovery: false } + renderRecoveryComponent(props) + screen.getByText('Cancel run') + }) +}) diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/shared/RecoveryFooterButtons.tsx b/app/src/organisms/ErrorRecoveryFlows/shared/RecoveryFooterButtons.tsx similarity index 96% rename from app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/shared/RecoveryFooterButtons.tsx rename to app/src/organisms/ErrorRecoveryFlows/shared/RecoveryFooterButtons.tsx index 316744a2d17..7f81e95c661 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/shared/RecoveryFooterButtons.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/shared/RecoveryFooterButtons.tsx @@ -9,7 +9,7 @@ import { SPACING, } from '@opentrons/components' -import { SmallButton } from '../../../../atoms/buttons' +import { SmallButton } from '../../../atoms/buttons' interface RecoveryOptionProps { isOnDevice: boolean diff --git a/app/src/organisms/ErrorRecoveryFlows/shared/RecoverySingleColumnContent.tsx b/app/src/organisms/ErrorRecoveryFlows/shared/RecoverySingleColumnContent.tsx new file mode 100644 index 00000000000..202efc161a6 --- /dev/null +++ b/app/src/organisms/ErrorRecoveryFlows/shared/RecoverySingleColumnContent.tsx @@ -0,0 +1,31 @@ +import * as React from 'react' + +import { + DIRECTION_COLUMN, + JUSTIFY_SPACE_BETWEEN, + SPACING, + Flex, +} from '@opentrons/components' + +import type { StyleProps } from '@opentrons/components' + +interface SingleColumnContentWrapperProps extends StyleProps { + children: React.ReactNode +} +// For flex-direction: column recovery content with one column only. +export function RecoverySingleColumnContent({ + children, + ...styleProps +}: SingleColumnContentWrapperProps): JSX.Element { + return ( + + {children} + + ) +} diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/shared/__tests__/RecoveryFooterButtons.test.tsx b/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/RecoveryFooterButtons.test.tsx similarity index 93% rename from app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/shared/__tests__/RecoveryFooterButtons.test.tsx rename to app/src/organisms/ErrorRecoveryFlows/shared/__tests__/RecoveryFooterButtons.test.tsx index e08ad726fa4..42ea142c246 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/shared/__tests__/RecoveryFooterButtons.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/RecoveryFooterButtons.test.tsx @@ -2,8 +2,8 @@ import * as React from 'react' import { vi, describe, it, expect, beforeEach } from 'vitest' import { screen, fireEvent } from '@testing-library/react' -import { renderWithProviders } from '../../../../../__testing-utils__' -import { i18n } from '../../../../../i18n' +import { renderWithProviders } from '../../../../__testing-utils__' +import { i18n } from '../../../../i18n' import { RecoveryFooterButtons } from '../RecoveryFooterButtons' import type { Mock } from 'vitest' diff --git a/app/src/organisms/ErrorRecoveryFlows/shared/index.ts b/app/src/organisms/ErrorRecoveryFlows/shared/index.ts new file mode 100644 index 00000000000..7a4d99f06a3 --- /dev/null +++ b/app/src/organisms/ErrorRecoveryFlows/shared/index.ts @@ -0,0 +1,2 @@ +export { RecoveryFooterButtons } from './RecoveryFooterButtons' +export { RecoverySingleColumnContent } from './RecoverySingleColumnContent' diff --git a/app/src/organisms/IncompatibleModule/IncompatibleModuleDesktopModalBody.tsx b/app/src/organisms/IncompatibleModule/IncompatibleModuleDesktopModalBody.tsx index ac4c8c5993d..020ad066760 100644 --- a/app/src/organisms/IncompatibleModule/IncompatibleModuleDesktopModalBody.tsx +++ b/app/src/organisms/IncompatibleModule/IncompatibleModuleDesktopModalBody.tsx @@ -31,7 +31,7 @@ export function IncompatibleModuleDesktopModalBody({ const displayName = isFlex ? 'Flex' : 'OT-2' return ( ) : ( {headerTitle}} + iconHeading={{headerTitle}} iconName={iconName} type="intervention-required" >