Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fixed Multi-selection functionality. #9345

Closed
wants to merge 44 commits into from
Closed
Show file tree
Hide file tree
Changes from 25 commits
Commits
Show all changes
44 commits
Select commit Hold shift + click to select a range
72c25c7
Fix multi-selection, closes #9144
Jeffrin2005 Dec 10, 2024
5e98869
Merge branch 'develop' into multiselection
Jeffrin2005 Dec 10, 2024
e8499fd
updated , closes 9144
Jeffrin2005 Dec 10, 2024
db2fb4c
updated index.tsx
Jeffrin2005 Dec 11, 2024
2098454
updated index.tsx
Jeffrin2005 Dec 13, 2024
7d99ffe
Merge branch 'develop' into multiselection
Jeffrin2005 Dec 17, 2024
b9c2913
final Board.tsx
Jeffrin2005 Dec 18, 2024
e721590
Switch out all the ButtonV2s and used button
Jeffrin2005 Dec 19, 2024
c2fe3a9
Merge branch 'multiselection' of https://github.com/Jeffrin2005/care_…
Jeffrin2005 Dec 19, 2024
7cd8d7f
button instead of buttonV2
Jeffrin2005 Dec 21, 2024
d869d1b
Merge branch 'develop' into multiselection
Jeffrin2005 Dec 21, 2024
aa51880
Final-updated shadcn button
Jeffrin2005 Dec 23, 2024
102bc74
Merge branch 'multiselection' of https://github.com/Jeffrin2005/care_…
Jeffrin2005 Dec 23, 2024
9a73618
orginal-used shadcn button
Jeffrin2005 Dec 23, 2024
1f667e2
Final-Latest-used shadcn Button
Jeffrin2005 Dec 23, 2024
fc1ff52
used_variant=primary,secondary
Jeffrin2005 Dec 23, 2024
de35c4d
Merge branch 'develop' into multiselection
nihal467 Dec 24, 2024
1ad5032
Merge branch 'develop' into multiselection
nihal467 Dec 24, 2024
b900f6f
latest-investigation
Jeffrin2005 Dec 26, 2024
db0e22e
Merge branch 'develop' into multiselection
Jeffrin2005 Dec 28, 2024
bad7863
Correct Failing Test Case
Jeffrin2005 Dec 28, 2024
0faeb64
Merge branch 'multiselection' of https://github.com/Jeffrin2005/care_…
Jeffrin2005 Dec 28, 2024
f52117b
Correct Failing Test case
Jeffrin2005 Dec 28, 2024
564a461
changed patientInvestigation.ts
Jeffrin2005 Dec 28, 2024
18fcf32
final test fail
Jeffrin2005 Dec 28, 2024
ce00cdf
latest check
Jeffrin2005 Dec 28, 2024
a2d1d38
final
Jeffrin2005 Dec 28, 2024
e6702d0
latest-final
Jeffrin2005 Dec 28, 2024
6c91c63
final1
Jeffrin2005 Dec 28, 2024
0c9da17
cypress test fail
Jeffrin2005 Dec 28, 2024
623dfa0
cypress3 test
Jeffrin2005 Dec 28, 2024
762c1ea
final2
Jeffrin2005 Dec 28, 2024
6a67690
orginal
Jeffrin2005 Dec 28, 2024
34d40da
latest1
Jeffrin2005 Dec 28, 2024
e9129e5
finall
Jeffrin2005 Dec 28, 2024
ec73916
rever
Jeffrin2005 Dec 28, 2024
33653fb
orginal
Jeffrin2005 Dec 28, 2024
a971c59
test final
Jeffrin2005 Dec 28, 2024
21601fe
temp1
Jeffrin2005 Dec 28, 2024
6d63a69
Test final
Jeffrin2005 Dec 28, 2024
a920771
Final commit of tests
Jeffrin2005 Dec 28, 2024
d6fe9b4
test1
Jeffrin2005 Dec 28, 2024
197c9a9
UP1
Jeffrin2005 Dec 28, 2024
4c56047
FINAL_COMMIT
Jeffrin2005 Dec 28, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
13 changes: 11 additions & 2 deletions cypress/e2e/patient_spec/PatientInvestigation.cy.ts
Original file line number Diff line number Diff line change
Expand Up @@ -22,16 +22,25 @@ describe("Patient Investigation Creation from Patient consultation page", () =>

it("Create a investigation for a patient and verify its reflection", () => {
patientPage.visitPatient(patientName);
cy.url().should("include", "/patient");

patientInvestigation.clickInvestigationTab();
cy.get("#consultation_tab_nav").should("exist");

patientInvestigation.clickLogLabResults();
cy.get("#log-lab-results").should("exist");

patientInvestigation.selectInvestigationOption([
"Haematology",
"Urine Test",
]);
cy.clickSubmitButton("Save Investigation");

cy.get("button")
.contains("Save Investigation")
.should("be.visible")
.click();
cy.verifyNotification("Please Enter at least one value");
cy.closeNotification();
// Temporary workflow for investigation since we dont have dummy data and moving away from existing module
});

afterEach(() => {
Expand Down
11 changes: 10 additions & 1 deletion cypress/pageobject/Patient/PatientInvestigation.ts
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,16 @@ class PatientInvestigation {
}

selectInvestigationOption(options: string[]) {
cy.clickAndMultiSelectOption("#investigations", options);
// Click to open dropdown
cy.get("#investigation-select").should("exist").click();

// Select each option
options.forEach((option) => {
cy.get("[role='option']").contains(option).should("be.visible").click();
});

// Click outside to close dropdown (if needed)
cy.get("body").click(0, 0);
}

clickLogLabResults() {
Expand Down
77 changes: 41 additions & 36 deletions src/components/Facility/Investigations/Reports/index.tsx
Original file line number Diff line number Diff line change
@@ -1,7 +1,8 @@
import { useCallback, useReducer, useState } from "react";
import { useTranslation } from "react-i18next";

import ButtonV2 from "@/components/Common/ButtonV2";
import { Button } from "@/components/ui/button";

import CircularProgress from "@/components/Common/CircularProgress";
import Loading from "@/components/Common/Loading";
import Page from "@/components/Common/Page";
Expand Down Expand Up @@ -53,42 +54,36 @@ const initialState: InitialState = {

const investigationReportsReducer = (state = initialState, action: any) => {
switch (action.type) {
case "set_investigation_groups": {
case "set_investigation_groups":
return {
...state,
investigationGroups: action.payload,
};
}
case "set_selected_group": {
case "set_selected_group":
return {
...state,
selectedGroup: action.payload,
};
}
case "set_investigations": {
case "set_investigations":
return {
...state,
investigations: action.payload,
};
}
case "set_selected_investigations": {
case "set_selected_investigations":
return {
...state,
selectedInvestigations: action.payload,
};
}
case "set_investigation_table_data": {
case "set_investigation_table_data":
return {
...state,
investigationTableData: action.payload,
};
}
case "set_loading": {
case "set_loading":
return {
...state,
isLoading: action.payload,
};
}
default:
return state;
}
Expand All @@ -113,6 +108,10 @@ const InvestigationReports = ({ id }: any) => {
selectedInvestigations,
} = state as InitialState;

const clearSelectedInvestigations = () => {
dispatch({ type: "set_selected_investigations", payload: [] });
};

const fetchInvestigationsData = useCallback(
async (
onSuccess: (
Expand Down Expand Up @@ -314,14 +313,13 @@ const InvestigationReports = ({ id }: any) => {
/>
</div>
{!isLoading.investigationLoading && (
<ButtonV2
<Button
onClick={() => fetchInvestigation()}
disabled={getTestDisabled}
variant="primary"
className="my-2.5"
>
{t("get_tests")}
</ButtonV2>
</Button>
)}
{!!isLoading.investigationLoading && (
<CircularProgress className="text-primary-500" />
Expand All @@ -342,23 +340,31 @@ const InvestigationReports = ({ id }: any) => {
})
}
optionLabel={(option) => option.name}
optionValue={(option) => option}
optionValue={(option) => option.external_id}
isLoading={isLoading.investigationLoading}
placeholder={t("select_investigations")}
selectAll={true}
/>
</div>

<ButtonV2
onClick={() => {
setSessionPage(1);
handleGenerateReports(1);
}}
disabled={generateReportDisabled}
variant="primary"
className="my-2.5"
>
{t("generate_report")}
</ButtonV2>
<div className="flex space-x-2">
<Button
onClick={() => {
setSessionPage(1);
handleGenerateReports(1);
}}
disabled={generateReportDisabled}
variant="primary"
>
{t("generate_report")}
</Button>
<Button
onClick={clearSelectedInvestigations}
disabled={!selectedInvestigations.length}
variant="secondary"
>
{t("clear")}
</Button>
</div>
</>
)}
{isLoading.tableData && (
Expand All @@ -370,19 +376,19 @@ const InvestigationReports = ({ id }: any) => {
{!!investigationTableData.length && (
<>
<div className="my-2.5">
<ButtonV2
<Button
onClick={() => handleSessionPage("NEXT")}
disabled={prevSessionDisabled}
>
{isLoading.tableData ? "Loading..." : t("next_sessions")}
</ButtonV2>
<ButtonV2
</Button>
<Button
onClick={() => handleSessionPage("PREV")}
disabled={nextSessionDisabled}
className="ml-3"
>
{isLoading.tableData ? "Loading..." : t("prev_sessions")}
</ButtonV2>
</Button>
</div>

<ReportTable
Expand All @@ -396,14 +402,13 @@ const InvestigationReports = ({ id }: any) => {
/>

{!loadMoreDisabled && (
<ButtonV2
<Button
disabled={loadMoreDisabled}
onClick={handleLoadMore}
className="my-2.5 w-full"
variant="primary"
>
{t("load_more")}
</ButtonV2>
</Button>
)}
</>
)}
Expand Down
108 changes: 56 additions & 52 deletions src/components/Facility/Investigations/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,8 @@ import { useTranslation } from "react-i18next";

import Card from "@/CAREUI/display/Card";

import { Button } from "@/components/ui/button";

import { Submit } from "@/components/Common/ButtonV2";
import Loading from "@/components/Common/Loading";
import Page from "@/components/Common/Page";
Expand Down Expand Up @@ -37,10 +39,6 @@ export interface InvestigationType {
ideal_value?: string;
groups: InvestigationGroup[];
}
type SearchItem = InvestigationGroup | InvestigationType;
function isInvestigation(e: SearchItem): e is InvestigationType {
return (e as InvestigationType).groups !== undefined;
}

const testFormReducer = (state = initialState, action: any) => {
switch (action.type) {
Expand Down Expand Up @@ -93,7 +91,7 @@ const Investigation = (props: {
name: investigation.split(" -- ")[0],
groups: investigation
.split(" -- ")[1]
.split(",")
.split(", ")
.map((group) => group.split("( ")[1].split(" )")[0]),
};
},
Expand All @@ -106,8 +104,6 @@ const Investigation = (props: {
>([]);
const [saving, setSaving] = useState(false);
const [session, setSession] = useState("");
const [selectedItems, selectItems] = useState<SearchItem[]>([]);

const { data: investigations, loading: listInvestigationDataLoading } =
useTanStackQueryInstead(routes.listInvestigations, {});

Expand Down Expand Up @@ -150,7 +146,6 @@ const Investigation = (props: {
const investigation = investigations.results.find(
(investigation) => investigation.name === inv.name,
);
// check if investigation contains all groups
if (
inv.groups.every((group: string) =>
investigation?.groups.find(
Expand All @@ -173,7 +168,6 @@ const Investigation = (props: {
.flat(),
];
setSelectedGroup(Array.from(new Set(allGroups)));
selectItems([...prefilledGroups, ...prefilledInvestigations]);
}
}, [investigations, investigationGroups]);

Expand Down Expand Up @@ -262,39 +256,58 @@ const Investigation = (props: {
[patientId]: { name: patientData?.name },
}}
>
<div className="flex flex-col gap-2">
<AutocompleteMultiSelectFormField
className="mt-5"
name="investigations"
placeholder={t("search_investigation_placeholder")}
options={[
...(investigationGroups?.results || []),
...(investigations?.results || []),
]}
value={selectedItems}
optionLabel={(option) => option.name}
optionValue={(option) => option}
onChange={({ value }) => {
selectItems(value);
setSelectedInvestigations(value.filter(isInvestigation));
setSelectedGroup(
[
...value
.filter((e) => !isInvestigation(e))
.map((e) => e.external_id),
...value.reduce<string[]>(
(acc, option) =>
acc.concat(
isInvestigation(option)
? option.groups.map((e) => e.external_id)
: [],
),
[],
),
].filter((v, i, a) => a.indexOf(v) == i),
);
}}
/>
<div className="flex flex-col gap-4">
<div className="mt-5">
<AutocompleteMultiSelectFormField
id="investigation-select"
name="investigation-select"
label={t("select_investigations")}
options={investigations?.results || []}
value={selectedInvestigations.map((inv) => inv.external_id)}
onChange={({ value }) => {
const selectedValues = Array.isArray(value) ? value : [value];
const newSelectedInvestigations = selectedValues
.map((val) =>
investigations?.results.find(
(inv) => inv.external_id === val,
),
)
.filter((inv): inv is InvestigationType => inv !== undefined);

setSelectedInvestigations(newSelectedInvestigations);

const groupIds = newSelectedInvestigations.reduce<string[]>(
(acc, inv) => acc.concat(inv.groups.map((g) => g.external_id)),
[],
);
setSelectedGroup(Array.from(new Set(groupIds)));
}}
optionLabel={(option) => option.name}
optionValue={(option) => option.external_id}
placeholder={t("select_investigations")}
selectAll
className="w-full"
/>
</div>

<div className="flex gap-2">
<Submit
onClick={handleSubmit}
disabled={saving || !selectedInvestigations.length}
label={t("save_investigation")}
/>
{selectedInvestigations.length > 0 && (
<Button
onClick={() => {
setSelectedInvestigations([]);
setSelectedGroup([]);
}}
variant="secondary"
>
{t("clear")}
</Button>
)}
</div>

{selectedGroup.map((group_id) => {
const currentGroupsInvestigations = selectedInvestigations.filter(
Expand All @@ -305,7 +318,7 @@ const Investigation = (props: {
: listOfInvestigations(group_id, investigations?.results || []);
const group = findGroup(group_id, investigationGroups?.results || []);
return (
<Card>
<Card key={group_id}>
<TestTable
data={filteredInvestigations}
title={group?.name}
Expand All @@ -316,15 +329,6 @@ const Investigation = (props: {
</Card>
);
})}

<div className="mt-4 flex justify-end">
<Submit
className="w-full md:w-auto"
onClick={handleSubmit}
disabled={saving || !selectedGroup.length}
label={t("save_investigation")}
/>
</div>
</div>
</Page>
);
Expand Down
Loading