Skip to content

Commit

Permalink
Integrates useInfiniteQuery for data fetching and resolves Infinite L…
Browse files Browse the repository at this point in the history
…oad issue in Notes. (#9190)

* Fix Inifinite Load issue in notes

* Add useInfiniteQWuery Hook

* Fix useInfiniteQuery hook on thread change

* Fix Notes duplication by hashMap

* Added hashmap in hook itself

* Implemented useInfiniteQuery hook

* Remove Comments

* Remove Unnecessary Export

* Add Requested Changes

* simplify generic types

* Simplified useInfiniteQuery Implementation

* Removed Unnecessary types and state

* Remove unnecessary state updates

* Mandate duplicate function

* Fix Note add issue on side notes

* Remove reload state

* Fix Message Add

* Fix Notes Add Issue

* Fix Notes Add Issue

* Move over to useInfiniteQuery

* Fix Direct Use of inbuilt hook

* Fix Deduplication of notes

* Add Support of useMutation

* Add Changes

* Add Util hook for reusability

* Fix type of thread

* Remove error throw line

* remove console

* Fix Mutation function

* Move towards using callAppi

* Add Requested Changes

* Add Requested Changes

* Fix Test According to changes

* Add Requested Changes

* Re-run Test Suit

* Remove queryParam

* Fix Path Name

* Fix queryKey

* Re-run Test Suit

---------

Co-authored-by: rithviknishad <[email protected]>
  • Loading branch information
JavidSumra and rithviknishad authored Dec 29, 2024
1 parent 1b6fa7a commit 94585fd
Show file tree
Hide file tree
Showing 11 changed files with 242 additions and 228 deletions.
3 changes: 0 additions & 3 deletions cypress/e2e/patient_spec/PatientDoctorNotes.cy.ts
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,6 @@ describe("Patient Discussion notes in the consultation page", () => {
const patientNurseReplyNote = "Test nurse reply Notes";
const discussionNotesSubscribeWarning =
"Please subscribe to notifications to get live updates on discussion notes.";
const discussionNotesSuccessMessage = "Note added successfully";

before(() => {
loginPage.loginByRole("districtAdmin");
Expand All @@ -34,7 +33,6 @@ describe("Patient Discussion notes in the consultation page", () => {
patientDoctorNotes.selectNurseDiscussion();
patientDoctorNotes.addDiscussionNotes(patientNurseNote);
patientDoctorNotes.postDiscussionNotes();
cy.verifyNotification(discussionNotesSuccessMessage);
// verify the auto-switching of tab to nurse notes if the user is a nurse
patientDoctorNotes.signout();
loginPage.loginManuallyAsNurse();
Expand All @@ -49,7 +47,6 @@ describe("Patient Discussion notes in the consultation page", () => {
// Post a reply comment to the message
patientDoctorNotes.addDiscussionNotes(patientNurseReplyNote);
patientDoctorNotes.postDiscussionNotes();
cy.verifyNotification(discussionNotesSuccessMessage);
patientDoctorNotes.verifyDiscussionMessage(patientNurseReplyNote);
});

Expand Down
73 changes: 38 additions & 35 deletions src/components/Facility/ConsultationDoctorNotes/index.tsx
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
import { useQuery } from "@tanstack/react-query";
import { t } from "i18next";
import { useState } from "react";
import { useEffect, useState } from "react";
import useKeyboardShortcut from "use-keyboard-shortcut";

import CareIcon from "@/CAREUI/icons/CareIcon";
Expand All @@ -13,6 +14,7 @@ import {
PatientNoteStateType,
} from "@/components/Facility/models";
import AutoExpandingTextInputFormField from "@/components/Form/FormFields/AutoExpandingTextInputFormField";
import { useAddPatientNote } from "@/components/Patient/Utils";

import useAuthUser from "@/hooks/useAuthUser";
import { useMessageListener } from "@/hooks/useMessageListener";
Expand All @@ -22,8 +24,7 @@ import { PATIENT_NOTES_THREADS } from "@/common/constants";
import { NonReadOnlyUsers } from "@/Utils/AuthorizeFor";
import * as Notification from "@/Utils/Notifications";
import routes from "@/Utils/request/api";
import request from "@/Utils/request/request";
import useTanStackQueryInstead from "@/Utils/request/useQuery";
import query from "@/Utils/request/query";
import { classNames, isAppleDevice, keysOf } from "@/Utils/utils";

interface ConsultationDoctorNotesProps {
Expand Down Expand Up @@ -54,53 +55,47 @@ const ConsultationDoctorNotes = (props: ConsultationDoctorNotesProps) => {

const initialData: PatientNoteStateType = {
notes: [],
cPage: 1,
totalPages: 1,
facilityId: facilityId,
patientId: patientId,
};
const [state, setState] = useState(initialData);

const onAddNote = async () => {
const { mutate: addNote } = useAddPatientNote({
patientId,
thread,
consultationId,
});

const onAddNote = () => {
if (!/\S+/.test(noteField)) {
Notification.Error({
msg: "Note Should Contain At Least 1 Character",
});
return;
}

const { res } = await request(routes.addPatientNote, {
pathParams: {
patientId: patientId,
},
body: {
note: noteField,
thread,
consultation: consultationId,
reply_to: reply_to?.id,
},
setReplyTo(undefined);
setNoteField("");
addNote({
note: noteField,
reply_to: reply_to?.id,
thread,
consultation: consultationId,
});

if (res?.status === 201) {
Notification.Success({ msg: "Note added successfully" });
setState({ ...state, cPage: 1 });
setNoteField("");
setReload(true);
setReplyTo(undefined);
}
};

useTanStackQueryInstead(routes.getPatient, {
pathParams: { id: patientId },
onResponse: ({ data }) => {
if (data) {
setPatientActive(data.is_active ?? true);
setPatientName(data.name ?? "");
setFacilityName(data.facility_object?.name ?? "");
}
},
const { data } = useQuery({
queryKey: ["patient", patientId],
queryFn: query(routes.getPatient, {
pathParams: { patientId },
}),
});

useEffect(() => {
setPatientActive(data?.is_active ?? true);
setPatientName(data?.name ?? "");
setFacilityName(data?.facility_object?.name ?? "");
}, [data]);

useMessageListener((data) => {
const message = data?.message;
if (
Expand Down Expand Up @@ -147,13 +142,21 @@ const ConsultationDoctorNotes = (props: ConsultationDoctorNotesProps) => {
? "border-primary-500 font-bold text-secondary-800"
: "border-secondary-300 text-secondary-800",
)}
onClick={() => setThread(PATIENT_NOTES_THREADS[current])}
onClick={() => {
if (thread !== PATIENT_NOTES_THREADS[current]) {
setThread(PATIENT_NOTES_THREADS[current]);
setState(initialData);
setReplyTo(undefined);
setNoteField("");
}
}}
>
{t(`patient_notes_thread__${current}`)}
</button>
))}
</div>
<PatientConsultationNotesList
key={`patient-notes-${patientId}-${thread}`}
state={state}
setState={setState}
reload={reload}
Expand Down
6 changes: 4 additions & 2 deletions src/components/Facility/DoctorNote.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -14,10 +14,12 @@ interface DoctorNoteProps {
handleNext: () => void;
disableEdit?: boolean;
setReplyTo?: (reply_to: PatientNotesModel | undefined) => void;
hasMore: boolean;
}

const DoctorNote = (props: DoctorNoteProps) => {
const { state, handleNext, setReload, disableEdit, setReplyTo } = props;
const { state, handleNext, setReload, disableEdit, setReplyTo, hasMore } =
props;

return (
<div
Expand All @@ -27,7 +29,7 @@ const DoctorNote = (props: DoctorNoteProps) => {
{state.notes.length ? (
<InfiniteScroll
next={handleNext}
hasMore={state.cPage < state.totalPages}
hasMore={hasMore}
loader={
<div className="flex items-center justify-center">
<CircularProgress />
Expand Down
95 changes: 40 additions & 55 deletions src/components/Facility/PatientConsultationNotesList.tsx
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import { Dispatch, SetStateAction, useEffect, useState } from "react";
import { useInfiniteQuery } from "@tanstack/react-query";
import { Dispatch, SetStateAction, useEffect } from "react";

import CircularProgress from "@/components/Common/CircularProgress";
import DoctorNote from "@/components/Facility/DoctorNote";
Expand All @@ -12,7 +13,7 @@ import useSlug from "@/hooks/useSlug";
import { RESULTS_PER_PAGE_LIMIT } from "@/common/constants";

import routes from "@/Utils/request/api";
import request from "@/Utils/request/request";
import { callApi } from "@/Utils/request/query";

interface PatientNotesProps {
state: PatientNoteStateType;
Expand All @@ -24,80 +25,63 @@ interface PatientNotesProps {
setReplyTo?: (value: PatientNotesModel | undefined) => void;
}

const pageSize = RESULTS_PER_PAGE_LIMIT;

const PatientConsultationNotesList = (props: PatientNotesProps) => {
const {
state,
setState,
reload,
setReload,
disableEdit,
thread,
setReplyTo,
reload,
} = props;
const consultationId = useSlug("consultation") ?? "";

const [isLoading, setIsLoading] = useState(true);

const fetchNotes = async () => {
setIsLoading(true);

const { data } = await request(routes.getPatientNotes, {
pathParams: {
patientId: props.state.patientId || "",
},
query: {
consultation: consultationId,
thread,
offset: (state.cPage - 1) * RESULTS_PER_PAGE_LIMIT,
},
});
const consultationId = useSlug("consultation") ?? "";

if (data) {
if (state.cPage === 1) {
setState((prevState) => ({
...prevState,
notes: data.results,
totalPages: Math.ceil(data.count / pageSize),
}));
} else {
setState((prevState) => ({
...prevState,
notes: [...prevState.notes, ...data.results],
totalPages: Math.ceil(data.count / pageSize),
}));
const { data, isLoading, fetchNextPage, hasNextPage } = useInfiniteQuery({
queryKey: ["notes", state.patientId, thread, consultationId],
queryFn: async ({ pageParam = 0, signal }) => {
const response = await callApi(routes.getPatientNotes, {
pathParams: { patientId: state.patientId! },
queryParams: {
thread,
offset: pageParam,
consultation: consultationId,
},
signal,
});
return {
results: response?.results ?? [],
nextPage: pageParam + RESULTS_PER_PAGE_LIMIT,
totalResults: response?.count ?? 0,
};
},
getNextPageParam: (lastPage, allPages) => {
const currentResults = allPages.flatMap((page) => page.results).length;
if (currentResults < lastPage.totalResults) {
return lastPage.nextPage;
}
}
setIsLoading(false);
setReload?.(false);
};
return undefined;
},
initialPageParam: 0,
});

useEffect(() => {
if (reload) {
fetchNotes();
}
}, [reload]);
if (data?.pages) {
const allNotes = data.pages.flatMap((page) => page.results);

useEffect(() => {
fetchNotes();
}, [thread]);
const notesMap = new Map(allNotes.map((note) => [note.id, note]));

useEffect(() => {
setReload?.(true);
}, []);
const deduplicatedNotes = Array.from(notesMap.values());

const handleNext = () => {
if (state.cPage < state.totalPages) {
setState((prevState) => ({
...prevState,
cPage: prevState.cPage + 1,
notes: deduplicatedNotes,
}));
setReload?.(true);
}
};
}, [data]);

if (isLoading) {
if (isLoading || reload) {
return (
<div className="flex h-full w-full items-center justify-center bg-white">
<CircularProgress />
Expand All @@ -108,10 +92,11 @@ const PatientConsultationNotesList = (props: PatientNotesProps) => {
return (
<DoctorNote
state={state}
handleNext={handleNext}
handleNext={fetchNextPage}
setReload={setReload}
disableEdit={disableEdit}
setReplyTo={setReplyTo}
hasMore={hasNextPage}
/>
);
};
Expand Down
Loading

0 comments on commit 94585fd

Please sign in to comment.