Skip to content

Commit 7e1e100

Browse files
authored
feat(app): save RTP list scroll position when selecting RTPs on ODD (#17816)
closes AUTH-1586 closes https://opentrons.atlassian.net/browse/RQA-3523
1 parent f334ed0 commit 7e1e100

File tree

5 files changed

+115
-16
lines changed

5 files changed

+115
-16
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,32 @@
1+
import { createContext, useCallback, useState } from 'react'
2+
import type { FC, ReactNode } from 'react'
3+
4+
export interface SharedScrollRefContextType {
5+
refCallback: (node: HTMLElement | null) => void
6+
element: HTMLElement | null
7+
}
8+
9+
export const SharedScrollRefContext = createContext<SharedScrollRefContextType | null>(
10+
null
11+
)
12+
13+
// This provider exists to capture the ref of the main scrollable Box element in the ODD
14+
// This is so that we can do things like auto scroll (using the ref) across components
15+
export const SharedScrollRefProvider: FC<{
16+
children: ReactNode
17+
}> = ({ children }) => {
18+
const [element, setCurrentElement] = useState<HTMLElement | null>(null)
19+
20+
// Callback ref that updates both the state and the ref
21+
// This is necessary because we need a ref to be attached to the DOM
22+
// But also refs don't trigger rerenders, which we need in order to detect scrolling
23+
const refCallback = useCallback((node: HTMLElement | null) => {
24+
setCurrentElement(node)
25+
}, [])
26+
27+
return (
28+
<SharedScrollRefContext.Provider value={{ refCallback, element }}>
29+
{children}
30+
</SharedScrollRefContext.Provider>
31+
)
32+
}

app/src/App/OnDeviceDisplayApp.tsx

+13-11
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
import { useEffect, useState, useCallback } from 'react'
1+
import { useEffect } from 'react'
22
import { useDispatch, useSelector } from 'react-redux'
33
import { Routes, Route, Navigate, useLocation } from 'react-router-dom'
44
import { css } from 'styled-components'
@@ -9,7 +9,6 @@ import {
99
COLORS,
1010
OVERFLOW_AUTO,
1111
POSITION_RELATIVE,
12-
useScrolling,
1312
} from '@opentrons/components'
1413
import { ApiHostProvider } from '@opentrons/react-api-client'
1514
import NiceModal from '@ebay/nice-modal-react'
@@ -46,13 +45,18 @@ import { Welcome } from '/app/pages/ODD/Welcome'
4645
import { InitialLoadingScreen } from '/app/pages/ODD/InitialLoadingScreen'
4746
import { DeckConfigurationEditor } from '/app/pages/ODD/DeckConfiguration'
4847
import { PortalRoot as ModalPortalRoot } from './portal'
48+
import { SharedScrollRefProvider } from './ODDProviders/ScrollRefProvider'
4949
import {
5050
getOnDeviceDisplaySettings,
5151
updateConfigValue,
5252
} from '/app/redux/config'
5353
import { updateBrightness } from '/app/redux/shell'
5454
import { useScreenIdle, SLEEP_NEVER_MS } from '/app/local-resources/dom-utils'
55-
import { useProtocolReceiptToast, useSoftwareUpdatePoll } from './hooks'
55+
import {
56+
useProtocolReceiptToast,
57+
useScrollRef,
58+
useSoftwareUpdatePoll,
59+
} from './hooks'
5660
import { ODDTopLevelRedirects } from './ODDTopLevelRedirects'
5761
import { ReactQueryDevtools } from '/app/App/tools'
5862

@@ -199,7 +203,9 @@ export const OnDeviceDisplayApp = (): JSX.Element => {
199203
<NiceModal.Provider>
200204
<ToasterOven>
201205
<ProtocolReceiptToasts />
202-
<OnDeviceDisplayAppRoutes />
206+
<SharedScrollRefProvider>
207+
<OnDeviceDisplayAppRoutes />
208+
</SharedScrollRefProvider>
203209
</ToasterOven>
204210
</NiceModal.Provider>
205211
</MaintenanceRunTakeover>
@@ -225,14 +231,10 @@ const getTargetPath = (unfinishedUnboxingFlowRoute: string | null): string => {
225231
// split to a separate function because scrollRef rerenders on every route change
226232
// this avoids rerendering parent providers as well
227233
export function OnDeviceDisplayAppRoutes(): JSX.Element {
228-
const [currentNode, setCurrentNode] = useState<null | HTMLElement>(null)
229-
const scrollRef = useCallback((node: HTMLElement | null) => {
230-
setCurrentNode(node)
231-
}, [])
232-
const isScrolling = useScrolling(currentNode)
234+
const { isScrolling, refCallback, element } = useScrollRef()
233235
const location = useLocation()
234236
useEffect(() => {
235-
currentNode?.scrollTo({
237+
element?.scrollTo({
236238
top: 0,
237239
left: 0,
238240
behavior: 'auto',
@@ -271,7 +273,7 @@ export function OnDeviceDisplayAppRoutes(): JSX.Element {
271273
key={path}
272274
path={path}
273275
element={
274-
<Box css={TOUCH_SCREEN_STYLE} ref={scrollRef}>
276+
<Box css={TOUCH_SCREEN_STYLE} ref={refCallback}>
275277
<ModalPortalRoot />
276278
{getPathComponent(path)}
277279
</Box>

app/src/App/__tests__/OnDeviceDisplayApp.test.tsx

+6-1
Original file line numberDiff line numberDiff line change
@@ -27,7 +27,7 @@ import { getOnDeviceDisplaySettings } from '/app/redux/config'
2727
import { getIsShellReady } from '/app/redux/shell'
2828
import { getLocalRobot } from '/app/redux/discovery'
2929
import { mockConnectedRobot } from '/app/redux/discovery/__fixtures__'
30-
import { useProtocolReceiptToast } from '../hooks'
30+
import { useProtocolReceiptToast, useScrollRef } from '../hooks'
3131
import { useNotifyCurrentMaintenanceRun } from '/app/resources/maintenance_runs'
3232
import { ODDTopLevelRedirects } from '../ODDTopLevelRedirects'
3333

@@ -93,6 +93,11 @@ describe('OnDeviceDisplayApp', () => {
9393
vi.mocked(getIsShellReady).mockReturnValue(true)
9494
vi.mocked(ODDTopLevelRedirects).mockReturnValue(null)
9595
vi.mocked(getLocalRobot).mockReturnValue(mockConnectedRobot)
96+
vi.mocked(useScrollRef).mockReturnValue({
97+
isScrolling: false,
98+
refCallback: () => null,
99+
element: null,
100+
})
96101
vi.mocked(useNotifyCurrentMaintenanceRun).mockReturnValue({
97102
data: {
98103
data: {

app/src/App/hooks.ts

+36-3
Original file line numberDiff line numberDiff line change
@@ -1,17 +1,21 @@
1-
import { useCallback, useRef, useEffect } from 'react'
1+
import { useCallback, useRef, useEffect, useContext } from 'react'
22
import difference from 'lodash/difference'
33
import { useTranslation } from 'react-i18next'
44
import { useQueryClient } from 'react-query'
55
import { useDispatch } from 'react-redux'
66

7-
import { useInterval, truncateString } from '@opentrons/components'
7+
import {
8+
useInterval,
9+
truncateString,
10+
useScrolling,
11+
} from '@opentrons/components'
812
import {
913
useAllProtocolIdsQuery,
1014
useHost,
1115
useCreateLiveCommandMutation,
1216
} from '@opentrons/react-api-client'
1317
import { getProtocol } from '@opentrons/api-client'
14-
18+
import { SharedScrollRefContext } from './ODDProviders/ScrollRefProvider'
1519
import { checkShellUpdate } from '/app/redux/shell'
1620
import { useToaster } from '/app/organisms/ToasterOven'
1721

@@ -113,3 +117,32 @@ export function useProtocolReceiptToast(): void {
113117
// eslint-disable-next-line react-hooks/exhaustive-deps
114118
}, [protocolIds])
115119
}
120+
121+
export function useScrollRef(): {
122+
isScrolling: boolean
123+
refCallback: (node: HTMLElement | null) => void
124+
element: HTMLElement | null
125+
} {
126+
const refData = useContext(SharedScrollRefContext)
127+
const isScrolling = useScrolling(refData?.element ?? null) // Assuming useScrolling is properly handling scroll state
128+
129+
if (refData == null) {
130+
// log non critical error instead of throwing error to prevent white screens
131+
console.error(
132+
'useScrollRef must be used within a SharedScrollRefProvider. Falling back to dummy refs.'
133+
)
134+
return {
135+
refCallback: () => null,
136+
isScrolling: false,
137+
element: null,
138+
}
139+
}
140+
141+
const { refCallback, element } = refData
142+
143+
return {
144+
refCallback,
145+
isScrolling,
146+
element,
147+
}
148+
}

app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/ProtocolSetupParameters.tsx

+28-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
import { useState, Fragment } from 'react'
1+
import { useEffect, useState, Fragment } from 'react'
22
import { useTranslation } from 'react-i18next'
33
import { useNavigate } from 'react-router-dom'
44
import {
@@ -30,6 +30,7 @@ import { ChooseNumber } from './ChooseNumber'
3030
import { ChooseCsvFile } from './ChooseCsvFile'
3131
import { useToaster } from '/app/organisms/ToasterOven'
3232
import { ProtocolSetupStep } from '../ProtocolSetupStep'
33+
import { useScrollRef } from '/app/App/hooks'
3334
import type {
3435
CompletedProtocolAnalysis,
3536
ChoiceParameter,
@@ -68,6 +69,10 @@ export function ProtocolSetupParameters({
6869
chooseCsvFileScreen,
6970
setChooseCsvFileScreen,
7071
] = useState<CsvFileParameter | null>(null)
72+
const [prevScrollPosition, setPrevScrollPosition] = useState<number | null>(
73+
null
74+
)
75+
const { element } = useScrollRef()
7176
const [resetValuesModal, showResetValuesModal] = useState<boolean>(false)
7277
const [startSetup, setStartSetup] = useState<boolean>(false)
7378
const [runTimeParametersOverrides, setRunTimeParametersOverrides] = useState<
@@ -82,6 +87,27 @@ export function ProtocolSetupParameters({
8287
)
8388
)
8489

90+
// Scroll back to the place where the user was before they went into a specific RTP selection screen
91+
useEffect(() => {
92+
const isShowingParametersList =
93+
chooseValueScreen == null &&
94+
chooseCsvFileScreen == null &&
95+
showNumericalInputScreen == null
96+
const canRestoreScrollPosition =
97+
prevScrollPosition != null && element != null
98+
99+
if (isShowingParametersList && canRestoreScrollPosition) {
100+
element.scrollTop = prevScrollPosition
101+
setPrevScrollPosition(null) // Reset scroll position
102+
}
103+
}, [
104+
chooseCsvFileScreen,
105+
chooseValueScreen,
106+
element,
107+
prevScrollPosition,
108+
showNumericalInputScreen,
109+
])
110+
85111
const hasMissingFileParam =
86112
runTimeParametersOverrides?.some((parameter): boolean => {
87113
if (parameter.type !== 'csv_file') {
@@ -300,6 +326,7 @@ export function ProtocolSetupParameters({
300326
: parameter.displayName
301327
}
302328
onClickSetupStep={() => {
329+
setPrevScrollPosition(element?.scrollTop ?? null)
303330
handleSetParameter(parameter)
304331
}}
305332
detail={detail}

0 commit comments

Comments
 (0)