diff --git a/webapp/src/redux/ducks/studies.ts b/webapp/src/redux/ducks/studies.ts index 5abb3cfd6d..2a52eda706 100644 --- a/webapp/src/redux/ducks/studies.ts +++ b/webapp/src/redux/ducks/studies.ts @@ -47,6 +47,7 @@ export interface StudiesSortConf { export interface StudiesState extends AsyncEntityState { current: string; + prevStudyId: string; scrollPosition: number; versionList: string[]; favorites: Array; @@ -73,6 +74,7 @@ const initialState = studiesAdapter.getInitialState({ status: FetchStatus.Idle, error: null as string | null, current: "", + prevStudyId: "", scrollPosition: 0, versionList: [] as string[], favorites: [], @@ -297,6 +299,7 @@ export default createReducer(initialState, (builder) => { draftState.versionList = action.payload; }) .addCase(setCurrentStudy, (draftState, action) => { + draftState.prevStudyId = draftState.current; draftState.current = action.payload; }) .addCase(setStudyScrollPosition, (draftState, action) => { diff --git a/webapp/src/redux/hooks/useStudyMaps.ts b/webapp/src/redux/hooks/useStudyMaps.ts index 49652627ad..0057ee95e0 100644 --- a/webapp/src/redux/hooks/useStudyMaps.ts +++ b/webapp/src/redux/hooks/useStudyMaps.ts @@ -15,47 +15,57 @@ import usePromise, { PromiseStatus } from "../../hooks/usePromise"; interface Props { studyId: StudyMetadata["id"]; - selector?: (state: AppState, studyId: StudyMetadata["id"]) => T; + selector: (state: AppState, studyId: StudyMetadata["id"]) => T; } -export default function useStudyMaps(props: Props): Response { - const { studyId, selector } = props; - const synthesisRes = useStudySynthesis({ studyId }); - const isMapsExist = useAppSelector((state) => !!getStudyMap(state, studyId)); - const data = useAppSelector((state) => - isMapsExist && selector ? selector(state, studyId) : undefined, - ); - const dispatch = useAppDispatch(); +export default function useStudyMaps({ + studyId, + selector, +}: Props): Response { const [status, setStatus] = useState(PromiseStatus.Idle); const [error, setError] = useState(); + const dispatch = useAppDispatch(); + const synthesis = useStudySynthesis({ studyId }); + const isMapsExists = useAppSelector((state) => !!getStudyMap(state, studyId)); + const data = useAppSelector((state) => selector(state, studyId) || undefined); + const prevStudyId = useAppSelector((state) => state.studies.prevStudyId); usePromise(async () => { - if (synthesisRes.status === PromiseStatus.Rejected) { - setError(synthesisRes.error); + if (synthesis.status === PromiseStatus.Rejected) { + setError(synthesis.error); setStatus(PromiseStatus.Rejected); return; } - if (synthesisRes.status !== PromiseStatus.Resolved) { + if (synthesis.status !== PromiseStatus.Resolved) { setStatus(PromiseStatus.Pending); return; } - if (!isMapsExist) { - setStatus(PromiseStatus.Pending); + setStatus(PromiseStatus.Pending); - try { - await dispatch(fetchStudyMapLayers(studyId)).unwrap(); - await dispatch(fetchStudyMapDistricts(studyId)).unwrap(); - await dispatch(createStudyMap(studyId)).unwrap(); - } catch (e) { - setError(e as Error); - setStatus(PromiseStatus.Rejected); + try { + /* + * Conditionally fetch study map data to address two scenarios: + * 1. The map data does not exist for the current study (isMapsExist is false). + * 2. The user has navigated to a different study than previously viewed (prevStudyId !== studyId). + * + * Particularly critical for accuracy when users return to a previously viewed study. + */ + if (!isMapsExists || prevStudyId !== studyId) { + await Promise.all([ + dispatch(fetchStudyMapLayers(studyId)).unwrap(), + dispatch(fetchStudyMapDistricts(studyId)).unwrap(), + dispatch(createStudyMap(studyId)).unwrap(), + ]); } - } else { + setStatus(PromiseStatus.Resolved); + } catch (err) { + setError(err as Error); + setStatus(PromiseStatus.Rejected); } - }, [isMapsExist, studyId, synthesisRes.status]); + }, [isMapsExists, studyId, synthesis.status]); return { data, status, error }; }