From 9b2d9473712912950cb7f24c2373a0ef74e7f14a Mon Sep 17 00:00:00 2001 From: Khavin Shankar Date: Wed, 29 Nov 2023 19:53:55 +0530 Subject: [PATCH 01/21] Make facilityMiddlewareHostname as a state (#6753) --- src/Components/Facility/Consultations/Feed.tsx | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/src/Components/Facility/Consultations/Feed.tsx b/src/Components/Facility/Consultations/Feed.tsx index 0389fe298dc..43d5b89cb38 100644 --- a/src/Components/Facility/Consultations/Feed.tsx +++ b/src/Components/Facility/Consultations/Feed.tsx @@ -55,15 +55,15 @@ export const Feed: React.FC = ({ consultationId, facilityId }) => { const [isFullscreen, setFullscreen] = useFullscreen(); const [videoStartTime, setVideoStartTime] = useState(null); const [statusReported, setStatusReported] = useState(false); + const [facilityMiddlewareHostname, setFacilityMiddlewareHostname] = + useState(""); const authUser = useAuthUser(); - let facilityMiddlewareHostname = ""; - useQuery(routes.getPermittedFacility, { pathParams: { id: facilityId || "" }, onResponse: ({ res, data }) => { if (res && res.status === 200 && data && data.middleware_address) { - facilityMiddlewareHostname = data.middleware_address; + setFacilityMiddlewareHostname(data.middleware_address); } }, }); From a18016b6b3d063d07ede0dd096c98ffb5b09803c Mon Sep 17 00:00:00 2001 From: Mohammed Nihal <57055998+nihal467@users.noreply.github.com> Date: Wed, 29 Nov 2023 19:56:44 +0530 Subject: [PATCH 02/21] facility advance filter (#6736) --- cypress/e2e/facility_spec/facility.cy.ts | 2 +- .../e2e/facility_spec/facility_homepage.cy.ts | 24 ++++++++++++++++++- .../pageobject/Facility/FacilityCreation.ts | 16 +++++++++++-- 3 files changed, 38 insertions(+), 4 deletions(-) diff --git a/cypress/e2e/facility_spec/facility.cy.ts b/cypress/e2e/facility_spec/facility.cy.ts index 5ae5d7cb9db..da06c65ed53 100644 --- a/cypress/e2e/facility_spec/facility.cy.ts +++ b/cypress/e2e/facility_spec/facility.cy.ts @@ -49,7 +49,7 @@ describe("Facility Creation", () => { facilityPage.visitUpdateFacilityPage(facilityUrl); facilityPage.clickManageFacilityDropdown(); facilityPage.clickUpdateFacilityOption(); - facilityPage.clickUpdateFacilityType(); + facilityPage.clickUpdateFacilityType("Request Approving Center"); facilityPage.fillFacilityName("cypress facility updated"); facilityPage.fillAddress("Cypress Facility Updated Address"); facilityPage.fillOxygenCapacity("100"); diff --git a/cypress/e2e/facility_spec/facility_homepage.cy.ts b/cypress/e2e/facility_spec/facility_homepage.cy.ts index bf6ce12536e..3d916b4ba47 100644 --- a/cypress/e2e/facility_spec/facility_homepage.cy.ts +++ b/cypress/e2e/facility_spec/facility_homepage.cy.ts @@ -4,17 +4,22 @@ import LoginPage from "../../pageobject/Login/LoginPage"; import FacilityHome from "../../pageobject/Facility/FacilityHome"; import ManageUserPage from "../../pageobject/Users/ManageUserPage"; import FacilityPage from "../../pageobject/Facility/FacilityCreation"; +import { UserPage } from "../../pageobject/Users/UserSearch"; -describe("Facility Creation", () => { +describe("Facility Homepage Function", () => { const loginPage = new LoginPage(); const facilityHome = new FacilityHome(); const facilityPage = new FacilityPage(); const manageUserPage = new ManageUserPage(); + const userPage = new UserPage(); const facilitiesAlias = "downloadFacilitiesCSV"; const capacitiesAlias = "downloadCapacitiesCSV"; const doctorsAlias = "downloadDoctorsCSV"; const triagesAlias = "downloadTriagesCSV"; const facilityname = "Dummy Facility 1"; + const statename = "Kerala"; + const district = "Ernakulam"; + const facilitytype = "Private Hospital"; before(() => { loginPage.loginAsDisctrictAdmin(); @@ -26,6 +31,23 @@ describe("Facility Creation", () => { cy.awaitUrl("/facility"); }); + it("Verify the functionality of advance filter", () => { + userPage.clickAdvancedFilters(); + facilityPage.selectState(statename); + facilityPage.selectDistrict(district); + // facilityPage.selectLocalBody("Anthikad Grama"); current dummy data have issue in local body + facilityPage.clickUpdateFacilityType(facilitytype); + userPage.applyFilter(); + facilityPage.verifyStateBadgeContent(statename); + facilityPage.verifyDistrictBadgeContent(district); + facilityPage.verifyFacilityTypeBadgeContent(facilitytype); + manageUserPage.assertFacilityInCard(facilityname); + userPage.clearFilters(); + userPage.verifyDataTestIdNotVisible("State"); + userPage.verifyDataTestIdNotVisible("District"); + userPage.verifyDataTestIdNotVisible("Facility type"); + }); + it("Search a facility in homepage", () => { manageUserPage.typeFacilitySearch(facilityname); facilityPage.verifyFacilityBadgeContent(facilityname); diff --git a/cypress/pageobject/Facility/FacilityCreation.ts b/cypress/pageobject/Facility/FacilityCreation.ts index d27433f04e7..8c20b808c98 100644 --- a/cypress/pageobject/Facility/FacilityCreation.ts +++ b/cypress/pageobject/Facility/FacilityCreation.ts @@ -17,11 +17,11 @@ class FacilityPage { cy.get("#manage-facility-dropdown button").should("be.visible"); } - clickUpdateFacilityType() { + clickUpdateFacilityType(facilityType) { cy.get("#facility_type") .click() .then(() => { - cy.get("[role='option']").contains("Request Approving Center").click(); + cy.get("[role='option']").contains(facilityType).click(); }); } @@ -195,6 +195,18 @@ class FacilityPage { ); } + verifyStateBadgeContent(expectedText: string) { + cy.get("[data-testid='State']").should("contain", expectedText); + } + + verifyDistrictBadgeContent(expectedText: string) { + cy.get("[data-testid='District']").should("contain", expectedText); + } + + verifyFacilityTypeBadgeContent(expectedText: string) { + cy.get("[data-testid='Facility type']").should("contain", expectedText); + } + verifyfacilitycreateassetredirection() { cy.url().should("include", "/assets/new"); } From fe28c4e029f38e84c2708b55a3a0d9997bc61431 Mon Sep 17 00:00:00 2001 From: Kshitij Verma <101321276+kshitijv256@users.noreply.github.com> Date: Thu, 30 Nov 2023 10:12:45 +0530 Subject: [PATCH 03/21] showing errors on wrong numbers in PhoneNumberFormField (#6675) * showing errors wrong number in PhoneNumberFormField * made error field show only when there is an error * Update src/Components/Form/FormFields/FormField.tsx Co-authored-by: Rithvik Nishad * changed margin to remove extra space in userfilter * Apply suggestions from code review --------- Co-authored-by: Rithvik Nishad --- src/Components/Shifting/ListFilter.tsx | 1 - src/Components/Users/UserFilter.tsx | 40 +++++++++++++------------- 2 files changed, 20 insertions(+), 21 deletions(-) diff --git a/src/Components/Shifting/ListFilter.tsx b/src/Components/Shifting/ListFilter.tsx index c5b3555e8c4..ef366d5b038 100644 --- a/src/Components/Shifting/ListFilter.tsx +++ b/src/Components/Shifting/ListFilter.tsx @@ -410,7 +410,6 @@ export default function ListFilter(props: any) { name="patient_phone_number" value={filterState.patient_phone_number} onChange={handleFormFieldChange} - errorClassName="hidden" types={["mobile", "landline"]} /> - - - - +
+ +
+
+ +
); } From f5a6ac3b46120d2c2da3ed8144a0b4e7e9bd1dab Mon Sep 17 00:00:00 2001 From: Rithvik Nishad Date: Thu, 30 Nov 2023 10:12:59 +0530 Subject: [PATCH 04/21] Show discharge reason for OP consultations (#6724) --- src/Components/Patient/PatientInfoCard.tsx | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/src/Components/Patient/PatientInfoCard.tsx b/src/Components/Patient/PatientInfoCard.tsx index e49607629fa..78e299fe218 100644 --- a/src/Components/Patient/PatientInfoCard.tsx +++ b/src/Components/Patient/PatientInfoCard.tsx @@ -348,7 +348,11 @@ export default function PatientInfoCard(props: {
{!consultation?.discharge_reason ? ( - UNKNOWN + + {consultation.suggestion === "OP" + ? "OP file closed" + : "UNKNOWN"} + ) : consultation?.discharge_reason === "EXP" ? ( EXPIRED ) : ( From 31b7690ce92d5b07e6d52aa9dc8d0546c40a59f0 Mon Sep 17 00:00:00 2001 From: Gampa Sri Harsh <114745442+sriharsh05@users.noreply.github.com> Date: Thu, 30 Nov 2023 10:13:10 +0530 Subject: [PATCH 05/21] Convert the unit from grams to kg in inventory form (#6750) --- src/Components/Facility/AddInventoryForm.tsx | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) diff --git a/src/Components/Facility/AddInventoryForm.tsx b/src/Components/Facility/AddInventoryForm.tsx index df217cad3ec..9e4f761b550 100644 --- a/src/Components/Facility/AddInventoryForm.tsx +++ b/src/Components/Facility/AddInventoryForm.tsx @@ -146,7 +146,7 @@ export const AddInventoryForm = (props: any) => { if (unitName === "Dozen") { return Number(unitData.quantity) * 12; } - if (unitName === "Gram") { + if (unitName === "gram") { return Number(unitData.quantity) / 1000; } return Number(unitData.quantity); @@ -238,6 +238,11 @@ export const AddInventoryForm = (props: any) => { }; // if user has selected "Add stock" or "stockValidation" function is true if (data.is_incoming || stockValidation(data)) { + // if user has selected grams as unit then convert it to kg + if (data.unit === 5) { + data.quantity = data.quantity / 1000; + data.unit = 6; + } const res = await dispatchAction(postInventory(data, { facilityId })); setIsLoading(false); From 747e5f130a08ba6e20d61b4a035a46a07a4fa203 Mon Sep 17 00:00:00 2001 From: Rithvik Nishad Date: Fri, 1 Dec 2023 10:47:45 +0530 Subject: [PATCH 06/21] Adds support for capturing time in Discharge Date (#6757) * Adds support for capturing time in Discharge Date * Refactor discharge form field labels and add conditional rendering for death-related fields * show time in patient card * fix minor issue * fix discharge prescriptions being shown --- src/Components/Facility/DischargeModal.tsx | 149 +++++++++------------ src/Components/Patient/PatientInfoCard.tsx | 3 +- 2 files changed, 62 insertions(+), 90 deletions(-) diff --git a/src/Components/Facility/DischargeModal.tsx b/src/Components/Facility/DischargeModal.tsx index 138e74fb9b9..3384efa686b 100644 --- a/src/Components/Facility/DischargeModal.tsx +++ b/src/Components/Facility/DischargeModal.tsx @@ -8,9 +8,7 @@ import ClaimDetailCard from "../HCX/ClaimDetailCard"; import { ConsultationModel } from "./models"; import CreateClaimCard from "../HCX/CreateClaimCard"; import { DISCHARGE_REASONS } from "../../Common/constants"; -import DateFormField from "../Form/FormFields/DateFormField"; import DialogModal from "../Common/Dialog"; -import { FieldChangeEvent } from "../Form/FormFields/Utils"; import { FieldLabel } from "../Form/FormFields/FormField"; import { HCXActions } from "../../Redux/actions"; import { HCXClaimModel } from "../HCX/models"; @@ -161,11 +159,6 @@ const DischargeModal = ({ setIsSendingDischargeApi(false); if (dischargeResponse?.status === 200) { - // TODO: check this later - // const dischargeData = Object.assign({}, patientData); - // dischargeData["discharge"] = value; - // setPatientData(dischargeData); - Notification.Success({ msg: "Patient Discharged Successfully", }); @@ -174,15 +167,6 @@ const DischargeModal = ({ } }; - const handleDateChange = (e: FieldChangeEvent) => { - setPreDischargeForm((form) => { - return { - ...form, - discharge_date: e.value.toString(), - }; - }); - }; - const handleFacilitySelect = (selected: FacilityModel) => { setFacility(selected); const { id, name } = selected || {}; @@ -248,11 +232,10 @@ const DischargeModal = ({ - {preDischargeForm.discharge_reason === "REC" && ( -
- + { + const updates: Record = { + discharge_date: undefined, + death_datetime: undefined, + }; + updates[e.name] = e.value; + setPreDischargeForm((form) => ({ ...form, ...updates })); + }} + required + min={dayjs( + consultationData?.admission_date ?? consultationData?.created_date + ).format("YYYY-MM-DDTHH:mm")} + max={dayjs().format("YYYY-MM-DDTHH:mm")} + error={ + preDischargeForm.discharge_reason === "EXP" + ? errors?.death_datetime + : errors?.discharge_date + } + /> + {preDischargeForm.discharge_reason === "REC" && ( + <>
Discharge Prescription Medications @@ -288,61 +296,24 @@ const DischargeModal = ({ Discharge PRN Prescriptions
-
+ )} {preDischargeForm.discharge_reason === "EXP" && ( -
- { - setPreDischargeForm((form) => { - return { - ...form, - death_datetime: e.value, - }; - }); - }} - required - min={dayjs(consultationData?.admission_date).format( - "YYYY-MM-DDTHH:mm" - )} - max={dayjs().format("YYYY-MM-DDTHH:mm")} - /> - { - setPreDischargeForm((form) => { - return { - ...form, - death_confirmed_doctor: e.value, - }; - }); - }} - required - placeholder="Attending Doctor's Name and Designation" - /> -
- )} - {["REF", "LAMA"].includes(preDischargeForm.discharge_reason) && ( -
- -
+ { + setPreDischargeForm((form) => { + return { + ...form, + death_confirmed_doctor: e.value, + }; + }); + }} + required + placeholder="Attending Doctor's Name and Designation" + /> )}
diff --git a/src/Components/Patient/PatientInfoCard.tsx b/src/Components/Patient/PatientInfoCard.tsx index 78e299fe218..8bf20a65f64 100644 --- a/src/Components/Patient/PatientInfoCard.tsx +++ b/src/Components/Patient/PatientInfoCard.tsx @@ -330,7 +330,8 @@ export default function PatientInfoCard(props: { ) : ( {" "} - Discharged on {formatDate(consultation?.discharge_date)} + Discharged on{" "} + {formatDateTime(consultation?.discharge_date)} )} From 47511a322f8bf3d79489e00f79285a46414f30da Mon Sep 17 00:00:00 2001 From: Ashesh <3626859+Ashesh3@users.noreply.github.com> Date: Fri, 1 Dec 2023 20:01:45 +0530 Subject: [PATCH 07/21] Update tooltip position in ConfigureHealthFacility (#6764) * Update tooltip position in ConfigureHealthFacility component * Update tooltip positioning in ConfigureHealthFacility component --- src/Components/ABDM/ConfigureHealthFacility.tsx | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/Components/ABDM/ConfigureHealthFacility.tsx b/src/Components/ABDM/ConfigureHealthFacility.tsx index a84e4b6d4b9..bccf58e47c2 100644 --- a/src/Components/ABDM/ConfigureHealthFacility.tsx +++ b/src/Components/ABDM/ConfigureHealthFacility.tsx @@ -178,7 +178,7 @@ export const ConfigureHealthFacility = (props: any) => { > {state.form.health_facility?.registered ? ( <> -
+
The ABDM health facility is successfully linked with care{" "} @@ -192,7 +192,7 @@ export const ConfigureHealthFacility = (props: any) => { ) : ( <> -
+
The ABDM health facility is successfully linked with care{" "} From 737ef2dcbbcf835c766ca428c38324236d788d7f Mon Sep 17 00:00:00 2001 From: Rithvik Nishad Date: Fri, 1 Dec 2023 21:24:58 +0530 Subject: [PATCH 08/21] fixes #6766; Misaligned label in external results (#6769) --- src/Components/ExternalResult/ResultList.tsx | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/Components/ExternalResult/ResultList.tsx b/src/Components/ExternalResult/ResultList.tsx index d148b26e3f5..9aa365c7303 100644 --- a/src/Components/ExternalResult/ResultList.tsx +++ b/src/Components/ExternalResult/ResultList.tsx @@ -264,11 +264,11 @@ export default function ResultList() { value={qParams.name} placeholder="Search by name" /> -
Search by number
setPhoneNum(e.value)} error={phoneNumberError} From 4ed48a96fedcebf5bd80683c2584ce836bdaea1d Mon Sep 17 00:00:00 2001 From: Rithvik Nishad Date: Fri, 1 Dec 2023 22:10:23 +0530 Subject: [PATCH 09/21] fixes #6774; vitals content overlapping bed dropdown (#6775) --- src/Components/VitalsMonitor/HL7PatientVitalsMonitor.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/Components/VitalsMonitor/HL7PatientVitalsMonitor.tsx b/src/Components/VitalsMonitor/HL7PatientVitalsMonitor.tsx index 6533d133089..d91b0c56b2f 100644 --- a/src/Components/VitalsMonitor/HL7PatientVitalsMonitor.tsx +++ b/src/Components/VitalsMonitor/HL7PatientVitalsMonitor.tsx @@ -236,7 +236,7 @@ export const VitalsNonWaveformContent = ({ }: { children: JSX.Element | JSX.Element[]; }) => ( -
+
{children}
); From b49247cfc154db3f3d1adc6e5a8c18c3ecc4d65c Mon Sep 17 00:00:00 2001 From: Ashesh <3626859+Ashesh3@users.noreply.github.com> Date: Fri, 1 Dec 2023 23:07:04 +0530 Subject: [PATCH 10/21] Add middleware address to AssetLocationObject and (#6778) update HL7 and ventilator middleware --- src/Components/Assets/AssetTypes.tsx | 1 + .../ConsultationDetails/ConsultationUpdatesTab.tsx | 9 +++++++-- 2 files changed, 8 insertions(+), 2 deletions(-) diff --git a/src/Components/Assets/AssetTypes.tsx b/src/Components/Assets/AssetTypes.tsx index 8b96b6beeb7..a4005404da1 100644 --- a/src/Components/Assets/AssetTypes.tsx +++ b/src/Components/Assets/AssetTypes.tsx @@ -8,6 +8,7 @@ export interface AssetLocationObject { description: string; created_date?: string; modified_date?: string; + middleware_address?: string; facility: { id: string; name: string; diff --git a/src/Components/Facility/ConsultationDetails/ConsultationUpdatesTab.tsx b/src/Components/Facility/ConsultationDetails/ConsultationUpdatesTab.tsx index 736a7c81575..bb584ae93e6 100644 --- a/src/Components/Facility/ConsultationDetails/ConsultationUpdatesTab.tsx +++ b/src/Components/Facility/ConsultationDetails/ConsultationUpdatesTab.tsx @@ -60,7 +60,10 @@ export const ConsultationUpdatesTab = (props: ConsultationTabProps) => { setMonitorBedData(monitorBedData); const assetDataForMonitor = monitorBedData?.asset_object; const hl7Meta = assetDataForMonitor?.meta; - const hl7Middleware = hl7Meta?.middleware_hostname || middleware_address; + const hl7Middleware = + hl7Meta?.middleware_hostname || + assetDataForMonitor?.location_object?.middleware_address || + middleware_address; if (hl7Middleware && hl7Meta?.local_ip_address) { setHL7SocketUrl( `wss://${hl7Middleware}/observations/${hl7Meta.local_ip_address}` @@ -85,7 +88,9 @@ export const ConsultationUpdatesTab = (props: ConsultationTabProps) => { setVentilatorBedData(ventilatorBedData); const ventilatorMeta = ventilatorBedData?.asset_object?.meta; const ventilatorMiddleware = - ventilatorMeta?.middleware_hostname || middleware_address; + ventilatorMeta?.middleware_hostname || + consultationBedVentilator?.location_object.middleware_address || + middleware_address; if (ventilatorMiddleware && ventilatorMeta?.local_ip_address) { setVentilatorSocketUrl( `wss://${ventilatorMiddleware}/observations/${ventilatorMeta?.local_ip_address}` From e9fbcc0e8ffc8e63c0007d21abc831b78ddb306b Mon Sep 17 00:00:00 2001 From: Mohammed Nihal <57055998+nihal467@users.noreply.github.com> Date: Mon, 4 Dec 2023 09:41:34 +0530 Subject: [PATCH 11/21] Updated login credentials in README.md (#6786) --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index 002afc979bc..a6a2806558d 100644 --- a/README.md +++ b/README.md @@ -50,7 +50,7 @@ Once the development server has started, open [localhost:4000](http://localhost: Authenticate to staging API with any of the following credentials ```yaml -- username: devdistrictadmin +- username: dev-districtadmin password: Coronasafe@123 role: District Admin From 9fea2d85c1c7b85323363f70a46ec89283d5bf27 Mon Sep 17 00:00:00 2001 From: Mohammed Nihal <57055998+nihal467@users.noreply.github.com> Date: Mon, 4 Dec 2023 15:05:26 +0530 Subject: [PATCH 12/21] New Cypress Test | Create new Facility in multiple combination | Facility Tab (#6785) * Inital File Change * revert vite * Final converted remaining test part * minor fix * minor fix component change --- cypress/e2e/facility_spec/facility.cy.ts | 82 ------ .../e2e/facility_spec/facility_creation.cy.ts | 245 ++++++++++++++++++ .../pageobject/Facility/FacilityCreation.ts | 87 +++++++ .../Facility/FacilityBedCapacity.tsx | 4 +- src/Components/Facility/FacilityCreate.tsx | 9 +- .../Facility/FacilityDoctorList.tsx | 4 +- src/Components/Facility/FacilityHome.tsx | 16 +- 7 files changed, 354 insertions(+), 93 deletions(-) delete mode 100644 cypress/e2e/facility_spec/facility.cy.ts create mode 100644 cypress/e2e/facility_spec/facility_creation.cy.ts diff --git a/cypress/e2e/facility_spec/facility.cy.ts b/cypress/e2e/facility_spec/facility.cy.ts deleted file mode 100644 index da06c65ed53..00000000000 --- a/cypress/e2e/facility_spec/facility.cy.ts +++ /dev/null @@ -1,82 +0,0 @@ -// FacilityCreation -import { cy, describe, before, beforeEach, it, afterEach } from "local-cypress"; -import FacilityPage from "../../pageobject/Facility/FacilityCreation"; -import LoginPage from "../../pageobject/Login/LoginPage"; - -describe("Facility Creation", () => { - let facilityUrl: string; - const facilityPage = new FacilityPage(); - const loginPage = new LoginPage(); - - before(() => { - loginPage.loginAsDisctrictAdmin(); - cy.saveLocalStorage(); - }); - - beforeEach(() => { - cy.viewport(1280, 720); - cy.restoreLocalStorage(); - cy.awaitUrl("/facility"); - }); - - it("Create a new facility", () => { - facilityPage.visitCreateFacilityPage(); - facilityPage.fillFacilityName("cypress facility"); - facilityPage.fillPincode("682001"); - facilityPage.selectState("Kerala"); - facilityPage.selectDistrict("Ernakulam"); - facilityPage.selectLocalBody("Aluva"); - facilityPage.selectWard("4"); - facilityPage.fillAddress("Cypress Address"); - facilityPage.fillPhoneNumber("9898469865"); - facilityPage.submitForm(); - - facilityPage.selectBedType("Oxygen beds"); - facilityPage.fillTotalCapacity("10"); - facilityPage.fillCurrentlyOccupied("5"); - facilityPage.saveAndExitBedCapacityForm(); - - facilityPage.selectAreaOfSpecialization("General Medicine"); - facilityPage.fillDoctorCount("5"); - facilityPage.saveAndExitDoctorForm(); - facilityPage.verifyfacilitynewurl(); - cy.url().then((newUrl) => { - facilityUrl = newUrl; - }); - }); - - it("Update the existing facility", () => { - facilityPage.visitUpdateFacilityPage(facilityUrl); - facilityPage.clickManageFacilityDropdown(); - facilityPage.clickUpdateFacilityOption(); - facilityPage.clickUpdateFacilityType("Request Approving Center"); - facilityPage.fillFacilityName("cypress facility updated"); - facilityPage.fillAddress("Cypress Facility Updated Address"); - facilityPage.fillOxygenCapacity("100"); - facilityPage.fillExpectedOxygenRequirement("80"); - facilityPage.selectLocation("Kochi, Kerala"); - facilityPage.submitForm(); - - cy.url().should("not.include", "/update"); - }); - - it("Configure the existing facility", () => { - facilityPage.visitUpdateFacilityPage(facilityUrl); - facilityPage.clickManageFacilityDropdown(); - facilityPage.clickConfigureFacilityOption(); - facilityPage.fillMiddleWareAddress("dev_middleware.coronasafe.live"); - facilityPage.clickupdateMiddleWare(); - facilityPage.verifySuccessNotification("Facility updated successfully"); - }); - - it("Delete a facility", () => { - facilityPage.visitUpdateFacilityPage(facilityUrl); - facilityPage.clickManageFacilityDropdown(); - facilityPage.clickDeleteFacilityOption(); - facilityPage.confirmDeleteFacility(); - }); - - afterEach(() => { - cy.saveLocalStorage(); - }); -}); diff --git a/cypress/e2e/facility_spec/facility_creation.cy.ts b/cypress/e2e/facility_spec/facility_creation.cy.ts new file mode 100644 index 00000000000..d43aa093a69 --- /dev/null +++ b/cypress/e2e/facility_spec/facility_creation.cy.ts @@ -0,0 +1,245 @@ +// FacilityCreation +import { cy, describe, before, beforeEach, it, afterEach } from "local-cypress"; +import FacilityPage from "../../pageobject/Facility/FacilityCreation"; +import LoginPage from "../../pageobject/Login/LoginPage"; +import FacilityHome from "../../pageobject/Facility/FacilityHome"; +import ManageUserPage from "../../pageobject/Users/ManageUserPage"; +import { UserCreationPage } from "../../pageobject/Users/UserCreation"; + +describe("Facility Creation", () => { + let facilityUrl1: string; + const facilityPage = new FacilityPage(); + const loginPage = new LoginPage(); + const facilityHome = new FacilityHome(); + const manageUserPage = new ManageUserPage(); + const userCreationPage = new UserCreationPage(); + const facilityFeature = [ + "CT Scan", + "X-Ray", + "Maternity Care", + "Neonatal Care", + "Operation Theater", + "Blood Bank", + ]; + const bedCapacity = "10"; + const bedOccupancy = "5"; + const oxygenCapacity = "100"; + const oxygenExpected = "80"; + const totalCapacity = "20"; + const totalOccupancy = "10"; + const doctorCapacity = "5"; + const totalDoctor = "10"; + const facilityName = "cypress facility"; + const facilityAddress = "cypress address"; + const facilityNumber = "9898469865"; + const facilityErrorMessage = [ + "Required", + "Invalid Pincode", + "Required", + "Required", + "Required", + "Required", + "Required", + "Required", + "Invalid Phone Number", + ]; + const bedErrorMessage = [ + "Field is required", + "Total capacity cannot be 0", + "Field is required", + ]; + const doctorErrorMessage = ["Field is required", "Field is required"]; + + before(() => { + loginPage.loginAsDisctrictAdmin(); + cy.saveLocalStorage(); + }); + + beforeEach(() => { + cy.viewport(1280, 720); + cy.restoreLocalStorage(); + cy.awaitUrl("/facility"); + }); + + it("Create a new facility with multiple bed and doctor capacity", () => { + // create facility with multiple capacity and verify form error message for facility form + facilityPage.visitCreateFacilityPage(); + facilityPage.submitForm(); + userCreationPage.verifyErrorMessages(facilityErrorMessage); + facilityPage.fillFacilityName(facilityName); + facilityPage.clickfacilityfeatureoption(); + facilityFeature.forEach((featureText) => { + cy.get("[role='option']").contains(featureText).click(); + }); + facilityPage.fillPincode("682001"); + facilityPage.selectLocalBody("Aluva"); + facilityPage.selectWard("4"); + facilityPage.fillAddress(facilityAddress); + facilityPage.fillPhoneNumber(facilityNumber); + facilityPage.fillOxygenCapacity(oxygenCapacity); + facilityPage.fillExpectedOxygenRequirement(oxygenExpected); + facilityPage.fillBTypeCylinderCapacity(oxygenCapacity); + facilityPage.fillExpectedBTypeCylinderRequirement(oxygenExpected); + facilityPage.fillCTypeCylinderCapacity(oxygenCapacity); + facilityPage.fillExpectedCTypeCylinderRequirement(oxygenExpected); + facilityPage.fillDTypeCylinderCapacity(oxygenCapacity); + facilityPage.fillExpectedDTypeCylinderRequirement(oxygenExpected); + facilityPage.selectLocation("Kochi, Kerala"); + facilityPage.submitForm(); + // create multiple bed capacity and verify card reflection + facilityPage.selectBedType("Oxygen beds"); + facilityPage.fillTotalCapacity(bedCapacity); + facilityPage.fillCurrentlyOccupied(bedOccupancy); + facilityPage.clickbedcapcityaddmore(); + facilityPage.selectBedType("Ordinary Bed"); + facilityPage.fillTotalCapacity(bedCapacity); + facilityPage.fillCurrentlyOccupied(bedOccupancy); + facilityPage.clickbedcapcityaddmore(); + facilityPage.getTotalBedCapacity().contains(totalCapacity); + facilityPage.getTotalBedCapacity().contains(totalOccupancy); + facilityPage.clickcancelbutton(); + // create multiple bed capacity and verify card reflection + facilityPage.selectAreaOfSpecialization("General Medicine"); + facilityPage.fillDoctorCount(doctorCapacity); + facilityPage.clickdoctorcapacityaddmore(); + facilityPage.selectAreaOfSpecialization("Pulmonology"); + facilityPage.fillDoctorCount(doctorCapacity); + facilityPage.clickdoctorcapacityaddmore(); + facilityPage.getTotalDoctorCapacity().contains(doctorCapacity); + facilityPage.clickcancelbutton(); + facilityPage.verifyfacilitynewurl(); + // verify the facility card + facilityPage.getFacilityName().contains(facilityName).should("be.visible"); + facilityPage + .getAddressDetailsView() + .contains(facilityAddress) + .should("be.visible"); + facilityPage + .getPhoneNumberView() + .contains(facilityNumber) + .should("be.visible"); + facilityPage + .getFacilityAvailableFeatures() + .invoke("text") + .then((text) => { + facilityFeature.forEach((feature) => { + expect(text).to.contain(feature); + }); + }); + facilityPage.getFacilityOxygenInfo().scrollIntoView(); + facilityPage + .getFacilityOxygenInfo() + .contains(oxygenCapacity) + .should("be.visible"); + facilityPage.getFacilityTotalBedCapacity().scrollIntoView(); + facilityPage.getFacilityTotalBedCapacity().contains(totalCapacity); + facilityPage.getFacilityTotalBedCapacity().contains(totalOccupancy); + facilityPage.getFacilityTotalDoctorCapacity().scrollIntoView(); + facilityPage.getFacilityTotalDoctorCapacity().contains(totalDoctor); + }); + + it("Create a new facility with single bed and doctor capacity", () => { + facilityPage.visitCreateFacilityPage(); + facilityPage.fillFacilityName(facilityName); + facilityPage.fillPincode("682001"); + facilityPage.selectLocalBody("Aluva"); + facilityPage.selectWard("4"); + facilityPage.fillAddress(facilityAddress); + facilityPage.fillPhoneNumber(facilityNumber); + facilityPage.submitForm(); + // add the bed capacity + facilityPage.selectBedType("Oxygen beds"); + facilityPage.fillTotalCapacity(oxygenCapacity); + facilityPage.fillCurrentlyOccupied(oxygenExpected); + facilityPage.saveAndExitBedCapacityForm(); + // add the doctor capacity + facilityPage.selectAreaOfSpecialization("General Medicine"); + facilityPage.fillDoctorCount(doctorCapacity); + facilityPage.saveAndExitDoctorForm(); + facilityPage.verifyfacilitynewurl(); + // verify the created facility details + facilityPage.getFacilityName().contains(facilityName).should("be.visible"); + facilityPage + .getAddressDetailsView() + .contains(facilityAddress) + .should("be.visible"); + facilityPage + .getPhoneNumberView() + .contains(facilityNumber) + .should("be.visible"); + // verify the facility homepage + cy.visit("/facility"); + manageUserPage.typeFacilitySearch(facilityName); + facilityPage.verifyFacilityBadgeContent(facilityName); + manageUserPage.assertFacilityInCard(facilityName); + facilityHome.verifyURLContains(facilityName); + }); + + it("Create a new facility with no bed and doctor capacity", () => { + facilityPage.visitCreateFacilityPage(); + facilityPage.fillFacilityName(facilityName); + facilityPage.fillPincode("682001"); + facilityPage.selectLocalBody("Aluva"); + facilityPage.selectWard("4"); + facilityPage.fillAddress(facilityAddress); + facilityPage.fillPhoneNumber(facilityNumber); + facilityPage.submitForm(); + // add no bed capacity and verify form error message + facilityPage.isVisibleselectBedType(); + facilityPage.saveAndExitBedCapacityForm(); + userCreationPage.verifyErrorMessages(bedErrorMessage); + facilityPage.clickcancelbutton(); + // add no doctor capacity and verify form error message + facilityPage.isVisibleAreaOfSpecialization(); + facilityPage.clickdoctorcapacityaddmore(); + userCreationPage.verifyErrorMessages(doctorErrorMessage); + facilityPage.clickcancelbutton(); + cy.url().then((newUrl) => { + facilityUrl1 = newUrl; + }); + // verify the created facility details + facilityPage.getFacilityName().contains(facilityName).should("be.visible"); + facilityPage + .getAddressDetailsView() + .contains(facilityAddress) + .should("be.visible"); + facilityPage + .getPhoneNumberView() + .contains(facilityNumber) + .should("be.visible"); + }); + + it("Update the existing facility", () => { + facilityPage.visitUpdateFacilityPage(facilityUrl1); + facilityPage.clickManageFacilityDropdown(); + facilityPage.clickUpdateFacilityOption(); + facilityPage.clickUpdateFacilityType("Request Approving Center"); + facilityPage.fillFacilityName("cypress facility updated"); + facilityPage.fillAddress("Cypress Facility Updated Address"); + facilityPage.fillOxygenCapacity("100"); + facilityPage.fillExpectedOxygenRequirement("80"); + facilityPage.selectLocation("Kochi, Kerala"); + facilityPage.submitForm(); + cy.url().should("not.include", "/update"); + }); + + it("Configure the existing facility", () => { + facilityPage.visitUpdateFacilityPage(facilityUrl1); + facilityPage.clickManageFacilityDropdown(); + facilityPage.clickConfigureFacilityOption(); + facilityPage.fillMiddleWareAddress("dev_middleware.coronasafe.live"); + facilityPage.clickupdateMiddleWare(); + facilityPage.verifySuccessNotification("Facility updated successfully"); + }); + + it("Delete a facility", () => { + facilityPage.visitUpdateFacilityPage(facilityUrl1); + facilityPage.clickManageFacilityDropdown(); + facilityPage.clickDeleteFacilityOption(); + facilityPage.confirmDeleteFacility(); + }); + + afterEach(() => { + cy.saveLocalStorage(); + }); +}); diff --git a/cypress/pageobject/Facility/FacilityCreation.ts b/cypress/pageobject/Facility/FacilityCreation.ts index 8c20b808c98..b4d3808d645 100644 --- a/cypress/pageobject/Facility/FacilityCreation.ts +++ b/cypress/pageobject/Facility/FacilityCreation.ts @@ -70,6 +70,10 @@ class FacilityPage { cy.get("[role='option']").contains(bedType).click(); } + isVisibleselectBedType() { + cy.get("div#bed-type button").should("be.visible"); + } + fillTotalCapacity(capacity: string) { cy.get("input#total-capacity").click().type(capacity); } @@ -87,6 +91,10 @@ class FacilityPage { cy.get("[role='option']").contains(area).click(); } + isVisibleAreaOfSpecialization() { + cy.get("div#area-of-specialization button").should("be.visible"); + } + fillDoctorCount(count: string) { cy.get("input#count").click().type(count); } @@ -99,6 +107,33 @@ class FacilityPage { cy.get("#expected_oxygen_requirement").click().clear().type(requirement); } + fillBTypeCylinderCapacity(capacity: string) { + cy.get("#type_b_cylinders").click().clear().type(capacity); + } + + fillExpectedBTypeCylinderRequirement(requirement: string) { + cy.get("#expected_type_b_cylinders").focus().clear(); + cy.get("#expected_type_b_cylinders").focus().type(requirement); + } + + fillCTypeCylinderCapacity(capacity: string) { + cy.get("#type_c_cylinders").click().clear().type(capacity); + } + + fillExpectedCTypeCylinderRequirement(requirement: string) { + cy.get("#expected_type_c_cylinders").focus().clear(); + cy.get("#expected_type_c_cylinders").focus().type(requirement); + } + + fillDTypeCylinderCapacity(capacity: string) { + cy.get("#type_d_cylinders").click().clear().type(capacity); + } + + fillExpectedDTypeCylinderRequirement(requirement: string) { + cy.get("#expected_type_d_cylinders").focus().clear(); + cy.get("#expected_type_d_cylinders").focus().type(requirement); + } + saveAndExitDoctorForm() { cy.intercept("GET", "**/api/v1/facility/**").as("createFacilities"); cy.get("button#save-and-exit").click(); @@ -138,6 +173,42 @@ class FacilityPage { cy.get("#inventory-management").click(); } + getTotalBedCapacity() { + return cy.get("#total-bed-capacity"); + } + + getFacilityTotalBedCapacity() { + return cy.get("#facility-bed-capacity-details"); + } + + getFacilityTotalDoctorCapacity() { + return cy.get("#facility-doctor-capacity-details"); + } + + getTotalDoctorCapacity() { + return cy.get("#total-doctor-capacity"); + } + + getFacilityName() { + return cy.get("#facility-name"); + } + + getAddressDetailsView() { + return cy.get("#address-details-view"); + } + + getPhoneNumberView() { + return cy.get("#phone-number-view"); + } + + getFacilityAvailableFeatures() { + return cy.get("#facility-available-features"); + } + + getFacilityOxygenInfo() { + return cy.get("#facility-oxygen-info"); + } + clickResourceRequestOption() { cy.get("#resource-request").contains("Resource Request").click(); } @@ -146,6 +217,22 @@ class FacilityPage { cy.get("#delete-facility").contains("Delete Facility").click(); } + clickfacilityfeatureoption() { + cy.get("#features").click(); + } + + clickbedcapcityaddmore() { + cy.get("#bed-capacity-save").click(); + } + + clickdoctorcapacityaddmore() { + cy.get("#doctor-save").click(); + } + + clickcancelbutton() { + cy.get("#cancel").click(); + } + verifyfacilitynewurl() { cy.url().should("match", /facility\/[a-z\d-]+/); } diff --git a/src/Components/Facility/FacilityBedCapacity.tsx b/src/Components/Facility/FacilityBedCapacity.tsx index 33583840643..0c05fb022cb 100644 --- a/src/Components/Facility/FacilityBedCapacity.tsx +++ b/src/Components/Facility/FacilityBedCapacity.tsx @@ -82,7 +82,7 @@ export const FacilityBedCapacity = (props: any) => { } return ( -
+
Bed Capacity
@@ -114,6 +114,6 @@ export const FacilityBedCapacity = (props: any) => { /> )} -
+
); }; diff --git a/src/Components/Facility/FacilityCreate.tsx b/src/Components/Facility/FacilityCreate.tsx index e742caefb82..8a7887ebc25 100644 --- a/src/Components/Facility/FacilityCreate.tsx +++ b/src/Components/Facility/FacilityCreate.tsx @@ -570,7 +570,10 @@ export const FacilityCreate = (props: FacilityProps) => { }); capacityList = ( -
+
{
{t("doctors_list")}
-
{doctorList}
+
+ {doctorList} +
); diff --git a/src/Components/Facility/FacilityDoctorList.tsx b/src/Components/Facility/FacilityDoctorList.tsx index e6bbc7f7f3b..d9a8b0d1a37 100644 --- a/src/Components/Facility/FacilityDoctorList.tsx +++ b/src/Components/Facility/FacilityDoctorList.tsx @@ -83,7 +83,7 @@ export const FacilityDoctorList = (props: any) => { } return ( -
+
Doctors List
@@ -116,6 +116,6 @@ export const FacilityDoctorList = (props: any) => { /> )} -
+
); }; diff --git a/src/Components/Facility/FacilityHome.tsx b/src/Components/Facility/FacilityHome.tsx index e971f0f3b9d..3ad295a3df3 100644 --- a/src/Components/Facility/FacilityHome.tsx +++ b/src/Components/Facility/FacilityHome.tsx @@ -198,7 +198,7 @@ export const FacilityHome = (props: any) => { )} {editCoverImageTooltip}
-
+

{facilityData?.name}

{facilityData?.modified_date && ( {
-
+

Address

@@ -222,7 +222,7 @@ export const FacilityHome = (props: any) => {
-
+

Phone Number

@@ -270,7 +270,10 @@ export const FacilityHome = (props: any) => { ) && (

Available features

)} -
+
{facilityData?.features?.map( (feature: number, i: number) => FACILITY_FEATURE_TYPES.some((f) => f.id === feature) && ( @@ -426,7 +429,10 @@ export const FacilityHome = (props: any) => {

Oxygen Information

-
+
Date: Mon, 4 Dec 2023 17:40:19 +0530 Subject: [PATCH 13/21] Update cypress badge in README.md (#6788) --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index a6a2806558d..7f2c3e926bd 100644 --- a/README.md +++ b/README.md @@ -18,7 +18,7 @@ [![Quality Gate Status](https://sonarcloud.io/api/project_badges/measure?project=coronasafe_care_fe&metric=alert_status)](https://sonarcloud.io/summary/new_code?id=coronasafe_care_fe) ![Code scanning - action](https://github.com/coronasafe/care_fe/workflows/Code%20scanning%20-%20action/badge.svg) ![OSSAR](https://github.com/coronasafe/care_fe/workflows/OSSAR/badge.svg) -[![Cypress Tests](https://github.com/coronasafe/care_fe/actions/workflows/cypress.yaml/badge.svg)](https://github.com/coronasafe/care_fe/actions/workflows/cypress.yaml) +[![Cypress Tests](https://img.shields.io/endpoint?url=https://cloud.cypress.io/badge/simple/wf7d2m/develop&style=flat&logo=cypress)](https://cloud.cypress.io/projects/wf7d2m/runs) ![Staging Release](https://github.com/coronasafe/care_fe/workflows/CARE%20Develop%20Registry/badge.svg) ![Production Release](https://github.com/coronasafe/care_fe/workflows/Production%20Release/badge.svg) [![Codacy Badge](https://api.codacy.com/project/badge/Grade/200482ab117e4b5397ff3f5ae5719aa2)](https://www.codacy.com/gh/coronasafe/care_fe?utm_source=github.com&utm_medium=referral&utm_content=coronasafe/care_fe&utm_campaign=Badge_Grade) From 9504655aac0622ac325fcf2351f96919f5db9427 Mon Sep 17 00:00:00 2001 From: Mohammed Nihal <57055998+nihal467@users.noreply.github.com> Date: Tue, 5 Dec 2023 11:43:33 +0530 Subject: [PATCH 14/21] Enhance existing Cypress Test | Update existing facility | Facility Tab (#6792) * facility update * minor fix --- .../e2e/facility_spec/facility_creation.cy.ts | 22 ++++++++++++++----- 1 file changed, 17 insertions(+), 5 deletions(-) diff --git a/cypress/e2e/facility_spec/facility_creation.cy.ts b/cypress/e2e/facility_spec/facility_creation.cy.ts index d43aa093a69..5d271a0c3bb 100644 --- a/cypress/e2e/facility_spec/facility_creation.cy.ts +++ b/cypress/e2e/facility_spec/facility_creation.cy.ts @@ -31,6 +31,7 @@ describe("Facility Creation", () => { const totalDoctor = "10"; const facilityName = "cypress facility"; const facilityAddress = "cypress address"; + const facilityUpdateAddress = "cypress updated address"; const facilityNumber = "9898469865"; const facilityErrorMessage = [ "Required", @@ -210,17 +211,28 @@ describe("Facility Creation", () => { }); it("Update the existing facility", () => { + // update a existing dummy data facility facilityPage.visitUpdateFacilityPage(facilityUrl1); facilityPage.clickManageFacilityDropdown(); facilityPage.clickUpdateFacilityOption(); - facilityPage.clickUpdateFacilityType("Request Approving Center"); - facilityPage.fillFacilityName("cypress facility updated"); - facilityPage.fillAddress("Cypress Facility Updated Address"); - facilityPage.fillOxygenCapacity("100"); - facilityPage.fillExpectedOxygenRequirement("80"); + facilityPage.clickUpdateFacilityType("Govt Hospital"); + facilityPage.fillAddress(facilityUpdateAddress); + facilityPage.fillOxygenCapacity(oxygenCapacity); + facilityPage.fillExpectedOxygenRequirement(oxygenExpected); facilityPage.selectLocation("Kochi, Kerala"); facilityPage.submitForm(); cy.url().should("not.include", "/update"); + // verify the updated data + facilityPage.getFacilityOxygenInfo().scrollIntoView(); + facilityPage + .getFacilityOxygenInfo() + .contains(oxygenCapacity) + .should("be.visible"); + facilityPage.getAddressDetailsView().scrollIntoView(); + facilityPage + .getAddressDetailsView() + .contains(facilityUpdateAddress) + .should("be.visible"); }); it("Configure the existing facility", () => { From b4edef2cea20f0aa8bfc64730b629539ef5a85cc Mon Sep 17 00:00:00 2001 From: Mohammed Nihal <57055998+nihal467@users.noreply.github.com> Date: Tue, 5 Dec 2023 11:44:06 +0530 Subject: [PATCH 15/21] New Cypress Test | Facility Triage Functionality | Facility Tab (#6795) * facility update * minor fix * Facility Triage Test --- .../e2e/facility_spec/facility_creation.cy.ts | 47 +++++++++++++++++++ .../pageobject/Facility/FacilityCreation.ts | 47 +++++++++++++++++++ .../Facility/FacilityHomeTriage.tsx | 7 ++- 3 files changed, 100 insertions(+), 1 deletion(-) diff --git a/cypress/e2e/facility_spec/facility_creation.cy.ts b/cypress/e2e/facility_spec/facility_creation.cy.ts index 5d271a0c3bb..cc2d1b7f921 100644 --- a/cypress/e2e/facility_spec/facility_creation.cy.ts +++ b/cypress/e2e/facility_spec/facility_creation.cy.ts @@ -30,9 +30,13 @@ describe("Facility Creation", () => { const doctorCapacity = "5"; const totalDoctor = "10"; const facilityName = "cypress facility"; + const facilityName2 = "Dummy Facility 1"; const facilityAddress = "cypress address"; const facilityUpdateAddress = "cypress updated address"; const facilityNumber = "9898469865"; + const triageDate = "02122023"; + const initialTriageValue = "60"; + const modifiedTriageValue = "50"; const facilityErrorMessage = [ "Required", "Invalid Pincode", @@ -50,6 +54,7 @@ describe("Facility Creation", () => { "Field is required", ]; const doctorErrorMessage = ["Field is required", "Field is required"]; + const triageErrorMessage = ["Field is required"]; before(() => { loginPage.loginAsDisctrictAdmin(); @@ -62,6 +67,48 @@ describe("Facility Creation", () => { cy.awaitUrl("/facility"); }); + it("Verify Facility Triage Function", () => { + // mandatory field error throw + manageUserPage.typeFacilitySearch(facilityName2); + facilityPage.verifyFacilityBadgeContent(facilityName2); + manageUserPage.assertFacilityInCard(facilityName2); + facilityHome.verifyURLContains(facilityName2); + facilityPage.visitAlreadyCreatedFacility(); + facilityPage.scrollToFacilityTriage(); + facilityPage.clickAddFacilityTriage(); + manageUserPage.clickSubmit(); + userCreationPage.verifyErrorMessages(triageErrorMessage); + // create a entry and verify reflection + facilityPage.fillEntryDate(triageDate); + facilityPage.fillTriageEntryFields( + initialTriageValue, + initialTriageValue, + initialTriageValue, + initialTriageValue, + initialTriageValue + ); + manageUserPage.clickSubmit(); + // edit the entry and verify reflection + facilityPage.scrollToFacilityTriage(); + facilityPage.verifyTriageTableContains(initialTriageValue); + facilityPage.clickEditButton(); + facilityPage.fillTriageEntryFields( + modifiedTriageValue, + modifiedTriageValue, + modifiedTriageValue, + modifiedTriageValue, + modifiedTriageValue + ); + manageUserPage.clickSubmit(); + facilityPage.scrollToFacilityTriage(); + facilityPage.verifyTriageTableContains(modifiedTriageValue); + // validate error of filling data on same date already data exist and verify reflection + facilityPage.scrollToFacilityTriage(); + facilityPage.clickAddFacilityTriage(); + facilityPage.fillEntryDate(triageDate); + facilityPage.clickButtonsMultipleTimes("button#submit"); + }); + it("Create a new facility with multiple bed and doctor capacity", () => { // create facility with multiple capacity and verify form error message for facility form facilityPage.visitCreateFacilityPage(); diff --git a/cypress/pageobject/Facility/FacilityCreation.ts b/cypress/pageobject/Facility/FacilityCreation.ts index b4d3808d645..c13f5beac8e 100644 --- a/cypress/pageobject/Facility/FacilityCreation.ts +++ b/cypress/pageobject/Facility/FacilityCreation.ts @@ -217,6 +217,53 @@ class FacilityPage { cy.get("#delete-facility").contains("Delete Facility").click(); } + scrollToFacilityTriage() { + cy.get("#add-facility-triage").scrollIntoView(); + } + + fillTriageEntryFields( + visited, + homeQuarantine, + isolation, + referred, + confirmedPositive + ) { + cy.get("#num_patients_visited").clear().click().type(visited); + cy.get("#num_patients_home_quarantine") + .clear() + .click() + .type(homeQuarantine); + cy.get("#num_patients_isolation").clear().click().type(isolation); + cy.get("#num_patient_referred").clear().click().type(referred); + cy.get("#num_patient_confirmed_positive") + .clear() + .click() + .type(confirmedPositive); + } + + fillEntryDate(date) { + cy.get("#entry_date").click(); + cy.get("#date-input").click().type(date); + } + + clickEditButton() { + cy.get("#edit-button").click(); + } + + clickButtonsMultipleTimes(selector) { + cy.get(selector).each(($button) => { + cy.wrap($button).click(); + }); + } + + verifyTriageTableContains(value) { + cy.get("#triage-table").contains(value); + } + + clickAddFacilityTriage() { + cy.get("#add-facility-triage").click(); + } + clickfacilityfeatureoption() { cy.get("#features").click(); } diff --git a/src/Components/Facility/FacilityHomeTriage.tsx b/src/Components/Facility/FacilityHomeTriage.tsx index f96a5181cfc..ba1cd199044 100644 --- a/src/Components/Facility/FacilityHomeTriage.tsx +++ b/src/Components/Facility/FacilityHomeTriage.tsx @@ -30,6 +30,7 @@ export const FacilityHomeTriage = (props: any) => { ); temp.push( {
Corona Triage
navigate(`/facility/${props.facilityId}/triage`)} authorizeFor={props.NonReadOnlyUsers} @@ -60,7 +62,10 @@ export const FacilityHomeTriage = (props: any) => { Add Triage
-
+
Date: Wed, 6 Dec 2023 09:21:22 +0530 Subject: [PATCH 16/21] Modified the external results sample format (#6694) * Update sample_format_external_result_import URL * Fix file path in config.json and refactor handleDownload function in ExternalResultUpload.tsx * Fix download functionality in ExternalResultUpload component --- public/External-Results-Template.csv | 3 +++ public/config.json | 2 +- src/Components/ExternalResult/ExternalResultUpload.tsx | 2 ++ 3 files changed, 6 insertions(+), 1 deletion(-) create mode 100644 public/External-Results-Template.csv diff --git a/public/External-Results-Template.csv b/public/External-Results-Template.csv new file mode 100644 index 00000000000..3ab2afc65ba --- /dev/null +++ b/public/External-Results-Template.csv @@ -0,0 +1,3 @@ +District,srf id,name,age,age in,gender,mobile number,address,ward,local body,local body type,source,Sample Collection Date,result date,test type,lab name,sample type,patient status,Is Repeat,patient category,result +Ernakulam,00/EKM/0000,Bodhi CSN,24,years,m,8888888888,"CSN HQ +Kochi, Kerala ",7,Poothrikka,grama panchayath,Secondary contact aparna,2020-10-14,2020-10-14,Antigen,Karothukuzhi Laboratory,Ag-SD_Biosensor_Standard_Q_COVID-19_Ag_detection_kit,Asymptomatic,NO,Cat 17: All individuals who wish to get themselves tested,Negative \ No newline at end of file diff --git a/public/config.json b/public/config.json index 444362cfb20..74e509fee59 100644 --- a/public/config.json +++ b/public/config.json @@ -21,6 +21,6 @@ "kasp_string": "KASP", "kasp_full_string": "Karunya Arogya Suraksha Padhathi", "sample_format_asset_import": "https://spreadsheets.google.com/feeds/download/spreadsheets/Export?key=11JaEhNHdyCHth4YQs_44YaRlP77Rrqe81VSEfg1glko&exportFormat=xlsx", - "sample_format_external_result_import": "https://docs.google.com/spreadsheets/d/17VfgryA6OYSYgtQZeXU9mp7kNvLySeEawvnLBO_1nuE/export?format=csv&id=17VfgryA6OYSYgtQZeXU9mp7kNvLySeEawvnLBO_1nuE", + "sample_format_external_result_import": "/External-Results-Template.csv", "enable_abdm": true } \ No newline at end of file diff --git a/src/Components/ExternalResult/ExternalResultUpload.tsx b/src/Components/ExternalResult/ExternalResultUpload.tsx index 20a2cec3341..5a9262b2990 100644 --- a/src/Components/ExternalResult/ExternalResultUpload.tsx +++ b/src/Components/ExternalResult/ExternalResultUpload.tsx @@ -112,6 +112,8 @@ export default function ExternalResultUpload() { {" "} {t("sample_format")} From 5e139447f924dad9747e29c00732334932ca8c34 Mon Sep 17 00:00:00 2001 From: Kshitij Verma <101321276+kshitijv256@users.noreply.github.com> Date: Wed, 6 Dec 2023 13:24:36 +0530 Subject: [PATCH 17/21] seperated bedname and location name (#6794) --- src/Components/Common/BedSelect.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/Components/Common/BedSelect.tsx b/src/Components/Common/BedSelect.tsx index d903d3b62e0..94caded12a3 100644 --- a/src/Components/Common/BedSelect.tsx +++ b/src/Components/Common/BedSelect.tsx @@ -74,7 +74,7 @@ export const BedSelect = (props: BedSelectProps) => { optionLabel={(option: any) => { if (Object.keys(option).length === 0) return ""; return ( - `${option.name} ${option?.location_object?.name || t("unknown")}` || + `${option.name}, ${option?.location_object?.name || t("unknown")}` || option?.location_object?.name ); }} From f746576b04a63ab7acd018cf2e2ea181deb21aea Mon Sep 17 00:00:00 2001 From: konavivekramakrishna <101407963+konavivekramakrishna@users.noreply.github.com> Date: Wed, 6 Dec 2023 13:25:44 +0530 Subject: [PATCH 18/21] Added a Date/time and round type filter for log updates (#6713) * Add DailyRoundsFilterModel interface and DailyRoundsFilter component * refactored dailyRoundList * useQuery: Fix GET/HEAD cannot contain body issue * Support for filtering by `taken_at` dt range * Refactor DailyRoundsFilter and DailyRoundsList components * Refactor roundTypeOptions in DailyRoundsFilter component * Improve translations coverage and minor refactors * update slugs for request --------- Co-authored-by: rithviknishad --- .../ConsultationUpdatesTab.tsx | 29 +--- .../DailyRounds/DefaultLogUpdateCard.tsx | 3 +- .../Consultations/DailyRoundsFilter.tsx | 115 +++++++++++++++ .../Consultations/DailyRoundsList.tsx | 134 +++++++++--------- src/Components/Facility/models.tsx | 2 +- .../Form/FormFields/SelectFormField.tsx | 2 +- src/Components/Patient/models.tsx | 7 +- src/Locale/en/Common.json | 3 +- src/Locale/en/Consultation.json | 5 +- src/Utils/request/utils.ts | 5 +- 10 files changed, 201 insertions(+), 104 deletions(-) create mode 100644 src/Components/Facility/Consultations/DailyRoundsFilter.tsx diff --git a/src/Components/Facility/ConsultationDetails/ConsultationUpdatesTab.tsx b/src/Components/Facility/ConsultationDetails/ConsultationUpdatesTab.tsx index bb584ae93e6..0d8a70781da 100644 --- a/src/Components/Facility/ConsultationDetails/ConsultationUpdatesTab.tsx +++ b/src/Components/Facility/ConsultationDetails/ConsultationUpdatesTab.tsx @@ -12,13 +12,12 @@ import PrescriptionsTable from "../../Medicine/PrescriptionsTable"; import Chip from "../../../CAREUI/display/Chip"; import { formatAge, formatDate, formatDateTime } from "../../../Utils/utils"; import ReadMore from "../../Common/components/Readmore"; -import { DailyRoundsList } from "../Consultations/DailyRoundsList"; +import DailyRoundsList from "../Consultations/DailyRoundsList"; const PageTitle = lazy(() => import("../../Common/PageTitle")); export const ConsultationUpdatesTab = (props: ConsultationTabProps) => { const dispatch: any = useDispatch(); - const [showAutomatedRounds, setShowAutomatedRounds] = useState(true); const [hl7SocketUrl, setHL7SocketUrl] = useState(); const [ventilatorSocketUrl, setVentilatorSocketUrl] = useState(); const [monitorBedData, setMonitorBedData] = useState(); @@ -674,31 +673,7 @@ export const ConsultationUpdatesTab = (props: ConsultationTabProps) => {
-
- -
- setShowAutomatedRounds((s) => !s)} - /> - -
-
- +
diff --git a/src/Components/Facility/Consultations/DailyRounds/DefaultLogUpdateCard.tsx b/src/Components/Facility/Consultations/DailyRounds/DefaultLogUpdateCard.tsx index 63b5087cff8..ff738f4acb6 100644 --- a/src/Components/Facility/Consultations/DailyRounds/DefaultLogUpdateCard.tsx +++ b/src/Components/Facility/Consultations/DailyRounds/DefaultLogUpdateCard.tsx @@ -4,10 +4,11 @@ import CareIcon from "../../../../CAREUI/icons/CareIcon"; import ButtonV2 from "../../../Common/components/ButtonV2"; import { DailyRoundsModel } from "../../../Patient/models"; import LogUpdateCardAttribute from "./LogUpdateCardAttribute"; +import { ConsultationModel } from "../../models"; interface Props { round: DailyRoundsModel; - consultationData: any; + consultationData: ConsultationModel; onViewDetails: () => void; onUpdateLog?: () => void; } diff --git a/src/Components/Facility/Consultations/DailyRoundsFilter.tsx b/src/Components/Facility/Consultations/DailyRoundsFilter.tsx new file mode 100644 index 00000000000..62b8d63e824 --- /dev/null +++ b/src/Components/Facility/Consultations/DailyRoundsFilter.tsx @@ -0,0 +1,115 @@ +import { Popover, Transition } from "@headlessui/react"; +import ButtonV2 from "../../Common/components/ButtonV2"; +import { Fragment } from "react"; +import { SelectFormField } from "../../Form/FormFields/SelectFormField"; +import TextFormField from "../../Form/FormFields/TextFormField"; +import CareIcon from "../../../CAREUI/icons/CareIcon"; +import dayjs from "dayjs"; +import { useState } from "react"; +import { useTranslation } from "react-i18next"; +import { DailyRoundTypes, DailyRoundsModel } from "../../Patient/models"; +import { FieldChangeEvent } from "../../Form/FormFields/Utils"; + +type FilterState = { + rounds_type?: DailyRoundsModel["rounds_type"]; + taken_at_after?: string; + taken_at_before?: string; +}; + +interface Props { + onApply: (filter: FilterState) => void; +} + +export default function DailyRoundsFilter(props: Props) { + const { t } = useTranslation(); + const [filter, setFilter] = useState({}); + + const field = (name: keyof FilterState) => ({ + name, + value: filter[name], + onChange: (e: FieldChangeEvent) => + setFilter({ ...filter, [e.name]: e.value }), + labelClassName: "text-sm", + errorClassName: "hidden", + }); + + return ( +
+ + + + + {t("filter")} + + + + +
+
+
+ + {t("filter_by")} + +
+
+
+ t(o)} + optionValue={(o) => o} + /> + + + + + { + setFilter({}); + props.onApply({}); + }} + border + className="w-full" + > + {t("clear")} + + + + props.onApply(filter)} + border + className="w-full" + > + {t("apply")} + + +
+
+
+
+
+
+ ); +} diff --git a/src/Components/Facility/Consultations/DailyRoundsList.tsx b/src/Components/Facility/Consultations/DailyRoundsList.tsx index ffc70ddf175..2060d8657e7 100644 --- a/src/Components/Facility/Consultations/DailyRoundsList.tsx +++ b/src/Components/Facility/Consultations/DailyRoundsList.tsx @@ -6,88 +6,84 @@ import { useTranslation } from "react-i18next"; import LoadingLogUpdateCard from "./DailyRounds/LoadingCard"; import routes from "../../../Redux/api"; import PaginatedList from "../../../CAREUI/misc/PaginatedList"; +import PageTitle from "../../Common/PageTitle"; +import DailyRoundsFilter from "./DailyRoundsFilter"; +import { ConsultationModel } from "../models"; +import { useSlugs } from "../../../Common/hooks/useSlug"; -export const DailyRoundsList = (props: any) => { +interface Props { + consultation: ConsultationModel; +} + +export default function DailyRoundsList({ consultation }: Props) { + const [facilityId, patientId, consultationId] = useSlugs( + "facility", + "patient", + "consultation" + ); const { t } = useTranslation(); - const { - facilityId, - patientId, - consultationId, - consultationData, - showAutomatedRounds, - } = props; + + const consultationUrl = `/facility/${facilityId}/patient/${patientId}/consultation/${consultationId}`; return ( - {(_) => ( -
-
- - - {t("no_consultation_updates")} - - - - <> - {Array.from({ length: 3 }).map((_, i) => ( - - ))} - - - className="flex grow flex-col gap-3"> - {(item, items) => { - if (item.rounds_type === "AUTOMATED") { + {({ refetch }) => ( + <> +
+ + refetch({ query })} /> +
+ +
+
+ + + {t("no_consultation_updates")} + + + + <> + {Array.from({ length: 3 }).map((_, i) => ( + + ))} + + + className="flex grow flex-col gap-3"> + {(item, items) => { + if (item.rounds_type === "AUTOMATED") { + return ( + + ); + } + + const itemUrl = + item.rounds_type === "NORMAL" + ? `${consultationUrl}/daily-rounds/${item.id}` + : `${consultationUrl}/daily_rounds/${item.id}`; + return ( - navigate(itemUrl)} + onUpdateLog={() => navigate(`${itemUrl}/update`)} /> ); - } - return ( - { - if (item.rounds_type === "NORMAL") { - navigate( - `/facility/${facilityId}/patient/${patientId}/consultation/${consultationId}/daily-rounds/${item.id}` - ); - } else { - navigate( - `/facility/${facilityId}/patient/${patientId}/consultation/${consultationId}/daily_rounds/${item.id}` - ); - } - }} - onUpdateLog={() => { - if (item.rounds_type === "NORMAL") { - navigate( - `/facility/${facilityId}/patient/${patientId}/consultation/${consultationId}/daily-rounds/${item.id}/update` - ); - } else { - navigate( - `/facility/${facilityId}/patient/${patientId}/consultation/${consultationId}/daily_rounds/${item.id}/update` - ); - } - }} - /> - ); - }} - -
- + }} + +
+ +
-
+ )} ); -}; +} diff --git a/src/Components/Facility/models.tsx b/src/Components/Facility/models.tsx index cdc7074c145..550012603f5 100644 --- a/src/Components/Facility/models.tsx +++ b/src/Components/Facility/models.tsx @@ -107,7 +107,7 @@ export interface ConsultationModel { history_of_present_illness?: string; facility?: number; facility_name?: string; - id?: string; + id: string; modified_date?: string; other_symptoms?: string; patient?: string; diff --git a/src/Components/Form/FormFields/SelectFormField.tsx b/src/Components/Form/FormFields/SelectFormField.tsx index 3c6613bb662..5afe4f11d63 100644 --- a/src/Components/Form/FormFields/SelectFormField.tsx +++ b/src/Components/Form/FormFields/SelectFormField.tsx @@ -7,7 +7,7 @@ type OptionCallback = (option: T) => R; type SelectFormFieldProps = FormFieldBaseProps & { placeholder?: React.ReactNode; - options: T[]; + options: readonly T[]; position?: "above" | "below"; optionLabel: OptionCallback; optionSelectedLabel?: OptionCallback; diff --git a/src/Components/Patient/models.tsx b/src/Components/Patient/models.tsx index 341e13e3c80..af69d8464bc 100644 --- a/src/Components/Patient/models.tsx +++ b/src/Components/Patient/models.tsx @@ -269,6 +269,8 @@ export interface DailyRoundsOutput { quantity: number; } +export const DailyRoundTypes = ["NORMAL", "VENTILATOR", "AUTOMATED"] as const; + export interface DailyRoundsModel { ventilator_spo2?: number; spo2?: string; @@ -290,7 +292,7 @@ export interface DailyRoundsModel { medication_given?: Array; additional_symptoms_text?: string; current_health?: string; - id?: any; + id: string; other_symptoms?: string; admitted_to?: string; patient_category?: PatientCategory; @@ -299,7 +301,7 @@ export interface DailyRoundsModel { created_date?: string; modified_date?: string; taken_at?: string; - rounds_type?: "NORMAL" | "VENTILATOR" | "ICU" | "AUTOMATED"; + rounds_type: (typeof DailyRoundTypes)[number]; last_updated_by_telemedicine?: boolean; created_by_telemedicine?: boolean; created_by?: { @@ -314,6 +316,7 @@ export interface DailyRoundsModel { }; bed?: string; } + export interface FacilityNameModel { id?: string; name?: string; diff --git a/src/Locale/en/Common.json b/src/Locale/en/Common.json index 7e357bc04b5..c455e3a989a 100644 --- a/src/Locale/en/Common.json +++ b/src/Locale/en/Common.json @@ -45,6 +45,7 @@ "clear": "Clear", "apply": "Apply", "filter_by": "Filter By", + "filter": "Filter", "ordering": "Ordering", "phone_number": "Phone Number", "emergency_contact_number": "Emergency Contact Number", @@ -158,4 +159,4 @@ "clear_selection": "Clear selection", "select_date": "Select date", "DD/MM/YYYY": "DD/MM/YYYY" -} +} \ No newline at end of file diff --git a/src/Locale/en/Consultation.json b/src/Locale/en/Consultation.json index a40f03c4f24..54b587eb81e 100644 --- a/src/Locale/en/Consultation.json +++ b/src/Locale/en/Consultation.json @@ -12,5 +12,8 @@ "discharge_summary_not_ready": "Discharge summary is not ready yet.", "download_discharge_summary": "Download discharge summary", "email_discharge_summary_description": "Enter your valid email address to receive the discharge summary", - "generated_summary_caution": "This is a computer generated summary using the information captured in the CARE system." + "generated_summary_caution": "This is a computer generated summary using the information captured in the CARE system.", + "NORMAL": "Normal", + "VENTILATOR": "Critical Care", + "AUTOMATED": "Automated" } diff --git a/src/Utils/request/utils.ts b/src/Utils/request/utils.ts index ec919c79490..f22dca369f2 100644 --- a/src/Utils/request/utils.ts +++ b/src/Utils/request/utils.ts @@ -82,7 +82,10 @@ export function mergeRequestOptions( ...overrides, query: { ...options.query, ...overrides.query }, - body: { ...(options.body ?? {}), ...(overrides.body ?? {}) }, + body: (options.body || overrides.body) && { + ...(options.body ?? {}), + ...(overrides.body ?? {}), + }, pathParams: { ...options.pathParams, ...overrides.pathParams }, onResponse: (res) => { From 947978c3f0215eefaac1a2dd2a85dbb7bae5bc37 Mon Sep 17 00:00:00 2001 From: Ashraf Mohammed <98876115+AshrafMd-1@users.noreply.github.com> Date: Wed, 6 Dec 2023 13:26:36 +0530 Subject: [PATCH 19/21] Use relative time for audit log details. (#6640) * convert time to relative time * change styling * change margin * add new classname --- src/CAREUI/display/RecordMeta.tsx | 14 +++++++++--- src/Components/Shifting/ShiftDetails.tsx | 27 +++++++++++------------- 2 files changed, 23 insertions(+), 18 deletions(-) diff --git a/src/CAREUI/display/RecordMeta.tsx b/src/CAREUI/display/RecordMeta.tsx index 944ddf27c8f..818553d9207 100644 --- a/src/CAREUI/display/RecordMeta.tsx +++ b/src/CAREUI/display/RecordMeta.tsx @@ -11,6 +11,7 @@ interface Props { time?: string; prefix?: ReactNode; className?: string; + inlineClassName?: string; user?: { first_name: string; last_name: string; @@ -23,7 +24,14 @@ interface Props { * A generic component to display relative time along with a tooltip and a user * if provided. */ -const RecordMeta = ({ time, user, prefix, className, inlineUser }: Props) => { +const RecordMeta = ({ + time, + user, + prefix, + className, + inlineClassName, + inlineUser, +}: Props) => { const isOnline = user && isUserOnline(user); let child = ( @@ -47,11 +55,11 @@ const RecordMeta = ({ time, user, prefix, className, inlineUser }: Props) => { if (prefix || user) { child = ( -
+
{prefix} {child} {user && inlineUser && by} - {user && } + {user && !inlineUser && } {user && inlineUser && ( {formatName(user)} )} diff --git a/src/Components/Shifting/ShiftDetails.tsx b/src/Components/Shifting/ShiftDetails.tsx index 3d2a1f60b8b..a898b0cde15 100644 --- a/src/Components/Shifting/ShiftDetails.tsx +++ b/src/Components/Shifting/ShiftDetails.tsx @@ -806,13 +806,12 @@ export default function ShiftDetails(props: { id: string }) { {t("created")}
-
- {data?.created_by_object?.first_name} - {data?.created_by_object?.last_name} -
-
- {data?.created_date && formatDateTime(data?.created_date)} -
+
@@ -820,14 +819,12 @@ export default function ShiftDetails(props: { id: string }) { {t("last_edited")}
-
- {data?.last_edited_by_object?.first_name}{" "} - {data?.last_edited_by_object?.last_name} -
-
- {data?.modified_date && - formatDateTime(data?.modified_date)} -
+
From a6ed2bc82f0297ab430f961c6f4384ccad37dd65 Mon Sep 17 00:00:00 2001 From: Devdeep Ghosh <63492939+thedevildude@users.noreply.github.com> Date: Wed, 6 Dec 2023 13:27:17 +0530 Subject: [PATCH 20/21] Replaced useDispatch w. useQuery/request: Users (src/Components/Users/**) [5 out of 5 components] (#6596) * replaced getUserDetails and getUserListSkills actions with useQuery * replaced partialUpdateUser action with request * replaced updatePassword action with request * replaced dispatch with useQuery in UserFilter.tsx * Bug Fix: UserFilter tried fetching district when district_id was not available * addUser and checkUsername action replaced with request * replaced useDispatch with request in UserAdd component * replaced useDispatch with useQuery and request in SkillsSlideOver * replaced useDispatch with useQuery and request in ManageUsers * solved issue #6652 | passed user skills as props to SkillSelect * removed unnecessary console logs * re-added showAll as dependency to skillSearch * removed unnecessary console logs * replaced fetchDistrict request with useQuery and removed isLoading * fixed error notification in ManageUsers * removed unnecessary useState from ManageUsers * code fixes in SkillsSlideOver * code fixes in UserAdd * code fix in UserFilter * code fix in UserProfile * removed redundant code * added proper types and fixed redundant code * removed redundant fireRequest actions * fix http 301 redirect due to missing trailing slash * Update TRes of userListFacility in src/Redux/api.tsx Co-authored-by: Rithvik Nishad * resolved imports --------- Co-authored-by: Rithvik Nishad Co-authored-by: Rithvik Nishad Co-authored-by: Mohammed Nihal <57055998+nihal467@users.noreply.github.com> --- src/Components/Common/SkillSelect.tsx | 21 +- src/Components/Facility/models.tsx | 1 + src/Components/Users/ManageUsers.tsx | 219 +++++++++------------ src/Components/Users/SkillsSlideOver.tsx | 67 +++---- src/Components/Users/UserAdd.tsx | 164 +++++----------- src/Components/Users/UserFilter.tsx | 25 +-- src/Components/Users/UserProfile.tsx | 237 +++++++++++------------ src/Components/Users/models.tsx | 14 +- src/Redux/actions.tsx | 35 +--- src/Redux/api.tsx | 36 +++- 10 files changed, 344 insertions(+), 475 deletions(-) diff --git a/src/Components/Common/SkillSelect.tsx b/src/Components/Common/SkillSelect.tsx index 3257971d03c..941c29790d1 100644 --- a/src/Components/Common/SkillSelect.tsx +++ b/src/Components/Common/SkillSelect.tsx @@ -1,8 +1,8 @@ import { useCallback } from "react"; import { useDispatch } from "react-redux"; -import { getAllSkills, getUserListSkills } from "../../Redux/actions"; +import { getAllSkills } from "../../Redux/actions"; import AutoCompleteAsync from "../Form/AutoCompleteAsync"; -import { SkillObjectModel } from "../Users/models"; +import { SkillModel, SkillObjectModel } from "../Users/models"; interface SkillSelectProps { id?: string; @@ -17,6 +17,7 @@ interface SkillSelectProps { selected: SkillObjectModel | SkillObjectModel[] | null; setSelected: (selected: SkillObjectModel) => void; username?: string; + userSkills?: SkillModel[]; } export const SkillSelect = (props: SkillSelectProps) => { @@ -32,7 +33,8 @@ export const SkillSelect = (props: SkillSelectProps) => { disabled = false, className = "", errors = "", - username, + //username, + userSkills, } = props; const dispatchAction: any = useDispatch(); @@ -47,21 +49,16 @@ export const SkillSelect = (props: SkillSelectProps) => { }; const res = await dispatchAction(getAllSkills(params)); - - const linkedSkills = await dispatchAction( - getUserListSkills({ username: username }) - ); - - const skillsList = linkedSkills?.data?.results; const skillsID: string[] = []; - skillsList.map((skill: any) => skillsID.push(skill.skill_object.id)); + userSkills?.map((skill: SkillModel) => + skillsID.push(skill.skill_object.id) + ); const skills = res?.data?.results.filter( (skill: any) => !skillsID.includes(skill.id) ); - return skills; }, - [dispatchAction, searchAll, showAll] + [dispatchAction, searchAll, userSkills, showAll] ); return ( diff --git a/src/Components/Facility/models.tsx b/src/Components/Facility/models.tsx index 550012603f5..0d8a112e021 100644 --- a/src/Components/Facility/models.tsx +++ b/src/Components/Facility/models.tsx @@ -7,6 +7,7 @@ import { RouteToFacility } from "../Common/RouteToFacilitySelect"; import { ConsultationDiagnosis, CreateDiagnosis } from "../Diagnosis/types"; export interface LocalBodyModel { + id: number; name: string; body_type: number; localbody_code: string; diff --git a/src/Components/Users/ManageUsers.tsx b/src/Components/Users/ManageUsers.tsx index 94791e55ae1..3b469081754 100644 --- a/src/Components/Users/ManageUsers.tsx +++ b/src/Components/Users/ManageUsers.tsx @@ -1,17 +1,5 @@ import * as Notification from "../../Utils/Notifications.js"; -import { - addUserFacility, - clearHomeFacility, - deleteUser, - deleteUserFacility, - getDistrict, - getUserList, - getUserListFacility, - partialUpdateUser, -} from "../../Redux/actions"; -import { statusType, useAbortableEffect } from "../../Common/utils"; -import { lazy, useCallback, useEffect, useState } from "react"; -import { useDispatch } from "react-redux"; +import { lazy, useState } from "react"; import { AdvancedFilterButton } from "../../CAREUI/interactive/FiltersSlideover"; import ButtonV2, { Submit } from "../Common/components/ButtonV2"; import CareIcon from "../../CAREUI/icons/CareIcon"; @@ -36,6 +24,9 @@ import Page from "../Common/components/Page.js"; import dayjs from "dayjs"; import TextFormField from "../Form/FormFields/TextFormField.js"; import useAuthUser from "../../Common/hooks/useAuthUser.js"; +import routes from "../../Redux/api.js"; +import useQuery from "../../Utils/request/useQuery.js"; +import request from "../../Utils/request/request.js"; const Loading = lazy(() => import("../Common/Loading")); @@ -49,19 +40,13 @@ export default function ManageUsers() { advancedFilter, resultsPerPage, } = useFilters({ limit: 18 }); - const dispatch: any = useDispatch(); - const initialData: any[] = []; let manageUsers: any = null; - const [users, setUsers] = useState(initialData); - const [isLoading, setIsLoading] = useState(false); const [expandSkillList, setExpandSkillList] = useState(false); - const [totalCount, setTotalCount] = useState(0); - const [districtName, setDistrictName] = useState(); const [expandFacilityList, setExpandFacilityList] = useState(false); const [selectedUser, setSelectedUser] = useState(null); const [expandWorkingHours, setExpandWorkingHours] = useState(false); const authUser = useAuthUser(); - const [weeklyHours, setWeeklyHours] = useState(0); + const [weeklyHours, setWeeklyHours] = useState("0"); const userIndex = USER_TYPES.indexOf(authUser.user_type); const userTypes = authUser.is_superuser ? [...USER_TYPES] @@ -79,58 +64,32 @@ export default function ManageUsers() { const isExtremeSmallScreen = width <= extremeSmallScreenBreakpoint ? true : false; - const fetchData = useCallback( - async (status: statusType) => { - setIsLoading(true); - const params = { - limit: resultsPerPage, - offset: (qParams.page ? qParams.page - 1 : 0) * resultsPerPage, - username: qParams.username, - first_name: qParams.first_name, - last_name: qParams.last_name, - phone_number: qParams.phone_number, - alt_phone_number: qParams.alt_phone_number, - user_type: qParams.user_type, - district_id: qParams.district_id, - }; - if (qParams.district_id) { - const dis = await dispatch(getDistrict(qParams.district_id)); - if (!status.aborted) { - if (dis && dis.data) { - setDistrictName(dis.data.name); - } - } - } else { - setDistrictName(undefined); - } - const res = await dispatch(getUserList(params)); - if (!status.aborted) { - if (res && res.data) { - setUsers(res.data.results); - setTotalCount(res.data.count); - } - setIsLoading(false); - } + const { + data: userListData, + loading: userListLoading, + refetch: refetchUserList, + } = useQuery(routes.userList, { + query: { + limit: resultsPerPage.toString(), + offset: ( + (qParams.page ? qParams.page - 1 : 0) * resultsPerPage + ).toString(), + username: qParams.username, + first_name: qParams.first_name, + last_name: qParams.last_name, + phone_number: qParams.phone_number, + alt_phone_number: qParams.alt_phone_number, + user_type: qParams.user_type, + district_id: qParams.district_id, }, - [ - resultsPerPage, - qParams.page, - qParams.username, - qParams.first_name, - qParams.last_name, - qParams.phone_number, - qParams.alt_phone_number, - qParams.user_type, - qParams.district_id, - dispatch, - ] - ); + }); - useAbortableEffect( - (status: statusType) => { - fetchData(status); - }, - [fetchData] + const { data: districtData, loading: districtDataLoading } = useQuery( + routes.getDistrict, + { + prefetch: !!qParams.district_id, + pathParams: { id: qParams.district_id }, + } ); const addUser = ( @@ -150,17 +109,15 @@ export default function ManageUsers() { const handleWorkingHourSubmit = async () => { const username = selectedUser; - if (!username || !weeklyHours || weeklyHours < 0 || weeklyHours > 168) { + if (!username || !weeklyHours || +weeklyHours < 0 || +weeklyHours > 168) { setWeeklyHoursError("Value should be between 0 and 168"); return; } - const res = await dispatch( - partialUpdateUser(username, { - weekly_working_hours: weeklyHours, - }) - ); - - if (res?.data) { + const { res, data, error } = await request(routes.partialUpdateUser, { + pathParams: { username }, + body: { weekly_working_hours: weeklyHours }, + }); + if (res && res.status === 200 && data) { Notification.Success({ msg: "Working hours updated successfully", }); @@ -168,29 +125,30 @@ export default function ManageUsers() { setSelectedUser(null); } else { Notification.Error({ - msg: "Error while updating working hours: " + (res.data.detail || ""), + msg: "Error while updating working hours: " + (error || ""), }); } - setWeeklyHours(0); + setWeeklyHours("0"); setWeeklyHoursError(""); - fetchData({ aborted: false }); + await refetchUserList(); }; const handleSubmit = async () => { - const username = userData.username; - const res = await dispatch(deleteUser(username)); + const { res, error } = await request(routes.deleteUser, { + pathParams: { username: userData.username }, + }); if (res?.status === 204) { Notification.Success({ msg: "User deleted successfully", }); } else { Notification.Error({ - msg: "Error while deleting User: " + (res?.data?.detail || ""), + msg: "Error while deleting User: " + (error || ""), }); } setUserData({ show: false, username: "", name: "" }); - fetchData({ aborted: false }); + await refetchUserList(); }; const handleDelete = (user: any) => { @@ -214,9 +172,9 @@ export default function ManageUsers() { let userList: any[] = []; - users && - users.length && - (userList = users.map((user: any, idx) => { + userListData?.results && + userListData.results.length && + (userList = userListData.results.map((user: any, idx) => { const cur_online = isUserOnline(user); return (
; - } else if (users?.length) { + } else if (userListData?.results.length) { manageUsers = (
{userList}
- +
); - } else if (users && users.length === 0) { + } else if (userListData?.results && userListData?.results.length === 0) { manageUsers = (
No Users Found
@@ -505,7 +463,7 @@ export default function ManageUsers() { open={expandWorkingHours} setOpen={(state) => { setExpandWorkingHours(state); - setWeeklyHours(0); + setWeeklyHours("0"); setWeeklyHoursError(""); }} slideFrom="right" @@ -539,8 +497,8 @@ export default function ManageUsers() {
@@ -574,7 +532,11 @@ export default function ManageUsers() { phoneNumber(), phoneNumber("WhatsApp no.", "alt_phone_number"), badge("Role", "user_type"), - value("District", "district_id", districtName || ""), + value( + "District", + "district_id", + qParams.district_id ? districtData?.name || "" : "" + ), ]} />
@@ -596,8 +558,6 @@ export default function ManageUsers() { function UserFacilities(props: { user: any }) { const { user } = props; const username = user.username; - const dispatch: any = useDispatch(); - const [facilities, setFacilities] = useState([]); const [isLoading, setIsLoading] = useState(false); const [facility, setFacility] = useState(null); const [unlinkFacilityData, setUnlinkFacilityData] = useState<{ @@ -635,62 +595,59 @@ function UserFacilities(props: { user: any }) { }); }; - const fetchFacilities = async () => { - setIsLoading(true); - const res = await dispatch(getUserListFacility({ username })); - if (res && res.data) { - setFacilities(res.data); - } - setIsLoading(false); - }; + const { + data: userFacilities, + loading: userFacilitiesLoading, + refetch: refetchUserFacilities, + } = useQuery(routes.userListFacility, { + pathParams: { username }, + }); const updateHomeFacility = async (username: string, facility: any) => { setIsLoading(true); - const res = await dispatch( - partialUpdateUser(username, { home_facility: facility.id }) - ); + const { res } = await request(routes.partialUpdateUser, { + pathParams: { username }, + body: { home_facility: facility.id.toString() }, + }); if (res && res.status === 200) user.home_facility_object = facility; - fetchFacilities(); + await refetchUserFacilities(); setIsLoading(false); }; const handleUnlinkFacilitySubmit = async () => { setIsLoading(true); if (unlinkFacilityData.isHomeFacility) { - const res = await dispatch( - clearHomeFacility(unlinkFacilityData.userName) - ); + const { res } = await request(routes.clearHomeFacility, { + pathParams: { username }, + }); if (res && res.status === 204) user.home_facility_object = null; } else { - await dispatch( - deleteUserFacility( - unlinkFacilityData.userName, - String(unlinkFacilityData?.facility?.id) - ) - ); + await request(routes.deleteUserFacility, { + pathParams: { username }, + body: { facility: unlinkFacilityData?.facility?.id?.toString() }, + }); } - fetchFacilities(); - setIsLoading(false); + await refetchUserFacilities(); hideUnlinkFacilityModal(); + setIsLoading(false); }; const addFacility = async (username: string, facility: any) => { setIsLoading(true); - const res = await dispatch(addUserFacility(username, String(facility.id))); + const { res } = await request(routes.addUserFacility, { + pathParams: { username }, + body: { facility: facility.id.toString() }, + }); if (res?.status !== 201) { Notification.Error({ msg: "Error while linking facility", }); } + await refetchUserFacilities(); setIsLoading(false); setFacility(null); - fetchFacilities(); }; - useEffect(() => { - fetchFacilities(); - }, []); - return (
{unlinkFacilityData.show && ( @@ -723,7 +680,7 @@ function UserFacilities(props: { user: any }) { Add
- {isLoading ? ( + {isLoading || userFacilitiesLoading ? (
@@ -761,13 +718,13 @@ function UserFacilities(props: { user: any }) { )} {/* Linked Facilities section */} - {facilities.length > 0 && ( + {userFacilities?.length && (
Linked Facilities
- {facilities.map((facility: any, i: number) => { + {userFacilities.map((facility: any, i: number) => { if (user?.home_facility_object?.id === facility.id) { // skip if it's a home facility return null; @@ -831,7 +788,7 @@ function UserFacilities(props: { user: any }) {
)} - {!user?.home_facility_object && facilities.length === 0 && ( + {!user?.home_facility_object && !userFacilities?.length && (
{ /* added const {t} hook here and relevant text to Common.json to avoid eslint error */ const { t } = useTranslation(); - const [skills, setSkills] = useState([]); const [selectedSkill, setSelectedSkill] = useState( null ); const [isLoading, setIsLoading] = useState(false); const [deleteSkill, setDeleteSkill] = useState(null); - const dispatch: any = useDispatch(); - const fetchSkills = useCallback( - async (username: string) => { - setIsLoading(true); - const res = await dispatch(getUserListSkills({ username })); - if (res && res.data) { - setSkills(res.data.results); - } - setIsLoading(false); - }, - [dispatch] - ); + const { + data: skills, + loading: skillsLoading, + refetch: refetchUserSkills, + } = useQuery(routes.userListSkill, { + pathParams: { username }, + }); const addSkill = useCallback( async (username: string, skill: SkillObjectModel | null) => { if (!skill) return; setIsLoading(true); - const res = await dispatch(addUserSkill(username, skill.id)); - if (res?.status !== 201) { + const { res } = await request(routes.addUserSkill, { + pathParams: { username }, + body: { skill: skill.id }, + }); + if (!res?.ok) { Notification.Error({ msg: "Error while adding skill", }); @@ -62,36 +56,32 @@ export default ({ show, setShow, username }: IProps) => { } setSelectedSkill(null); setIsLoading(false); - fetchSkills(username); + await refetchUserSkills(); }, - [dispatch, fetchSkills] + [refetchUserSkills] ); const removeSkill = useCallback( async (username: string, skillId: string) => { - const res = await dispatch(deleteUserSkill(username, skillId)); + const { res } = await request(routes.deleteUserSkill, { + pathParams: { username, id: skillId }, + }); if (res?.status !== 204) { Notification.Error({ msg: "Error while unlinking skill", }); } setDeleteSkill(null); - fetchSkills(username); + await refetchUserSkills(); }, - [dispatch, fetchSkills] + [refetchUserSkills] ); - useEffect(() => { - setIsLoading(true); - if (username) fetchSkills(username); - setIsLoading(false); - }, [username, fetchSkills]); - const authorizeForAddSkill = useIsAuthorized( AuthorizeFor(["DistrictAdmin", "StateAdmin"]) ); - const hasSkills = useMemo(() => skills.length > 0, [skills]); + const hasSkills = skills?.results?.length || 0 > 0; return (
@@ -114,7 +104,7 @@ export default ({ show, setShow, username }: IProps) => { >
- {!isLoading && ( + {(!isLoading || !skillsLoading) && (
{ setSelected={setSelectedSkill} errors="" username={username} + userSkills={skills?.results || []} /> { )}
)} - {isLoading ? ( + {isLoading || skillsLoading ? (
@@ -151,8 +142,8 @@ export default ({ show, setShow, username }: IProps) => {
{hasSkills ? ( diff --git a/src/Components/Users/UserAdd.tsx b/src/Components/Users/UserAdd.tsx index a6553bad01b..7df0089cdac 100644 --- a/src/Components/Users/UserAdd.tsx +++ b/src/Components/Users/UserAdd.tsx @@ -1,26 +1,17 @@ import { Link, navigate } from "raviger"; -import { lazy, useCallback, useEffect, useState } from "react"; -import { useDispatch } from "react-redux"; +import { lazy, useEffect, useState } from "react"; import { GENDER_TYPES, USER_TYPES, USER_TYPE_OPTIONS, } from "../../Common/constants"; -import { statusType, useAbortableEffect } from "../../Common/utils"; +import { useAbortableEffect } from "../../Common/utils"; import { validateEmailAddress, validateName, validatePassword, validateUsername, } from "../../Common/validation"; -import { - addUser, - getDistrictByState, - getLocalbodyByDistrict, - getStates, - getUserListFacility, - checkUsername, -} from "../../Redux/actions"; import * as Notification from "../../Utils/Notifications.js"; import { FacilitySelect } from "../Common/FacilitySelect"; import { FacilityModel } from "../Facility/models"; @@ -45,6 +36,9 @@ import { DraftSection, useAutoSaveReducer } from "../../Utils/AutoSave"; import dayjs from "../../Utils/dayjs"; import useAuthUser from "../../Common/hooks/useAuthUser"; import { PhoneNumberValidator } from "../Form/FieldValidators"; +import routes from "../../Redux/api"; +import request from "../../Utils/request/request"; +import useQuery from "../../Utils/request/useQuery"; const Loading = lazy(() => import("../Common/Loading")); @@ -163,7 +157,6 @@ export const validateRule = ( export const UserAdd = (props: UserProps) => { const { goBack } = useAppHistory(); - const dispatchAction: any = useDispatch(); const { userId } = props; const [state, dispatch] = useAutoSaveReducer( @@ -171,13 +164,9 @@ export const UserAdd = (props: UserProps) => { initialState ); const [isLoading, setIsLoading] = useState(false); - const [isStateLoading, setIsStateLoading] = useState(false); - const [isDistrictLoading, setIsDistrictLoading] = useState(false); - const [isLocalbodyLoading, setIsLocalbodyLoading] = useState(false); - const [_current_user_facilities, setFacilities] = useState< - Array - >([]); const [states, setStates] = useState([]); + const [selectedStateId, setSelectedStateId] = useState(0); + const [selectedDistrictId, setSelectedDistrictId] = useState(0); const [districts, setDistricts] = useState([]); const [localBodies, setLocalBodies] = useState([]); const [selectedFacility, setSelectedFacility] = useState([]); @@ -198,9 +187,9 @@ export const UserAdd = (props: UserProps) => { const check_username = async (username: string) => { setUsernameExists(userExistsEnums.checking); - const usernameCheck = await dispatchAction( - checkUsername({ username: username }) - ); + const { res: usernameCheck } = await request(routes.checkUsername, { + pathParams: { username }, + }); if (usernameCheck === undefined || usernameCheck.status === 409) setUsernameExists(userExistsEnums.exists); else if (usernameCheck.status === 200) @@ -254,101 +243,45 @@ export const UserAdd = (props: UserProps) => { state.form.user_type === "StaffReadOnly" ); - const fetchDistricts = useCallback( - async (id: number) => { - if (id > 0) { - setIsDistrictLoading(true); - const districtList = await dispatchAction(getDistrictByState({ id })); - if (districtList) { - if (userIndex <= USER_TYPES.indexOf("DistrictAdmin")) { - setDistricts([ - { - id: authUser.district!, - name: authUser.district_object?.name as string, - }, - ]); - } else { - setDistricts(districtList.data); - } - } - setIsDistrictLoading(false); + const { loading: isDistrictLoading } = useQuery(routes.getDistrictByState, { + prefetch: !!(selectedStateId > 0), + pathParams: { id: selectedStateId.toString() }, + onResponse: (result) => { + if (!result || !result.res || !result.data) return; + if (userIndex <= USER_TYPES.indexOf("DistrictAdmin")) { + setDistricts([authUser.district_object!]); + } else { + setDistricts(result.data); } }, - [dispatchAction] - ); - - const fetchLocalBody = useCallback( - async (id: number) => { - if (id > 0) { - setIsLocalbodyLoading(true); - const localBodyList = await dispatchAction( - getLocalbodyByDistrict({ id }) - ); - setIsLocalbodyLoading(false); - if (localBodyList) { - if (userIndex <= USER_TYPES.indexOf("LocalBodyAdmin")) { - setLocalBodies([ - { - id: authUser.local_body!, - name: authUser.local_body_object?.name as string, - }, - ]); - } else { - setLocalBodies(localBodyList.data); - } - } - } - }, - [dispatchAction] - ); - - const fetchStates = useCallback( - async (status: statusType) => { - setIsStateLoading(true); - const statesRes = await dispatchAction(getStates()); - if (!status.aborted && statesRes.data.results) { - if (userIndex <= USER_TYPES.indexOf("StateAdmin")) { - setStates([ - { - id: authUser.state!, - name: authUser.state_object?.name as string, - }, - ]); + }); + + const { loading: isLocalbodyLoading } = useQuery( + routes.getAllLocalBodyByDistrict, + { + prefetch: !!(selectedDistrictId > 0), + pathParams: { id: selectedDistrictId.toString() }, + onResponse: (result) => { + if (!result || !result.res || !result.data) return; + if (userIndex <= USER_TYPES.indexOf("LocalBodyAdmin")) { + setLocalBodies([authUser.local_body_object!]); } else { - setStates(statesRes.data.results); + setLocalBodies(result.data); } - } - setIsStateLoading(false); - }, - [dispatchAction] - ); - - const fetchFacilities = useCallback( - async (status: any) => { - setIsStateLoading(true); - const res = await dispatchAction( - getUserListFacility({ username: authUser.username }) - ); - if (!status.aborted && res && res.data) { - setFacilities(res.data); - } - setIsStateLoading(false); - }, - [dispatchAction, authUser.username] + }, + } ); - useAbortableEffect( - (status: statusType) => { - fetchStates(status); - if ( - authUser.user_type === "Staff" || - authUser.user_type === "StaffReadOnly" - ) { - fetchFacilities(status); + const { loading: isStateLoading } = useQuery(routes.statesList, { + onResponse: (result) => { + if (!result || !result.res || !result.data) return; + if (userIndex <= USER_TYPES.indexOf("StateAdmin")) { + setStates([authUser.state_object!]); + } else { + setStates(result.data.results); } }, - [dispatch] - ); + }); const handleDateChange = (e: FieldChangeEvent) => { if (dayjs(e.value).isValid()) { @@ -605,13 +538,10 @@ export const UserAdd = (props: UserProps) => { : undefined, }; - const res = await dispatchAction(addUser(data)); - if ( - res && - (res.data || res.data === "") && - res.status >= 200 && - res.status < 300 - ) { + const { res } = await request(routes.addUser, { + body: data, + }); + if (res?.ok) { dispatch({ type: "set_form", form: initForm }); if (!userId) { Notification.Success({ @@ -916,7 +846,7 @@ export const UserAdd = (props: UserProps) => { optionValue={(o) => o.id} onChange={(e) => { handleFieldChange(e); - if (e) fetchDistricts(e.value); + if (e) setSelectedStateId(e.value); }} /> )} @@ -934,7 +864,7 @@ export const UserAdd = (props: UserProps) => { optionValue={(o) => o.id} onChange={(e) => { handleFieldChange(e); - if (e) fetchLocalBody(e.value); + if (e) setSelectedDistrictId(e.value); }} /> )} diff --git a/src/Components/Users/UserFilter.tsx b/src/Components/Users/UserFilter.tsx index 3dca52d2463..4544fb8893a 100644 --- a/src/Components/Users/UserFilter.tsx +++ b/src/Components/Users/UserFilter.tsx @@ -1,6 +1,3 @@ -import { useEffect } from "react"; -import { useDispatch } from "react-redux"; -import { getDistrict } from "../../Redux/actions"; import { navigate } from "raviger"; import DistrictSelect from "../Facility/FacilityFilter/DistrictSelect"; import { parsePhoneNumber } from "../../Utils/utils"; @@ -11,6 +8,8 @@ import { USER_TYPE_OPTIONS } from "../../Common/constants"; import useMergeState from "../../Common/hooks/useMergeState"; import PhoneNumberFormField from "../Form/FormFields/PhoneNumberFormField"; import FiltersSlideover from "../../CAREUI/interactive/FiltersSlideover"; +import useQuery from "../../Utils/request/useQuery"; +import routes from "../../Redux/api"; const parsePhoneNumberForFilterParam = (phoneNumber: string) => { if (!phoneNumber) return ""; @@ -21,7 +20,6 @@ const parsePhoneNumberForFilterParam = (phoneNumber: string) => { export default function UserFilter(props: any) { const { filter, onChange, closeFilter } = props; - const dispatch: any = useDispatch(); const [filterState, setFilterState] = useMergeState({ first_name: filter.first_name || "", last_name: filter.last_name || "", @@ -69,17 +67,14 @@ export default function UserFilter(props: any) { onChange(data); }; - useEffect(() => { - async function fetchData() { - if (filter.district_id) { - const { data: districtData } = await dispatch( - getDistrict(filter.district_id, "district") - ); - setFilterState({ district_ref: districtData }); - } - } - fetchData(); - }, [dispatch]); + useQuery(routes.getDistrict, { + prefetch: !!filter.district_id, + pathParams: { id: filter.district_id }, + onResponse: (result) => { + if (!result || !result.data || !result.res) return; + setFilterState({ district_ref: result.data }); + }, + }); const handleChange = ({ name, value }: any) => setFilterState({ ...filterState, [name]: value }); diff --git a/src/Components/Users/UserProfile.tsx b/src/Components/Users/UserProfile.tsx index 441a6862634..76a94745c1a 100644 --- a/src/Components/Users/UserProfile.tsx +++ b/src/Components/Users/UserProfile.tsx @@ -1,13 +1,5 @@ -import { useState, useCallback, useReducer, lazy, FormEvent } from "react"; -import { statusType, useAbortableEffect } from "../../Common/utils"; +import { useState, useReducer, lazy, FormEvent } from "react"; import { GENDER_TYPES } from "../../Common/constants"; -import { useDispatch } from "react-redux"; -import { - getUserDetails, - getUserListSkills, - partialUpdateUser, - updateUserPassword, -} from "../../Redux/actions"; import { validateEmailAddress } from "../../Common/validation"; import * as Notification from "../../Utils/Notifications.js"; import LanguageSelector from "../../Components/Common/LanguageSelector"; @@ -18,15 +10,32 @@ import CareIcon from "../../CAREUI/icons/CareIcon"; import PhoneNumberFormField from "../Form/FormFields/PhoneNumberFormField"; import { FieldChangeEvent } from "../Form/FormFields/Utils"; import { SelectFormField } from "../Form/FormFields/SelectFormField"; -import { SkillModel, SkillObjectModel } from "../Users/models"; +import { GenderType, SkillModel, UpdatePasswordForm } from "../Users/models"; import UpdatableApp, { checkForUpdate } from "../Common/UpdatableApp"; import dayjs from "../../Utils/dayjs"; import useAuthUser from "../../Common/hooks/useAuthUser"; import { PhoneNumberValidator } from "../Form/FieldValidators"; +import useQuery from "../../Utils/request/useQuery"; +import routes from "../../Redux/api"; +import request from "../../Utils/request/request"; const Loading = lazy(() => import("../Common/Loading")); type EditForm = { + firstName: string; + lastName: string; + age: string; + gender: GenderType; + email: string; + phoneNumber: string; + altPhoneNumber: string; + user_type: string | undefined; + doctor_qualification: string | undefined; + doctor_experience_commenced_on: number | string | undefined; + doctor_medical_council_registration: string | undefined; + weekly_working_hours: string | undefined; +}; +type ErrorForm = { firstName: string; lastName: string; age: string; @@ -34,6 +43,7 @@ type EditForm = { email: string; phoneNumber: string; altPhoneNumber: string; + user_type: string | undefined; doctor_qualification: string | undefined; doctor_experience_commenced_on: number | string | undefined; doctor_medical_council_registration: string | undefined; @@ -41,27 +51,28 @@ type EditForm = { }; type State = { form: EditForm; - errors: EditForm; + errors: ErrorForm; }; type Action = | { type: "set_form"; form: EditForm } - | { type: "set_error"; errors: EditForm }; + | { type: "set_error"; errors: ErrorForm }; const initForm: EditForm = { firstName: "", lastName: "", age: "", - gender: "", + gender: "Male", email: "", phoneNumber: "", altPhoneNumber: "", + user_type: "", doctor_qualification: undefined, doctor_experience_commenced_on: undefined, doctor_medical_council_registration: undefined, weekly_working_hours: undefined, }; -const initError: EditForm = Object.assign( +const initError: ErrorForm = Object.assign( {}, ...Object.keys(initForm).map((k) => ({ [k]: "" })) ); @@ -87,9 +98,9 @@ const editFormReducer = (state: State, action: Action) => { } } }; + export default function UserProfile() { const [states, dispatch] = useReducer(editFormReducer, initialState); - const reduxDispatch: any = useDispatch(); const [updateStatus, setUpdateStatus] = useState({ isChecking: false, isUpdateAvailable: false, @@ -119,57 +130,44 @@ export default function UserProfile() { const [showEdit, setShowEdit] = useState(false); - const [isLoading, setIsLoading] = useState(false); - const dispatchAction: any = useDispatch(); - - const initialDetails: any = [{}]; - const [details, setDetails] = useState(initialDetails); - - const fetchData = useCallback( - async (status: statusType) => { - setIsLoading(true); - const res = await dispatchAction(getUserDetails(authUser.username)); - const resSkills = await dispatchAction( - getUserListSkills({ username: authUser.username }) - ); - if (!status.aborted) { - if (res && res.data && resSkills) { - res.data.skills = resSkills.data.results.map( - (skill: SkillModel) => skill.skill_object - ); - setDetails(res.data); - const formData: EditForm = { - firstName: res.data.first_name, - lastName: res.data.last_name, - age: res.data.age, - gender: res.data.gender, - email: res.data.email, - phoneNumber: res.data.phone_number, - altPhoneNumber: res.data.alt_phone_number, - doctor_qualification: res.data.doctor_qualification, - doctor_experience_commenced_on: dayjs().diff( - dayjs(res.data.doctor_experience_commenced_on), - "years" - ), - doctor_medical_council_registration: - res.data.doctor_medical_council_registration, - weekly_working_hours: res.data.weekly_working_hours, - }; - dispatch({ - type: "set_form", - form: formData, - }); - } - setIsLoading(false); - } - }, - [dispatchAction, authUser.username] - ); - useAbortableEffect( - (status: statusType) => { - fetchData(status); + const { + data: userData, + loading: isUserLoading, + refetch: refetchUserData, + } = useQuery(routes.getUserDetails, { + pathParams: { username: authUser.username }, + onResponse: (result) => { + if (!result || !result.res || !result.data) return; + const formData: EditForm = { + firstName: result.data.first_name, + lastName: result.data.last_name, + age: result.data.age?.toString() || "", + gender: result.data.gender || "Male", + email: result.data.email, + phoneNumber: result.data.phone_number?.toString() || "", + altPhoneNumber: result.data.alt_phone_number?.toString() || "", + user_type: result.data.user_type, + doctor_qualification: result.data.doctor_qualification, + doctor_experience_commenced_on: dayjs().diff( + dayjs(result.data.doctor_experience_commenced_on), + "years" + ), + doctor_medical_council_registration: + result.data.doctor_medical_council_registration, + weekly_working_hours: result.data.weekly_working_hours, + }; + dispatch({ + type: "set_form", + form: formData, + }); }, - [fetchData] + }); + + const { data: skillsView, loading: isSkillsLoading } = useQuery( + routes.userListSkill, + { + pathParams: { username: authUser.username }, + } ); const validateForm = () => { @@ -244,7 +242,7 @@ export default function UserProfile() { case "doctor_qualification": case "doctor_experience_commenced_on": case "doctor_medical_council_registration": - if (details.user_type === "Doctor" && !states.form[field]) { + if (states.form.user_type === "Doctor" && !states.form[field]) { errors[field] = "Field is required"; invalidForm = true; } @@ -298,13 +296,13 @@ export default function UserProfile() { phone_number: parsePhoneNumber(states.form.phoneNumber) ?? "", alt_phone_number: parsePhoneNumber(states.form.altPhoneNumber) ?? "", gender: states.form.gender, - age: states.form.age, + age: +states.form.age, doctor_qualification: - details.user_type === "Doctor" + states.form.user_type === "Doctor" ? states.form.doctor_qualification : undefined, doctor_experience_commenced_on: - details.user_type === "Doctor" + states.form.user_type === "Doctor" ? dayjs() .subtract( parseInt( @@ -316,34 +314,27 @@ export default function UserProfile() { .format("YYYY-MM-DD") : undefined, doctor_medical_council_registration: - details.user_type === "Doctor" + states.form.user_type === "Doctor" ? states.form.doctor_medical_council_registration : undefined, weekly_working_hours: states.form.weekly_working_hours, }; - const res = await dispatchAction( - partialUpdateUser(authUser.username, data) - ); - if (res && res.data) { + const { res } = await request(routes.partialUpdateUser, { + pathParams: { username: authUser.username }, + body: data, + }); + if (res?.ok) { Notification.Success({ msg: "Details updated successfully", }); - window.location.reload(); - setDetails({ - ...details, - first_name: states.form.firstName, - last_name: states.form.lastName, - age: states.form.age, - gender: states.form.gender, - email: states.form.email, - phone_number: states.form.phoneNumber, - alt_phone_number: states.form.altPhoneNumber, - }); + await refetchUserData(); setShowEdit(false); } } }; + const isLoading = isUserLoading || isSkillsLoading; + if (isLoading) { return ; } @@ -367,7 +358,7 @@ export default function UserProfile() { } }; - const changePassword = (e: any) => { + const changePassword = async (e: any) => { e.preventDefault(); //validating form if ( @@ -377,30 +368,28 @@ export default function UserProfile() { msg: "Passwords are different in the new and the confirmation column.", }); } else { - setIsLoading(true); - const form = { + const form: UpdatePasswordForm = { old_password: changePasswordForm.old_password, username: authUser.username, new_password: changePasswordForm.new_password_1, }; - reduxDispatch(updateUserPassword(form)).then((resp: any) => { - setIsLoading(false); - const res = resp && resp.data; - if (res.message === "Password updated successfully") { - Notification.Success({ - msg: "Password changed!", - }); - } else { - Notification.Error({ - msg: "There was some error. Please try again in some time.", - }); - } - setChangePasswordForm({ - ...changePasswordForm, - new_password_1: "", - new_password_2: "", - old_password: "", + const { res, data } = await request(routes.updatePassword, { + body: form, + }); + if (res?.ok && data?.message === "Password updated successfully") { + Notification.Success({ + msg: "Password changed!", + }); + } else { + Notification.Error({ + msg: "There was some error. Please try again in some time.", }); + } + setChangePasswordForm({ + ...changePasswordForm, + new_password_1: "", + new_password_2: "", + old_password: "", }); } }; @@ -432,7 +421,7 @@ export default function UserProfile() {
- {!showEdit && ( + {!showEdit && !isLoading && (
- {details.username || "-"} + {userData?.username || "-"}
- {details.phone_number || "-"} + {userData?.phone_number || "-"}
@@ -466,7 +455,7 @@ export default function UserProfile() { Whatsapp No
- {details.alt_phone_number || "-"} + {userData?.alt_phone_number || "-"}
- {details.email || "-"} + {userData?.email || "-"}
- {details.first_name || "-"} + {userData?.first_name || "-"}
- {details.last_name || "-"} + {userData?.last_name || "-"}
@@ -507,7 +496,7 @@ export default function UserProfile() { Age
- {details.age || "-"} + {userData?.age || "-"}
@@ -516,7 +505,7 @@ export default function UserProfile() {
{" "} - {details.user_type || "-"} + {userData?.user_type || "-"}
- {details.gender || "-"} + {userData?.gender || "-"}
@@ -535,7 +524,7 @@ export default function UserProfile() { Local Body
- {details.local_body_object?.name || "-"} + {userData?.local_body_object?.name || "-"}
@@ -543,7 +532,7 @@ export default function UserProfile() { District
- {details.district_object?.name || "-"} + {userData?.district_object?.name || "-"}
@@ -551,7 +540,7 @@ export default function UserProfile() { State
- {details.state_object?.name || "-"} + {userData?.state_object?.name || "-"}
@@ -563,11 +552,13 @@ export default function UserProfile() { className="flex flex-wrap gap-2" id="already-linked-skills" > - {details.skills && details.skills.length - ? details.skills?.map((skill: SkillObjectModel) => { + {skillsView?.results?.length + ? skillsView.results?.map((skill: SkillModel) => { return ( -

{skill.name}

+

+ {skill.skill_object.name} +

); }) @@ -583,7 +574,7 @@ export default function UserProfile() { Average weekly working hours
- {details.weekly_working_hours ?? "-"} + {userData?.weekly_working_hours || "-"}
@@ -649,7 +640,7 @@ export default function UserProfile() { required type="email" /> - {details.user_type === "Doctor" && ( + {states.form.user_type === "Doctor" && ( <> { export const signupUser = (params: object) => { return fireRequest("createUser", [], params); }; -export const addUser = (params: object) => { - return fireRequest("addUser", [], params); -}; export const deleteUser = (username: string) => { - return fireRequest("deleteUser", [username], {}); + return fireRequest("deleteUser", [], {}, { username }); }; export const checkResetToken = (params: object) => { @@ -34,10 +31,6 @@ export const postForgotPassword = (form: object) => { return fireRequest("forgotPassword", [], form); }; -export const updateUserPassword = (form: object) => { - return fireRequest("updatePassword", [], form); -}; - export const getUserPnconfig = (pathParams: object) => { return fireRequest("getUserPnconfig", [], {}, pathParams); }; @@ -62,14 +55,6 @@ export const deleteFacilityCoverImage = (id: string) => { export const getUserList = (params: object, key?: string) => { return fireRequest("userList", [], params, null, key); }; - -export const getUserListSkills = (pathParam: object) => { - return fireRequest("userListSkill", [], {}, pathParam); -}; - -export const partialUpdateUser = (username: string, data: any) => { - return fireRequest("partialUpdateUser", [], data, { username }); -}; export const getUserListFacility = (pathParam: object) => { return fireRequest("userListFacility", [], {}, pathParam); }; @@ -95,10 +80,6 @@ export const deleteUserFacility = (username: string, facility: string) => { ); }; -export const clearHomeFacility = (username: string) => { - return fireRequest("clearHomeFacility", [], {}, { username }); -}; - export const getPermittedFacilities = (params: object) => { return fireRequest("getPermittedFacilities", [], params); }; @@ -605,20 +586,6 @@ export const dischargePatient = (params: object, pathParams: object) => { //Profile -export const checkUsername = (params: object) => { - return fireRequest("checkUsername", [], {}, params, undefined, true); -}; - -export const getUserDetails = (username: string, suppress?: boolean) => { - return fireRequest( - "getUserDetails", - [], - {}, - { username: username }, - undefined, - suppress ?? true - ); -}; export const updateUserDetails = (username: string, data: object) => { return fireRequest("updateUserDetails", [username], data); }; diff --git a/src/Redux/api.tsx b/src/Redux/api.tsx index 4172effe432..2a64d921792 100644 --- a/src/Redux/api.tsx +++ b/src/Redux/api.tsx @@ -33,6 +33,7 @@ import { ConsultationModel, CreateBedBody, CurrentBed, + DistrictModel, DailyRoundsBody, DailyRoundsRes, DoctorModal, @@ -40,9 +41,10 @@ import { IFacilityNotificationRequest, IFacilityNotificationResponse, IUserFacilityRequest, - LocationModel, PatientStatsModel, WardModel, + LocationModel, + StateModel, } from "../Components/Facility/models"; import { IDeleteExternalResult, @@ -52,9 +54,13 @@ import { ILocalBodyByDistrict, IPartialUpdateExternalResult, } from "../Components/ExternalResult/models"; - +import { + SkillModel, + SkillObjectModel, + UpdatePasswordForm, + UserModel, +} from "../Components/Users/models"; import { Prescription } from "../Components/Medicine/models"; -import { UserModel } from "../Components/Users/models"; import { DailyRoundsModel, PatientModel } from "../Components/Patient/models"; import { PaginatedResponse } from "../Utils/request/types"; import { @@ -149,6 +155,8 @@ const routes = { updatePassword: { path: "/api/v1/password_change/", method: "PUT", + TRes: Type>(), + TBody: Type(), }, // User Endpoints currentUser: { @@ -164,11 +172,14 @@ const routes = { userListSkill: { path: "/api/v1/users/{username}/skill/", + method: "GET", + TRes: Type>(), }, userListFacility: { path: "/api/v1/users/{username}/get_facilities/", - TRes: Type(), + method: "GET", + TRes: Type(), }, addUserFacility: { @@ -181,6 +192,8 @@ const routes = { addUserSkill: { path: "/api/v1/users/{username}/skill/", method: "POST", + TBody: Type<{ skill: string }>(), + TRes: Type(), }, deleteUserFacility: { @@ -193,11 +206,13 @@ const routes = { clearHomeFacility: { path: "/api/v1/users/{username}/clear_home_facility/", method: "DELETE", + TRes: Type>(), }, deleteUserSkill: { path: "/api/v1/users/{username}/skill/{id}/", method: "DELETE", + TRes: Type>(), }, createUser: { @@ -214,6 +229,8 @@ const routes = { partialUpdateUser: { path: "/api/v1/users/{username}/", method: "PATCH", + TRes: Type(), + TBody: Type>(), }, deleteUser: { @@ -225,6 +242,7 @@ const routes = { addUser: { path: "/api/v1/users/add_user/", method: "POST", + TRes: Type(), }, searchUser: { @@ -251,6 +269,8 @@ const routes = { getAllSkills: { path: "/api/v1/skill/", + method: "GET", + TRes: Type>(), }, // Facility Endpoints @@ -651,6 +671,8 @@ const routes = { // States statesList: { path: "/api/v1/state/", + method: "GET", + TRes: Type>(), }, getState: { @@ -661,9 +683,13 @@ const routes = { getDistrict: { path: "/api/v1/district/{id}/", + method: "GET", + TRes: Type(), }, getDistrictByState: { path: "/api/v1/state/{id}/districts/", + method: "GET", + TRes: Type(), }, getDistrictByName: { path: "/api/v1/district/", @@ -775,11 +801,13 @@ const routes = { checkUsername: { path: "/api/v1/users/{username}/check_availability/", method: "GET", + TRes: Type>(), }, getUserDetails: { path: "/api/v1/users/{username}/", method: "GET", + TRes: Type(), }, updateUserDetails: { path: "/api/v1/users", From 745d4f87c3518750477b7a59effe45cc82fb1163 Mon Sep 17 00:00:00 2001 From: Rithvik Nishad Date: Thu, 7 Dec 2023 10:15:36 +0530 Subject: [PATCH 21/21] Enhancements to Location Live Feed (#6726) * Page: support for auto collapse sidebar * Adds utilities and `useOperateCamera` hook * Adds reusable component: `NetworkSignal` * stash * Update camera controls * rename files * refactor * refactor location filter * Add location select popup * Update location select logic * update reponsiveness * hide non working filters * adjust z-index --- src/CAREUI/display/NetworkSignal.tsx | 57 + src/CAREUI/interactive/KeyboardShortcut.tsx | 49 + src/CAREUI/misc/Fullscreen.tsx | 43 + src/Common/hooks/useFeedPTZ.ts | 7 + src/Common/hooks/useMSEplayer.ts | 45 - src/Components/Assets/AssetTypes.tsx | 1 + src/Components/CameraFeed/AssetBedSelect.tsx | 77 ++ src/Components/CameraFeed/CameraFeed.tsx | 199 ++++ .../CameraFeed/CameraFeedWithBedPresets.tsx | 31 + .../LiveMonitoringFilters.tsx | 141 +++ .../CentralLiveMonitoring/index.tsx | 84 ++ src/Components/CameraFeed/FeedAlert.tsx | 72 ++ src/Components/CameraFeed/FeedButton.tsx | 42 + src/Components/CameraFeed/FeedControls.tsx | 217 ++++ .../CameraFeed/FeedNetworkSignal.tsx | 58 + src/Components/CameraFeed/NoFeedAvailable.tsx | 58 + src/Components/CameraFeed/routes.ts | 11 + src/Components/CameraFeed/useOperateCamera.ts | 54 + src/Components/CameraFeed/usePlayer.tsx | 58 + src/Components/CameraFeed/utils.ts | 28 + src/Components/Common/components/Page.tsx | 23 +- .../Facility/CentralNursingStation.tsx | 18 +- .../Facility/Consultations/Feed.tsx | 2 +- .../Facility/Consultations/LiveFeed.tsx | 2 +- src/Components/Facility/FacilityHome.tsx | 76 +- src/Components/Facility/LiveFeedScreen.tsx | 230 ---- src/Components/Facility/LiveFeedTile.tsx | 1045 ----------------- .../VitalsMonitor/WaveformLabels.tsx | 7 +- src/Redux/api.tsx | 2 +- src/Routers/routes/FacilityLocationRoutes.tsx | 4 + src/Routers/routes/FacilityRoutes.tsx | 4 - src/Utils/utils.ts | 7 +- src/style/CAREUI.css | 2 +- src/style/index.css | 6 - 34 files changed, 1394 insertions(+), 1366 deletions(-) create mode 100644 src/CAREUI/display/NetworkSignal.tsx create mode 100644 src/CAREUI/interactive/KeyboardShortcut.tsx create mode 100644 src/CAREUI/misc/Fullscreen.tsx create mode 100644 src/Components/CameraFeed/AssetBedSelect.tsx create mode 100644 src/Components/CameraFeed/CameraFeed.tsx create mode 100644 src/Components/CameraFeed/CameraFeedWithBedPresets.tsx create mode 100644 src/Components/CameraFeed/CentralLiveMonitoring/LiveMonitoringFilters.tsx create mode 100644 src/Components/CameraFeed/CentralLiveMonitoring/index.tsx create mode 100644 src/Components/CameraFeed/FeedAlert.tsx create mode 100644 src/Components/CameraFeed/FeedButton.tsx create mode 100644 src/Components/CameraFeed/FeedControls.tsx create mode 100644 src/Components/CameraFeed/FeedNetworkSignal.tsx create mode 100644 src/Components/CameraFeed/NoFeedAvailable.tsx create mode 100644 src/Components/CameraFeed/routes.ts create mode 100644 src/Components/CameraFeed/useOperateCamera.ts create mode 100644 src/Components/CameraFeed/usePlayer.tsx create mode 100644 src/Components/CameraFeed/utils.ts delete mode 100644 src/Components/Facility/LiveFeedScreen.tsx delete mode 100644 src/Components/Facility/LiveFeedTile.tsx diff --git a/src/CAREUI/display/NetworkSignal.tsx b/src/CAREUI/display/NetworkSignal.tsx new file mode 100644 index 00000000000..b0ae2c541cb --- /dev/null +++ b/src/CAREUI/display/NetworkSignal.tsx @@ -0,0 +1,57 @@ +import { classNames } from "../../Utils/utils"; +import CareIcon from "../icons/CareIcon"; + +interface Props { + /** + * Strength of the signal, from 0 to 3 + * + * undefined: Error + * 0: No signal + * 1: Weak signal + * 2: Medium signal + * 3: Strong signal + */ + strength?: number; + children?: React.ReactNode; +} + +export default function NetworkSignal({ strength, children }: Props) { + return ( +
+
+ {strength === undefined ? ( + + ) : ( + Array.from({ length: 3 }, (_, i) => ( +
i ? "bg-current" : "bg-zinc-600" + )} + /> + )) + )} +
+ {children} +
+ ); +} diff --git a/src/CAREUI/interactive/KeyboardShortcut.tsx b/src/CAREUI/interactive/KeyboardShortcut.tsx new file mode 100644 index 00000000000..47a9fbfca28 --- /dev/null +++ b/src/CAREUI/interactive/KeyboardShortcut.tsx @@ -0,0 +1,49 @@ +import useKeyboardShortcut from "use-keyboard-shortcut"; +import { classNames, isAppleDevice } from "../../Utils/utils"; + +interface Props { + children: React.ReactNode; + shortcut: string[]; + onTrigger: () => void; + shortcutSeperator?: string; + helpText?: string; + tooltipClassName?: string; +} + +export default function KeyboardShortcut(props: Props) { + useKeyboardShortcut(props.shortcut, props.onTrigger, { + overrideSystem: true, + }); + + return ( +
+ {props.children} + + {props.helpText} + + {getShortcutKeyDescription(props.shortcut).join(" + ")} + + +
+ ); +} + +const SHORTCUT_KEY_MAP = { + Meta: "⌘", + Shift: "⇧Shift", + Alt: "⌥Alt", + Control: isAppleDevice ? "⌃Ctrl" : "Ctrl", + ArrowUp: "↑", + ArrowDown: "↓", + ArrowLeft: "←", + ArrowRight: "→", +} as Record; + +export const getShortcutKeyDescription = (shortcut: string[]) => { + return shortcut.map((key) => SHORTCUT_KEY_MAP[key] || key); +}; diff --git a/src/CAREUI/misc/Fullscreen.tsx b/src/CAREUI/misc/Fullscreen.tsx new file mode 100644 index 00000000000..5cfa7865128 --- /dev/null +++ b/src/CAREUI/misc/Fullscreen.tsx @@ -0,0 +1,43 @@ +import { useEffect, useRef } from "react"; + +interface Props { + className?: string; + fullscreenClassName?: string; + children: React.ReactNode; + fullscreen: boolean; + onExit: () => void; +} + +export default function Fullscreen(props: Props) { + const ref = useRef(null); + + useEffect(() => { + if (props.fullscreen) { + ref.current?.requestFullscreen(); + } else { + document.exitFullscreen(); + } + }, [props.fullscreen]); + + useEffect(() => { + const listener = () => { + if (!document.fullscreenElement) { + props.onExit(); + } + }; + + document.addEventListener("fullscreenchange", listener); + return () => { + document.removeEventListener("fullscreenchange", listener); + }; + }, [props.onExit]); + + return ( +
+ {props.children} +
+ ); +} diff --git a/src/Common/hooks/useFeedPTZ.ts b/src/Common/hooks/useFeedPTZ.ts index 32cc5e6bc25..b064e9180ee 100644 --- a/src/Common/hooks/useFeedPTZ.ts +++ b/src/Common/hooks/useFeedPTZ.ts @@ -1,3 +1,10 @@ +/** + * Deprecated. Use `useOperateAsset` instead. + * + * Preserving for backwards compatibility and preventing merge conflict with a + * co-related PR. Will be removed in the future. + */ + import { operateAsset } from "../../Redux/actions"; export interface IAsset { diff --git a/src/Common/hooks/useMSEplayer.ts b/src/Common/hooks/useMSEplayer.ts index 4d1bb36b9ac..898da28f3ad 100644 --- a/src/Common/hooks/useMSEplayer.ts +++ b/src/Common/hooks/useMSEplayer.ts @@ -5,12 +5,6 @@ export interface IAsset { middlewareHostname: string; } -interface PTZPayload { - x: number; - y: number; - zoom: number; -} - interface UseMSEMediaPlayerOption { config: IAsset; url?: string; @@ -40,16 +34,6 @@ export interface IOptions { onSuccess?: (resp: any) => void; onError?: (err: any) => void; } - -enum PTZ { - Up = "up", - Down = "down", - Left = "left", - Right = "right", - ZoomIn = "zoomIn", - ZoomOut = "zoomOut", -} - const stopStream = ({ middlewareHostname, @@ -69,38 +53,9 @@ const stopStream = .catch((err) => options.onError && options.onError(err)); }; -export const getPTZPayload = (action: PTZ): PTZPayload => { - let x = 0; - let y = 0; - let zoom = 0; - switch (action) { - case PTZ.Up: - y = 0.1; - break; - case PTZ.Down: - y = -0.1; - break; - case PTZ.Left: - x = -0.1; - break; - case PTZ.Right: - x = 0.1; - break; - case PTZ.ZoomIn: - zoom = 0.1; - break; - case PTZ.ZoomOut: - zoom = -0.1; - break; - } - - return { x, y, zoom }; -}; - /** * MSE player utility */ - const Utf8ArrayToStr = (array: string | any[] | Uint8Array) => { let out, i, c; let char2, char3; diff --git a/src/Components/Assets/AssetTypes.tsx b/src/Components/Assets/AssetTypes.tsx index a4005404da1..a894c87dcc5 100644 --- a/src/Components/Assets/AssetTypes.tsx +++ b/src/Components/Assets/AssetTypes.tsx @@ -13,6 +13,7 @@ export interface AssetLocationObject { id: string; name: string; }; + middleware_address?: string; } export enum AssetType { diff --git a/src/Components/CameraFeed/AssetBedSelect.tsx b/src/Components/CameraFeed/AssetBedSelect.tsx new file mode 100644 index 00000000000..17701dccbde --- /dev/null +++ b/src/Components/CameraFeed/AssetBedSelect.tsx @@ -0,0 +1,77 @@ +import { Fragment } from "react"; +import useSlug from "../../Common/hooks/useSlug"; +import routes from "../../Redux/api"; +import useQuery from "../../Utils/request/useQuery"; +import { AssetBedModel, AssetData } from "../Assets/AssetTypes"; +import { BedModel } from "../Facility/models"; +import { Listbox, Transition } from "@headlessui/react"; +import CareIcon from "../../CAREUI/icons/CareIcon"; + +interface Props { + asset?: AssetData; + bed?: BedModel; + value?: AssetBedModel; + onChange?: (value: AssetBedModel) => void; +} + +export default function AssetBedSelect(props: Props) { + const facility = useSlug("facility"); + + const { data, loading } = useQuery(routes.listAssetBeds, { + query: { + limit: 100, + facility, + asset: props.asset?.id, + bed: props.bed?.id, + }, + }); + + const selected = props.value; + + return ( + +
+ + + {selected?.bed_object.name ?? "No Preset"} + + + + + + + + {data?.results.map((obj) => ( + + `relative cursor-default select-none px-2 py-1 ${ + active ? "bg-zinc-700 text-white" : "text-zinc-400" + }` + } + value={obj} + > + {({ selected }) => ( + <> + + {obj.bed_object.name}: {obj.meta.preset_name} + + + )} + + ))} + + +
+
+ ); +} diff --git a/src/Components/CameraFeed/CameraFeed.tsx b/src/Components/CameraFeed/CameraFeed.tsx new file mode 100644 index 00000000000..4ec039e4e70 --- /dev/null +++ b/src/Components/CameraFeed/CameraFeed.tsx @@ -0,0 +1,199 @@ +import { LegacyRef, useCallback, useEffect, useRef, useState } from "react"; +import { AssetData } from "../Assets/AssetTypes"; +import useOperateCamera, { PTZPayload } from "./useOperateCamera"; +import usePlayer from "./usePlayer"; +import { getStreamUrl } from "./utils"; +import ReactPlayer from "react-player"; +import { classNames, isIOS } from "../../Utils/utils"; +import FeedAlert, { FeedAlertState } from "./FeedAlert"; +import FeedNetworkSignal from "./FeedNetworkSignal"; +import NoFeedAvailable from "./NoFeedAvailable"; +import FeedControls from "./FeedControls"; +import Fullscreen from "../../CAREUI/misc/Fullscreen"; + +interface Props { + children?: React.ReactNode; + asset: AssetData; + fallbackMiddleware: string; // TODO: remove this in favour of `asset.resolved_middleware.hostname` once https://github.com/coronasafe/care/pull/1741 is merged + preset?: PTZPayload; + silent?: boolean; + className?: string; + // Callbacks + onCameraPresetsObtained?: (presets: Record) => void; + onStreamSuccess?: () => void; + onStreamError?: () => void; + // Controls + constrolsDisabled?: boolean; + shortcutsDisabled?: boolean; +} + +export default function CameraFeed(props: Props) { + const playerRef = useRef(null); + const streamUrl = getStreamUrl(props.asset, props.fallbackMiddleware); + + const player = usePlayer(streamUrl, playerRef); + const operate = useOperateCamera(props.asset.id, props.silent); + + const [isFullscreen, setFullscreen] = useState(false); + const [state, setState] = useState(); + useEffect(() => setState(player.status), [player.status, setState]); + + // Move camera when selected preset has changed + useEffect(() => { + async function move(preset: PTZPayload) { + setState("moving"); + const { res } = await operate({ type: "absolute_move", data: preset }); + setTimeout(() => setState((s) => (s === "moving" ? undefined : s)), 4000); + if (res?.status === 500) { + setState("host_unreachable"); + } + } + + if (props.preset) { + move(props.preset); + } + }, [props.preset]); + + // Get camera presets (only if onCameraPresetsObtained is provided) + useEffect(() => { + if (!props.onCameraPresetsObtained) return; + async function getPresets(cb: (presets: Record) => void) { + const { res, data } = await operate({ type: "get_presets" }); + if (res?.ok && data) { + cb((data as { result: Record }).result); + } + } + getPresets(props.onCameraPresetsObtained); + }, [operate, props.onCameraPresetsObtained]); + + const initializeStream = useCallback(() => { + player.initializeStream({ + onSuccess: async () => { + props.onStreamSuccess?.(); + const { res } = await operate({ type: "get_status" }); + if (res?.status === 500) { + setState("host_unreachable"); + } + }, + onError: props.onStreamError, + }); + }, [player.initializeStream, props.onStreamSuccess, props.onStreamError]); + + // Start stream on mount + useEffect(() => initializeStream(), [initializeStream]); + + const resetStream = () => { + setState("loading"); + initializeStream(); + }; + + return ( + setFullscreen(false)}> +
+
+
+ + {props.asset.name} + +
+ +
+
+ {props.children} +
+ +
+ {/* Notifications */} + + + {/* No Feed informations */} + {state === "host_unreachable" && ( + + )} + {player.status === "offline" && ( + + )} + + {/* Video Player */} + {isIOS ? ( +
+ } + controls={false} + playsinline + playing + muted + width="100%" + height="100%" + onPlay={player.onPlayCB} + onEnded={() => player.setStatus("stop")} + onError={(e, _, hlsInstance) => { + if (e === "hlsError") { + const recovered = hlsInstance.recoverMediaError(); + console.info(recovered); + } + }} + /> +
+ ) : ( +
+
+
+ ); +} diff --git a/src/Components/CameraFeed/CameraFeedWithBedPresets.tsx b/src/Components/CameraFeed/CameraFeedWithBedPresets.tsx new file mode 100644 index 00000000000..b52071a8597 --- /dev/null +++ b/src/Components/CameraFeed/CameraFeedWithBedPresets.tsx @@ -0,0 +1,31 @@ +import { useState } from "react"; +import { AssetBedModel, AssetData } from "../Assets/AssetTypes"; +import CameraFeed from "./CameraFeed"; +import AssetBedSelect from "./AssetBedSelect"; + +interface Props { + asset: AssetData; + fallbackMiddleware?: string; +} + +export default function LocationFeedTile(props: Props) { + const [preset, setPreset] = useState(); + + return ( + +
+ +
+
+ ); +} diff --git a/src/Components/CameraFeed/CentralLiveMonitoring/LiveMonitoringFilters.tsx b/src/Components/CameraFeed/CentralLiveMonitoring/LiveMonitoringFilters.tsx new file mode 100644 index 00000000000..628518191de --- /dev/null +++ b/src/Components/CameraFeed/CentralLiveMonitoring/LiveMonitoringFilters.tsx @@ -0,0 +1,141 @@ +import { Popover, Transition } from "@headlessui/react"; +import ButtonV2 from "../../Common/components/ButtonV2"; +import { FieldLabel } from "../../Form/FormFields/FormField"; +import { LocationSelect } from "../../Common/LocationSelect"; +import Pagination from "../../Common/Pagination"; +import useFilters from "../../../Common/hooks/useFilters"; +import { Fragment } from "react"; +import CareIcon from "../../../CAREUI/icons/CareIcon"; +import useSlug from "../../../Common/hooks/useSlug"; +import { classNames } from "../../../Utils/utils"; + +interface Props { + perPageLimit: number; + isFullscreen: boolean; + setFullscreen: (state: boolean) => void; + totalCount: number; +} + +const LiveMonitoringFilters = (props: Props) => { + const facilityId = useSlug("facility"); + const { qParams, updateQuery, removeFilter, updatePage } = useFilters({ + limit: props.perPageLimit, + }); + + return ( +
+ + + + + Settings and Filters + + + + +
+
+
+ + {props.totalCount}{" "} + Camera(s) present + +
+
+
+
+ + Filter by Location + +
+ + location + ? updateQuery({ location }) + : removeFilter("location") + } + selected={qParams.location} + showAll={false} + multiple={false} + facilityId={facilityId} + errors="" + errorClassName="hidden" + /> +
+
+ {/* { + if (value) { + updateQuery({ [name]: value }); + } else { + removeFilter(name); + } + }} + labelClassName="text-sm" + errorClassName="hidden" + /> + { + if (value) { + updateQuery({ [name]: value }); + } else { + removeFilter(name); + } + }} + labelClassName="text-sm" + errorClassName="hidden" + /> */} + props.setFullscreen(!props.isFullscreen)} + className="tooltip !h-11" + > + + {props.isFullscreen ? "Exit Fullscreen" : "Fullscreen"} + +
+
+
+
+
+ + updatePage(page)} + /> +
+ ); +}; + +export default LiveMonitoringFilters; diff --git a/src/Components/CameraFeed/CentralLiveMonitoring/index.tsx b/src/Components/CameraFeed/CentralLiveMonitoring/index.tsx new file mode 100644 index 00000000000..d83b01201b9 --- /dev/null +++ b/src/Components/CameraFeed/CentralLiveMonitoring/index.tsx @@ -0,0 +1,84 @@ +import { useState } from "react"; +import Loading from "../../Common/Loading"; +import Page from "../../Common/components/Page"; +import useQuery from "../../../Utils/request/useQuery"; +import routes from "../../../Redux/api"; +import LocationFeedTile from "../CameraFeedWithBedPresets"; +import Fullscreen from "../../../CAREUI/misc/Fullscreen"; +import useBreakpoints from "../../../Common/hooks/useBreakpoints"; +import { useQueryParams } from "raviger"; +import LiveMonitoringFilters from "./LiveMonitoringFilters"; + +export default function CentralLiveMonitoring(props: { facilityId: string }) { + const [isFullscreen, setFullscreen] = useState(false); + const limit = useBreakpoints({ default: 4, "3xl": 9 }); + + const [qParams] = useQueryParams(); + + const facilityQuery = useQuery(routes.getPermittedFacility, { + pathParams: { id: props.facilityId }, + }); + + const { data, loading } = useQuery(routes.listAssets, { + query: { + ...qParams, + limit, + offset: (qParams.page ? qParams.page - 1 : 0) * limit, + asset_class: "ONVIF", + facility: props.facilityId, + location: qParams.location, + in_use_by_consultation: qParams.in_use_by_consultation, + }, + }); + + const totalCount = data?.count ?? 0; + + return ( + + } + > + {loading || + data === undefined || + facilityQuery.data === undefined || + facilityQuery.loading ? ( + + ) : data.results.length === 0 ? ( +
+ No Camera present in this location or facility. +
+ ) : ( + setFullscreen(false)} + > +
+ {data.results.map((asset) => ( +
+ +
+ ))} +
+
+ )} +
+ ); +} diff --git a/src/Components/CameraFeed/FeedAlert.tsx b/src/Components/CameraFeed/FeedAlert.tsx new file mode 100644 index 00000000000..0c2eb6aa429 --- /dev/null +++ b/src/Components/CameraFeed/FeedAlert.tsx @@ -0,0 +1,72 @@ +import { Transition } from "@headlessui/react"; +import { Fragment, useEffect, useState } from "react"; +import CareIcon, { IconName } from "../../CAREUI/icons/CareIcon"; +import { classNames } from "../../Utils/utils"; +import { StreamStatus } from "./usePlayer"; + +export type FeedAlertState = + | StreamStatus + | "moving" + | "zooming" + | "saving_preset" + | "host_unreachable"; + +interface Props { + state?: FeedAlertState; +} + +const ALERT_ICON_MAP: Record = { + playing: "l-play-circle", + stop: "l-stop-circle", + offline: "l-exclamation-triangle", + loading: "l-spinner", + moving: "l-expand-from-corner", + zooming: "l-search", + saving_preset: "l-save", + host_unreachable: "l-exclamation-triangle", +}; + +export default function FeedAlert({ state }: Props) { + const [show, setShow] = useState(false); + + useEffect(() => { + if (!state) return; + + setShow(true); + + if (state !== "loading") { + const timeout = setTimeout(() => setShow(false), 4000); + return () => { + clearTimeout(timeout); + }; + } + }, [state, setShow]); + + return ( + +
+ {state && ( + + )} + + {state?.replace("_", " ")} + +
+
+ ); +} diff --git a/src/Components/CameraFeed/FeedButton.tsx b/src/Components/CameraFeed/FeedButton.tsx new file mode 100644 index 00000000000..033ffcc11ed --- /dev/null +++ b/src/Components/CameraFeed/FeedButton.tsx @@ -0,0 +1,42 @@ +import KeyboardShortcut from "../../CAREUI/interactive/KeyboardShortcut"; +import { classNames } from "../../Utils/utils"; + +interface Props { + className?: string; + children?: React.ReactNode; + readonly shortcut?: string[]; + onTrigger: () => void; + helpText?: string; + shortcutsDisabled?: boolean; + tooltipClassName?: string; +} + +export default function FeedButton(props: Props) { + const child = ( + + ); + + if (props.shortcutsDisabled || !props.shortcut) { + return child; + } + + return ( + + {child} + + ); +} diff --git a/src/Components/CameraFeed/FeedControls.tsx b/src/Components/CameraFeed/FeedControls.tsx new file mode 100644 index 00000000000..3a5afb76209 --- /dev/null +++ b/src/Components/CameraFeed/FeedControls.tsx @@ -0,0 +1,217 @@ +import { useState } from "react"; +import { isAppleDevice } from "../../Utils/utils"; +import FeedButton from "./FeedButton"; +import CareIcon from "../../CAREUI/icons/CareIcon"; +import { PTZPayload } from "./useOperateCamera"; + +const Actions = { + UP: 1 << 0, + DOWN: 1 << 1, + LEFT: 1 << 2, + RIGHT: 1 << 3, + ZOOM_IN: 1 << 4, + ZOOM_OUT: 1 << 5, +} as const; + +const Shortcuts = { + MoveUp: [isAppleDevice ? "Meta" : "Ctrl", "Shift", "ArrowUp"], + MoveLeft: [isAppleDevice ? "Meta" : "Ctrl", "Shift", "ArrowLeft"], + MoveDown: [isAppleDevice ? "Meta" : "Ctrl", "Shift", "ArrowDown"], + MoveRight: [isAppleDevice ? "Meta" : "Ctrl", "Shift", "ArrowRight"], + TogglePrecision: ["Shift", "P"], + ZoomIn: [isAppleDevice ? "Meta" : "Ctrl", "I"], + ZoomOut: [isAppleDevice ? "Meta" : "Ctrl", "O"], + Reset: ["Shift", "R"], + SavePreset: [isAppleDevice ? "Meta" : "Ctrl", "Shift", "S"], + Fullscreen: ["Shift", "F"], +}; + +export type PTZAction = keyof typeof Actions; + +/** + * Returns the PTZ payload for the given action + * + * Example: + * ``` + * payload(Actions.TOP | Actions.LEFT); + * ``` + * + * @param action An Actions or a combination of Actions + * @param precision Precision of the PTZ action + * @returns The PTZ payload + */ +const payload = (action: number, precision: number) => { + let [x, y, zoom] = [0, 0, 0]; + const delta = 0.1 / Math.max(1, precision); + + const _ = (direction: number) => action & direction && delta; + + x -= _(Actions.LEFT); + x += _(Actions.RIGHT); + y += _(Actions.UP); + y -= _(Actions.DOWN); + zoom += _(Actions.ZOOM_IN); + zoom -= _(Actions.ZOOM_OUT); + + return { x, y, zoom }; +}; + +interface Props { + shortcutsDisabled?: boolean; + onMove: (payload: PTZPayload) => void; + isFullscreen: boolean; + setFullscreen: (state: boolean) => void; + onReset: () => void; +} + +export default function FeedControls({ shortcutsDisabled, ...props }: Props) { + const [precision, setPrecision] = useState(1); + const togglePrecision = () => setPrecision((p) => (p === 16 ? 1 : p << 1)); + + const move = (direction: number) => () => { + props.onMove(payload(direction, precision)); + }; + + return ( +
+
+
    +
  • + + + +
  • +
  • + + + +
  • +
  • + + + +
  • +
  • + + + +
  • + +
  • + + {precision}x + +
  • + +
  • + + + +
  • + +
  • + + + +
  • + +
  • + + + +
  • + +
  • + + + +
  • +
+
+ +
+
+ + + + + + + + + + {/* TODO: implement this when this is used in where presets can be saved. */} + {/* console.error("Not implemented")} + shortcutsDisabled={shortcutsDisabled} + > + + */} + props.setFullscreen(!props.isFullscreen)} + shortcutsDisabled={shortcutsDisabled} + > + + +
+
+
+ ); +} diff --git a/src/Components/CameraFeed/FeedNetworkSignal.tsx b/src/Components/CameraFeed/FeedNetworkSignal.tsx new file mode 100644 index 00000000000..55b62ba8620 --- /dev/null +++ b/src/Components/CameraFeed/FeedNetworkSignal.tsx @@ -0,0 +1,58 @@ +import { useEffect, useState } from "react"; +import { calculateVideoDelay } from "./utils"; +import NetworkSignal from "../../CAREUI/display/NetworkSignal"; +import { StreamStatus } from "./usePlayer"; + +interface Props { + playerRef: React.RefObject; + playedOn: Date | undefined; + status: StreamStatus; + onReset: () => void; +} + +export default function FeedNetworkSignal(props: Props) { + const [videoDelay, setVideoDelay] = useState(); + useEffect(() => { + const interval = setInterval(() => { + const delay = calculateVideoDelay(props.playerRef, props.playedOn); + setVideoDelay(delay); + + // Voluntarily resetting for negative delays too as: + // 1. We should not allow users to see what happens in the future! + // They'll figure out that we have a time machine in our hands. + // 2. This value may become negative when the web-socket stream + // disconnects while the tab was not in focus. + if (-5 > delay || delay > 5) { + props.onReset(); + } + }, 500); + + return () => { + clearInterval(interval); + }; + }, [props.playedOn, props.onReset, setVideoDelay]); + + return ( + + + {videoDelay ? ( + `${(videoDelay * 1e3) | 1} ms` + ) : ( + No signal + )} + + + ); +} + +const getStrength = (status: StreamStatus, videoDelay?: number) => { + if (status !== "playing" || videoDelay === undefined) { + return 0; + } + + const ms = videoDelay * 1e3; + + if (ms < 500) return 3; + if (ms < 2000) return 2; + return 1; +}; diff --git a/src/Components/CameraFeed/NoFeedAvailable.tsx b/src/Components/CameraFeed/NoFeedAvailable.tsx new file mode 100644 index 00000000000..aaf74dadec6 --- /dev/null +++ b/src/Components/CameraFeed/NoFeedAvailable.tsx @@ -0,0 +1,58 @@ +import CareIcon, { IconName } from "../../CAREUI/icons/CareIcon"; +import { classNames } from "../../Utils/utils"; +import { AssetData } from "../Assets/AssetTypes"; +import ButtonV2 from "../Common/components/ButtonV2"; + +interface Props { + className?: string; + icon: IconName; + message: string; + streamUrl: string; + onResetClick: () => void; + asset: AssetData; +} + +export default function NoFeedAvailable(props: Props) { + const redactedURL = props.streamUrl + // Replace all uuids in the URL with "ID_REDACTED" + .replace(/[a-f\d]{8}-[a-f\d]{4}-[a-f\d]{4}-[a-f\d]{4}-[a-f\d]{12}/gi, "***") + // Replace all numbers in the URL's path params with "ID_REDACTED" + .replace(/\/\d+/g, "/***"); + + return ( +
+ + {props.message} + + {redactedURL} + +
+ + + Retry + + + + Configure + +
+
+ ); +} diff --git a/src/Components/CameraFeed/routes.ts b/src/Components/CameraFeed/routes.ts new file mode 100644 index 00000000000..f3374a1776d --- /dev/null +++ b/src/Components/CameraFeed/routes.ts @@ -0,0 +1,11 @@ +import { Type } from "../../Redux/api"; +import { OperationAction } from "./useOperateCamera"; + +export const FeedRoutes = { + operateAsset: { + path: "/api/v1/asset/{id}/operate_assets/", + method: "POST", + TRes: Type(), + TBody: Type<{ action: OperationAction }>(), + }, +} as const; diff --git a/src/Components/CameraFeed/useOperateCamera.ts b/src/Components/CameraFeed/useOperateCamera.ts new file mode 100644 index 00000000000..259e45fcc89 --- /dev/null +++ b/src/Components/CameraFeed/useOperateCamera.ts @@ -0,0 +1,54 @@ +import request from "../../Utils/request/request"; +import { FeedRoutes } from "./routes"; + +export interface PTZPayload { + x: number; + y: number; + zoom: number; +} + +interface GetStatusOperation { + type: "get_status"; +} + +interface GetPresetsOperation { + type: "get_presets"; +} + +interface GoToPresetOperation { + type: "goto_preset"; + data: { + preset: number; + }; +} + +interface AbsoluteMoveOperation { + type: "absolute_move"; + data: PTZPayload; +} + +interface RelativeMoveOperation { + type: "relative_move"; + data: PTZPayload; +} + +export type OperationAction = + | GetStatusOperation + | GetPresetsOperation + | GoToPresetOperation + | AbsoluteMoveOperation + | RelativeMoveOperation; + +/** + * This hook is used to control the PTZ of a camera asset and retrieve other related information. + * @param id The external id of the camera asset + */ +export default function useOperateCamera(id: string, silent = false) { + return (action: OperationAction) => { + return request(FeedRoutes.operateAsset, { + pathParams: { id }, + body: { action }, + silent, + }); + }; +} diff --git a/src/Components/CameraFeed/usePlayer.tsx b/src/Components/CameraFeed/usePlayer.tsx new file mode 100644 index 00000000000..b106a25378f --- /dev/null +++ b/src/Components/CameraFeed/usePlayer.tsx @@ -0,0 +1,58 @@ +import { MutableRefObject, useCallback, useState } from "react"; +import ReactPlayer from "react-player"; +import { isIOS } from "../../Utils/utils"; +import { useHLSPLayer } from "../../Common/hooks/useHLSPlayer"; +import { IOptions, useMSEMediaPlayer } from "../../Common/hooks/useMSEplayer"; + +export type StreamStatus = "playing" | "stop" | "loading" | "offline"; + +export default function usePlayer( + streamUrl: string, + ref: MutableRefObject +) { + const [playedOn, setPlayedOn] = useState(); + const [status, setStatus] = useState("stop"); + + // Voluntarily disabling react-hooks/rules-of-hooks for this line as order of + // hooks is maintained (since platform won't change in runtime) + const _start = isIOS + ? // eslint-disable-next-line react-hooks/rules-of-hooks + useHLSPLayer(ref.current as ReactPlayer).startStream + : // eslint-disable-next-line react-hooks/rules-of-hooks + useMSEMediaPlayer({ + // Voluntarily set to "" as it's used by `stopStream` only (which is not + // used by this hook) + config: { middlewareHostname: "" }, + url: streamUrl, + videoEl: ref.current as HTMLVideoElement, + }).startStream; + + const initializeStream = useCallback( + ({ onSuccess, onError }: IOptions) => { + setPlayedOn(undefined); + setStatus("loading"); + _start({ + onSuccess, + onError: (args) => { + setStatus("offline"); + onError?.(args); + }, + }); + }, + [ref.current, streamUrl] + ); + + const onPlayCB = () => { + // Voluntarily updating only if previously undefined (as this method may be invoked by the HTML video element on tab re-focus) + setPlayedOn((prev) => (prev === undefined ? new Date() : prev)); + setStatus("playing"); + }; + + return { + status, + setStatus, + initializeStream, + playedOn, + onPlayCB, + }; +} diff --git a/src/Components/CameraFeed/utils.ts b/src/Components/CameraFeed/utils.ts new file mode 100644 index 00000000000..b5b8920fd5a --- /dev/null +++ b/src/Components/CameraFeed/utils.ts @@ -0,0 +1,28 @@ +import { MutableRefObject } from "react"; +import { AssetData } from "../Assets/AssetTypes"; +import { getCameraConfig } from "../../Utils/transformUtils"; +import { isIOS } from "../../Utils/utils"; + +export const calculateVideoDelay = ( + ref: MutableRefObject, + playedOn?: Date +) => { + const video = ref.current; + + if (!video || !playedOn) { + return 0; + } + + const playedDuration = (new Date().getTime() - playedOn.getTime()) / 1e3; + return playedDuration - video.currentTime; +}; + +export const getStreamUrl = (asset: AssetData, fallbackMiddleware?: string) => { + const config = getCameraConfig(asset); + const host = config.middleware_hostname || fallbackMiddleware; + const uuid = config.accessKey; + + return isIOS + ? `https://${host}/stream/${uuid}/channel/0/hls/live/index.m3u8?uuid=${uuid}&channel=0` + : `wss://${host}/stream/${uuid}/channel/0/mse?uuid=${uuid}&channel=0`; +}; diff --git a/src/Components/Common/components/Page.tsx b/src/Components/Common/components/Page.tsx index d3ca7e5e009..ce5e84deefe 100644 --- a/src/Components/Common/components/Page.tsx +++ b/src/Components/Common/components/Page.tsx @@ -1,16 +1,33 @@ -import { RefObject } from "react"; +import { RefObject, useContext, useEffect } from "react"; import PageTitle, { PageTitleProps } from "../PageTitle"; import { classNames } from "../../../Utils/utils"; +import { SidebarShrinkContext } from "../Sidebar/Sidebar"; interface PageProps extends PageTitleProps { - children: any; - options?: any; + children: React.ReactNode | React.ReactNode[]; + options?: React.ReactNode | React.ReactNode[]; className?: string; noImplicitPadding?: boolean; ref?: RefObject; + /** + * If true, the sidebar will be collapsed when mounted, and restored to original state when unmounted. + * @default false + **/ + collapseSidebar?: boolean; } export default function Page(props: PageProps) { + const sidebar = useContext(SidebarShrinkContext); + + useEffect(() => { + if (!props.collapseSidebar) return; + + sidebar.setShrinked(true); + return () => { + sidebar.setShrinked(sidebar.shrinked); + }; + }, [props.collapseSidebar]); + let padding = ""; if (!props.noImplicitPadding) { if (!props.hideBack || props.componentRight) diff --git a/src/Components/Facility/CentralNursingStation.tsx b/src/Components/Facility/CentralNursingStation.tsx index 0eae504399c..601f579e8d0 100644 --- a/src/Components/Facility/CentralNursingStation.tsx +++ b/src/Components/Facility/CentralNursingStation.tsx @@ -1,6 +1,6 @@ import { useDispatch } from "react-redux"; import useFullscreen from "../../Common/hooks/useFullscreen"; -import { Fragment, useContext, useEffect, useState } from "react"; +import { Fragment, useEffect, useState } from "react"; import { getPermittedFacility, listPatientAssetBeds, @@ -15,7 +15,6 @@ import CareIcon from "../../CAREUI/icons/CareIcon"; import { classNames } from "../../Utils/utils"; import { LocationSelect } from "../Common/LocationSelect"; import Pagination from "../Common/Pagination"; -import { SidebarShrinkContext } from "../Common/Sidebar/Sidebar"; import { PatientAssetBed } from "../Assets/AssetTypes"; import { Popover, Transition } from "@headlessui/react"; import { FieldLabel } from "../Form/FormFields/FormField"; @@ -42,7 +41,6 @@ export default function CentralNursingStation({ facilityId }: Props) { const { t } = useTranslation(); const dispatch = useDispatch(); const [isFullscreen, setFullscreen] = useFullscreen(); - const sidebar = useContext(SidebarShrinkContext); const [facilityObject, setFacilityObject] = useState(); const [data, setData] = @@ -52,15 +50,6 @@ export default function CentralNursingStation({ facilityId }: Props) { limit: PER_PAGE_LIMIT, }); - // To automatically collapse sidebar. - useEffect(() => { - sidebar.setShrinked(true); - - return () => { - sidebar.setShrinked(sidebar.shrinked); - }; - }, []); - useEffect(() => { async function fetchFacilityOrObject() { if (facilityObject) return facilityObject; @@ -135,6 +124,7 @@ export default function CentralNursingStation({ facilityId }: Props) { backUrl={`/facility/${facilityId}/`} noImplicitPadding breadcrumbs={false} + collapseSidebar options={
@@ -264,8 +254,8 @@ export default function CentralNursingStation({ facilityId }: Props) {
) : (
- {data.map((props) => ( -
+ {data.map((props, i) => ( +
= ({ consultationId, facilityId }) => { ); } )} -
+
diff --git a/src/Components/Facility/Consultations/LiveFeed.tsx b/src/Components/Facility/Consultations/LiveFeed.tsx index cd3f055921e..a8f98a3c53b 100644 --- a/src/Components/Facility/Consultations/LiveFeed.tsx +++ b/src/Components/Facility/Consultations/LiveFeed.tsx @@ -461,7 +461,7 @@ const LiveFeed = (props: any) => { ); })} -
+
diff --git a/src/Components/Facility/FacilityHome.tsx b/src/Components/Facility/FacilityHome.tsx index 3ad295a3df3..b36a3a7b36d 100644 --- a/src/Components/Facility/FacilityHome.tsx +++ b/src/Components/Facility/FacilityHome.tsx @@ -4,7 +4,7 @@ import AuthorizeFor, { NonReadOnlyUsers } from "../../Utils/AuthorizeFor"; import { FacilityModel } from "./models"; import { FACILITY_FEATURE_TYPES, USER_TYPES } from "../../Common/constants"; import DropdownMenu, { DropdownItem } from "../Common/components/Menu"; -import { lazy, useState } from "react"; +import { Fragment, lazy, useState } from "react"; import ButtonV2 from "../Common/components/ButtonV2"; import CareIcon from "../../CAREUI/icons/CareIcon"; @@ -27,6 +27,10 @@ import useQuery from "../../Utils/request/useQuery.js"; import { FacilityHomeTriage } from "./FacilityHomeTriage.js"; import { FacilityDoctorList } from "./FacilityDoctorList.js"; import { FacilityBedCapacity } from "./FacilityBedCapacity.js"; +import useSlug from "../../Common/hooks/useSlug.js"; +import { Popover, Transition } from "@headlessui/react"; +import { FieldLabel } from "../Form/FormFields/FormField.js"; +import { LocationSelect } from "../Common/LocationSelect.js"; const Loading = lazy(() => import("../Common/Loading")); @@ -265,7 +269,7 @@ export const FacilityHome = (props: any) => {
- {facilityData?.features?.some((feature: any) => + {facilityData?.features?.some((feature) => FACILITY_FEATURE_TYPES.some((f) => f.id === feature) ) && (

Available features

@@ -390,16 +394,7 @@ export const FacilityHome = (props: any) => { Central Nursing Station - navigate(`/facility/${facilityId}/livefeed`)} - > - - Live Monitoring - + { ); }; + +const LiveMonitoringButton = () => { + const facilityId = useSlug("facility"); + const [location, setLocation] = useState(); + + return ( + + + + + Live Monitoring + + + + +
+
+
+ + Choose a location + +
+ setLocation(v as string | undefined)} + selected={location ?? null} + showAll={false} + multiple={false} + facilityId={facilityId} + errors="" + errorClassName="hidden" + /> +
+
+ + Open Live Monitoring + +
+
+
+
+
+ ); +}; diff --git a/src/Components/Facility/LiveFeedScreen.tsx b/src/Components/Facility/LiveFeedScreen.tsx deleted file mode 100644 index e744ef404cd..00000000000 --- a/src/Components/Facility/LiveFeedScreen.tsx +++ /dev/null @@ -1,230 +0,0 @@ -import { Fragment, useContext, useEffect, useState } from "react"; -import useFilters from "../../Common/hooks/useFilters"; -import useFullscreen from "../../Common/hooks/useFullscreen"; -import { FacilityModel } from "./models"; -import Loading from "../Common/Loading"; -import Page from "../Common/components/Page"; -import ButtonV2 from "../Common/components/ButtonV2"; -import CareIcon from "../../CAREUI/icons/CareIcon"; -import { classNames } from "../../Utils/utils"; -import { LocationSelect } from "../Common/LocationSelect"; -import Pagination from "../Common/Pagination"; -import { SidebarShrinkContext } from "../Common/Sidebar/Sidebar"; -import { AssetData } from "../Assets/AssetTypes"; -import { Popover, Transition } from "@headlessui/react"; -import { FieldLabel } from "../Form/FormFields/FormField"; -import CheckBoxFormField from "../Form/FormFields/CheckBoxFormField"; -import LiveFeedTile from "./LiveFeedTile"; -import { getCameraConfig } from "../../Utils/transformUtils"; -import { getPermittedFacility, listAssets } from "../../Redux/actions"; -import { useDispatch } from "react-redux"; - -const PER_PAGE_LIMIT = 6; - -interface Props { - facilityId: string; -} - -export default function LiveFeedScreen({ facilityId }: Props) { - const dispatch = useDispatch(); - const [isFullscreen, setFullscreen] = useFullscreen(); - const sidebar = useContext(SidebarShrinkContext); - - const [facility, setFacility] = useState(); - const [assets, setAssets] = useState(); - const [totalCount, setTotalCount] = useState(0); - const { qParams, updateQuery, removeFilter, updatePage } = useFilters({ - limit: PER_PAGE_LIMIT, - }); - - const [refresh_presets_hash, setRefreshPresetsHash] = useState( - Number(new Date()) - ); - - // To automatically collapse sidebar. - useEffect(() => { - sidebar.setShrinked(true); - - return () => { - sidebar.setShrinked(sidebar.shrinked); - }; - }, []); - - useEffect(() => { - async function fetchFacilityOrObject() { - if (facility) return facility; - const res = await dispatch(getPermittedFacility(facilityId)); - if (res.status !== 200) return; - setFacility(res.data); - return res.data as FacilityModel; - } - - async function fetchData() { - setAssets(undefined); - - const filters = { - ...qParams, - page: qParams.page || 1, - limit: PER_PAGE_LIMIT, - offset: (qParams.page ? qParams.page - 1 : 0) * PER_PAGE_LIMIT, - asset_class: "ONVIF", - facility: facilityId || "", - location: qParams.location, - bed_is_occupied: qParams.bed_is_occupied, - }; - - const [facilityObj, res] = await Promise.all([ - fetchFacilityOrObject(), - dispatch(listAssets(filters)), - ]); - - if (!facilityObj || res.status !== 200) { - return; - } - console.log(facilityObj, res.data); - const entries = res.data.results; - - setTotalCount(entries.length); - setAssets(entries); - } - fetchData(); - setRefreshPresetsHash(Number(new Date())); - }, [ - dispatch, - facilityId, - qParams.page, - qParams.location, - qParams.ordering, - qParams.bed_is_occupied, - ]); - - return ( - - - - - - Settings and Filters - - - - -
-
-
- - {totalCount} Camera - present - -
-
-
-
- - Filter by Location - -
- { - location - ? updateQuery({ location }) - : removeFilter("location"); - }} - selected={qParams.location} - showAll={false} - multiple={false} - facilityId={facilityId} - errors="" - errorClassName="hidden" - /> -
-
- { - if (value) { - updateQuery({ [name]: value }); - } else { - removeFilter(name); - } - }} - labelClassName="text-sm" - errorClassName="hidden" - /> - setFullscreen(!isFullscreen)} - className="tooltip !h-11" - > - - {isFullscreen ? "Exit Fullscreen" : "Fullscreen"} - -
-
-
-
-
- - updatePage(page)} - /> -
- } - > - {assets === undefined ? ( - - ) : assets.length === 0 ? ( -
- No Camera present in this location or facility. -
- ) : ( -
- {assets.map((asset, idx) => ( -
- {/* */} - -
- ))} -
- )} - - ); -} diff --git a/src/Components/Facility/LiveFeedTile.tsx b/src/Components/Facility/LiveFeedTile.tsx deleted file mode 100644 index 945240b9c6c..00000000000 --- a/src/Components/Facility/LiveFeedTile.tsx +++ /dev/null @@ -1,1045 +0,0 @@ -// import axios from "axios"; -// import React, { useEffect, useState, useRef, useCallback } from "react"; -// import * as Notification from "../../Utils/Notifications.js"; -// import { useDispatch } from "react-redux"; -// import ReactPlayer from "react-player"; -// import { getAsset, listAssetBeds } from "../../Redux/actions"; -// import { statusType, useAbortableEffect } from "../../Common/utils"; -// import { useTranslation } from "react-i18next"; -// import useFullscreen from "../../Common/hooks/useFullscreen.js"; -// interface LiveFeedTileProps { -// assetId: string; -// } - -// interface CameraPosition { -// x: number; -// y: number; -// zoom: number; -// } - -// // string:string dictionary -// interface CameraPreset { -// [key: string]: string; -// } - -// export default function LiveFeedTile(props: LiveFeedTileProps) { -// const dispatch: any = useDispatch(); -// const { assetId } = props; -// const [sourceUrl, setSourceUrl] = useState(); -// const [asset, setAsset] = useState(); -// const [presets, setPresets] = useState([]); -// const [bedPresets, setBedPresets] = useState([]); -// const [loading, setLoading] = useState(true); -// // const [showControls, setShowControls] = useState(false); -// const [showDefaultPresets, setShowDefaultPresets] = useState(false); -// const [position, setPosition] = useState({ -// x: 0, -// y: 0, -// zoom: 0, -// }); -// const { t } = useTranslation(); -// const [_isFullscreen, setFullscreen] = useFullscreen(); -// // const [toggle, setToggle] = useState(false); - -// useEffect(() => { -// let loadingTimeout: any; -// if (loading === true) -// loadingTimeout = setTimeout(() => { -// setLoading(false); -// }, 6000); -// return () => { -// if (loadingTimeout) clearTimeout(loadingTimeout); -// }; -// }, [loading]); - -// const fetchData = useCallback( -// async (status: statusType) => { -// setLoading(true); -// console.log("fetching asset"); -// const assetData: any = await dispatch(getAsset(assetId)); -// if (!status.aborted) { -// // setLoading(false); -// if (!assetData.data) -// Notification.Error({ -// msg: t("something_went_wrong"), -// }); -// else { -// setAsset(assetData.data); -// } -// } -// }, -// [dispatch, assetId] -// ); - -// useAbortableEffect( -// (status: statusType) => fetchData(status), -// [dispatch, fetchData] -// ); -// const requestStream = () => { -// axios -// .post(`https://${asset.meta.middleware_hostname}/start`, { -// uri: "rtsp://remote:qwerty123@192.168.1.64:554/", -// }) -// .then((resp: any) => { -// setSourceUrl( -// `https://${asset.meta.middleware_hostname}${resp.data.uri}` -// ); -// }) -// .catch((_ex: any) => { -// // console.error('Error while refreshing',ex); -// }); -// }; -// const stopStream = (url: string | undefined) => { -// console.log("stop", url); -// if (url) { -// const urlSegments = url.split("/"); -// const id = urlSegments?.pop(); -// axios -// .post(`https://${asset.meta.middleware_hostname}/stop`, { -// id, -// }) -// .then((resp: any) => { -// console.log(resp); -// // setSourceUrl(`https://${middlewareHostname}${resp.data.uri}`); -// }) -// .catch((_ex: any) => { -// // console.error('Error while refreshing',ex); -// }); -// } -// }; -// const getCameraStatus = (asset: any) => { -// axios -// .get( -// `https://${asset.meta.middleware_hostname}/status?hostname=${asset.hostname}&port=${asset.port}&username=${asset.username}&password=${asset.password}` -// ) -// .then((resp: any) => { -// setPosition(resp.data.position); -// }) -// .catch((_ex: any) => { -// // console.error('Error while refreshing',ex); -// }); -// }; -// const getPresets = (asset: any) => { -// const url = `https://${asset.meta.middleware_hostname}/presets?hostname=${asset.hostname}&port=${asset.port}&username=${asset.username}&password=${asset.password}`; -// axios -// .get(url) -// .then((resp: any) => { -// setPresets(resp.data); -// }) -// .catch((_ex: any) => { -// // console.error("Error while refreshing", ex); -// }); -// }; -// const getBedPresets = async (_asset: any) => { -// const bedAssets = await dispatch(listAssetBeds({ asset: props.assetId })); -// setBedPresets(bedAssets.data.results); -// }; -// const gotoBedPreset = (preset: any) => { -// absoluteMove(preset.meta.position); -// }; -// const gotoPreset = (preset: number) => { -// axios -// .post(`https://${asset.meta.middleware_hostname}/gotoPreset`, { -// ...asset, -// preset, -// }) -// .then((resp: any) => { -// console.log(resp.data); -// }) -// .catch((_ex: any) => { -// // console.error('Error while refreshing',ex); -// }); -// }; -// const requestPTZ = (action: string) => { -// setLoading(true); -// if (!position) { -// getCameraStatus(asset); -// } else { -// const data = { -// x: 0, -// y: 0, -// zoom: 0, -// } as any; -// console.log(action); -// // Relative X Y Coordinates -// switch (action) { -// case "up": -// data.y = 0.05; -// break; -// case "down": -// data.y = -0.05; -// break; -// case "left": -// data.x = -0.05; -// break; -// case "right": -// data.x = 0.05; -// break; -// case "zoomIn": -// data.zoom = 0.05; -// break; -// case "zoomOut": -// data.zoom = -0.05; -// break; -// case "stop": -// stopStream(sourceUrl); -// setSourceUrl(undefined); -// return; -// case "reset": -// setSourceUrl(undefined); -// requestStream(); -// return; -// default: -// break; -// } -// axios -// .post(`https://${asset.meta.middleware_hostname}/relativeMove`, { -// ...data, -// ...asset, -// }) -// .then((resp: any) => { -// console.log(resp.data); -// getCameraStatus(asset); -// }) -// .catch((_ex: any) => { -// // console.error('Error while refreshing',ex); -// }); -// } -// }; - -// const absoluteMove = (data: any) => { -// setLoading(true); -// axios -// .post(`https://${asset.meta.middleware_hostname}/absoluteMove`, { -// ...data, -// ...asset, -// }) -// .then((_resp: any) => { -// getCameraStatus(asset); -// }) -// .catch((ex: any) => { -// console.error("Error while absolute move", ex); -// }); -// }; - -// useEffect(() => { -// if (asset) { -// getPresets(asset); -// getBedPresets(asset); -// requestStream(); -// } -// }, [asset]); - -// useEffect(() => { -// if (bedPresets.length > 0) absoluteMove(bedPresets[0].meta.position); -// }, [bedPresets]); - -// // useEffect(() => { -// // const timer = setTimeout(() => { -// // setShowControls(toggle); -// // }, 300); -// // return () => clearTimeout(timer); -// // }, [toggle]); - -// const liveFeedPlayerRef = useRef(null); -// const handleClickFullscreen = () => { -// if (liveFeedPlayerRef.current) { -// setFullscreen(true, liveFeedPlayerRef.current.wrapper); -// } -// }; - -// const viewOptions = presets -// ? Object.entries(presets) -// .map(([key, value]) => ({ label: key, value })) -// .slice(0, 10) -// : Array.from(Array(10), (_, i) => ({ -// label: t("monitor") + (i + 1), -// value: i + 1, -// })); - -// const cameraPTZ = [ -// { icon: "fa fa-arrow-up", label: t("up"), action: "up" }, -// { icon: "fa fa-arrow-down", label: t("down"), action: "down" }, -// { icon: "fa fa-arrow-left", label: t("left"), action: "left" }, -// { icon: "fa fa-arrow-right", label: t("right"), action: "right" }, -// { icon: "fa fa-search-plus", label: t("zoom_in"), action: "zoomIn" }, -// { icon: "fa fa-search-minus", label: t("zoom_out"), action: "zoomOut" }, -// { icon: "fa fa-stop", label: t("stop"), action: "stop" }, -// { icon: "fa fa-undo", label: t("reset"), action: "reset" }, -// ]; - -// return ( -//
-//
-//
-//
-// {sourceUrl ? ( -//
-// { -// // requestStream(); -// console.log("Error", e); -// console.log("Data", data); -// console.log("HLS Instance", hlsInstance); -// console.log("HLS Global", hlsGlobal); -// if (e === "hlsError") { -// const recovered = hlsInstance.recoverMediaError(); -// console.log(recovered); -// } -// }} -// /> -//
-// ) : ( -//
-//

-// STATUS: OFFLINE -//

-//

-// {t("feed_is_currently_not_live")} -//

-//
-// )} -//
-//
-//
-//
-//
-// {cameraPTZ.map((option: any) => ( -//
{ -// // console.log(option.action); -// requestPTZ(option.action); -// }} -// > -// -//
-// ))} -// -//
-// {/*
-// -//
*/} -//
-//
-//
-//
-// {/* div with "Loading" at the center */} -//
-// -// -// -// -//
{t("moving_camera")}
-//
-//
-//
-//
-// -// {showDefaultPresets -// ? viewOptions.map((option: any) => ( -//
{ -// setLoading(true); -// gotoPreset(option.value); -// }} -// > -// -//
-// )) -// : bedPresets.map((preset: any, index: number) => ( -//
{ -// setLoading(true); -// gotoBedPreset(preset); -// }} -// key={preset.id} -// > -// -//
-// ))} -//
-//
-//
-// ); -// } - -import { useEffect, useState, useRef } from "react"; -import { useDispatch } from "react-redux"; -import useKeyboardShortcut from "use-keyboard-shortcut"; -import { - listAssetBeds, - partialUpdateAssetBed, - deleteAssetBed, -} from "../../Redux/actions"; -import { getCameraPTZ } from "../../Common/constants"; -import { - StreamStatus, - useMSEMediaPlayer, -} from "../../Common/hooks/useMSEplayer"; -import { useFeedPTZ } from "../../Common/hooks/useFeedPTZ"; -import * as Notification from "../../Utils/Notifications.js"; -import { AxiosError } from "axios"; -import { BedSelect } from "../Common/BedSelect"; -import { BedModel } from "./models"; -import useWindowDimensions from "../../Common/hooks/useWindowDimensions"; -import CareIcon from "../../CAREUI/icons/CareIcon"; -import ConfirmDialog from "../Common/ConfirmDialog"; -import { FieldLabel } from "../Form/FormFields/FormField"; -import useFullscreen from "../../Common/hooks/useFullscreen"; -import { FeedCameraPTZHelpButton } from "./Consultations/Feed"; - -const LiveFeed = (props: any) => { - const middlewareHostname = - props.middlewareHostname || "dev_middleware.coronasafe.live"; - const [presetsPage, setPresetsPage] = useState(0); - const cameraAsset = props.asset; - const [presets, setPresets] = useState([]); - const [bedPresets, setBedPresets] = useState([]); - const [showDefaultPresets, setShowDefaultPresets] = useState(false); - const [precision, setPrecision] = useState(1); - const [streamStatus, setStreamStatus] = useState( - StreamStatus.Offline - ); - const [videoStartTime, setVideoStartTime] = useState(null); - const [bed, setBed] = useState({}); - const [preset, setNewPreset] = useState(""); - const [loading, setLoading] = useState(); - const dispatch: any = useDispatch(); - const [page, setPage] = useState({ - count: 0, - limit: 8, - offset: 0, - }); - const [toDelete, setToDelete] = useState(null); - const [toUpdate, setToUpdate] = useState(null); - const [_isFullscreen, setFullscreen] = useFullscreen(); - - const { width } = useWindowDimensions(); - const extremeSmallScreenBreakpoint = 320; - const isExtremeSmallScreen = - width <= extremeSmallScreenBreakpoint ? true : false; - const liveFeedPlayerRef = useRef(null); - - const videoEl = liveFeedPlayerRef.current as HTMLVideoElement; - - const url = `wss://${middlewareHostname}/stream/${cameraAsset?.accessKey}/channel/0/mse?uuid=${cameraAsset?.accessKey}&channel=0`; - - const { startStream } = useMSEMediaPlayer({ - config: { - middlewareHostname, - ...cameraAsset, - }, - url, - videoEl, - }); - - const refreshPresetsHash = props.refreshPresetsHash; - - // eslint-disable-next-line @typescript-eslint/no-unused-vars - const [currentPreset, setCurrentPreset] = useState(); - const { - absoluteMove, - getCameraStatus, - getPTZPayload, - getPresets, - gotoPreset, - relativeMove, - } = useFeedPTZ({ - config: { - middlewareHostname, - ...cameraAsset, - }, - dispatch, - }); - - const fetchCameraPresets = () => - getPresets({ - onSuccess: (resp) => { - setPresets(resp); - }, - onError: (resp) => { - resp instanceof AxiosError && - Notification.Error({ - msg: "Camera is offline", - }); - }, - }); - - const calculateVideoLiveDelay = () => { - const video = liveFeedPlayerRef.current as HTMLVideoElement; - if (!video || !videoStartTime) return 0; - - const timeDifference = - (new Date().getTime() - videoStartTime.getTime()) / 1000; - - return timeDifference - video.currentTime; - }; - - const getBedPresets = async (id: any) => { - const bedAssets = await dispatch( - listAssetBeds({ - asset: id, - limit: page.limit, - offset: page.offset, - }) - ); - setBedPresets(bedAssets?.data?.results); - setPage({ - ...page, - count: bedAssets?.data?.count, - }); - }; - - const deletePreset = async (id: any) => { - const res = await dispatch(deleteAssetBed(id)); - if (res?.status === 204) { - Notification.Success({ msg: "Preset deleted successfully" }); - getBedPresets(cameraAsset.id); - } else { - Notification.Error({ - msg: "Error while deleting Preset: " + (res?.data?.detail || ""), - }); - } - setToDelete(null); - }; - - const updatePreset = async (currentPreset: any) => { - const data = { - bed_id: bed.id, - preset_name: preset, - }; - const response = await dispatch( - partialUpdateAssetBed( - { - asset: currentPreset.asset_object.id, - bed: bed.id, - meta: { - ...currentPreset.meta, - ...data, - }, - }, - currentPreset?.id - ) - ); - if (response && response.status === 200) { - Notification.Success({ msg: "Preset Updated" }); - } else { - Notification.Error({ msg: "Something Went Wrong" }); - } - getBedPresets(cameraAsset?.id); - fetchCameraPresets(); - setToUpdate(null); - }; - - const gotoBedPreset = (preset: any) => { - setLoading("Moving"); - absoluteMove(preset.meta.position, { - onSuccess: () => setLoading(undefined), - }); - }; - - useEffect(() => { - if (cameraAsset?.hostname) { - fetchCameraPresets(); - } - }, []); - - useEffect(() => { - setNewPreset(toUpdate?.meta?.preset_name); - setBed(toUpdate?.bed_object); - }, [toUpdate]); - - useEffect(() => { - getBedPresets(cameraAsset.id); - if (bedPresets?.[0]?.position) { - absoluteMove(bedPresets[0]?.position, {}); - } - }, [page.offset, cameraAsset.id, refreshPresetsHash]); - - const viewOptions = (page: number) => { - return presets - ? Object.entries(presets) - .map(([key, value]) => ({ label: key, value })) - .slice(page, page + 10) - : Array.from(Array(10), (_, i) => ({ - label: "Monitor " + (i + 1), - value: i + 1, - })); - }; - useEffect(() => { - let tId: any; - if (streamStatus !== StreamStatus.Playing) { - setStreamStatus(StreamStatus.Loading); - tId = setTimeout(() => { - startStream({ - onSuccess: () => setStreamStatus(StreamStatus.Playing), - onError: () => setStreamStatus(StreamStatus.Offline), - }); - }, 500); - } - - return () => { - clearTimeout(tId); - }; - }, [startStream, streamStatus]); - - const handlePagination = (cOffset: number) => { - setPage({ - ...page, - offset: cOffset, - }); - }; - - const cameraPTZActionCBs: { [key: string]: (option: any) => void } = { - precision: () => { - setPrecision((precision: number) => - precision === 16 ? 1 : precision * 2 - ); - }, - reset: () => { - setStreamStatus(StreamStatus.Loading); - setVideoStartTime(null); - startStream({ - onSuccess: () => setStreamStatus(StreamStatus.Playing), - onError: () => setStreamStatus(StreamStatus.Offline), - }); - }, - fullScreen: () => { - if (!liveFeedPlayerRef.current) return; - setFullscreen(true, liveFeedPlayerRef.current); - }, - updatePreset: (option) => { - getCameraStatus({ - onSuccess: async (data) => { - console.log({ currentPreset, data }); - if (currentPreset?.asset_object?.id && data?.position) { - setLoading(option.loadingLabel); - console.log("Updating Preset"); - const response = await dispatch( - partialUpdateAssetBed( - { - asset: currentPreset.asset_object.id, - bed: currentPreset.bed_object.id, - meta: { - ...currentPreset.meta, - position: data?.position, - }, - }, - currentPreset?.id - ) - ); - if (response && response.status === 200) { - Notification.Success({ msg: "Preset Updated" }); - getBedPresets(cameraAsset?.id); - fetchCameraPresets(); - } - setLoading(undefined); - } - }, - }); - }, - other: (option) => { - setLoading(option.loadingLabel); - relativeMove(getPTZPayload(option.action, precision), { - onSuccess: () => setLoading(undefined), - }); - }, - }; - - const cameraPTZ = getCameraPTZ(precision).map((option) => { - const cb = - cameraPTZActionCBs[ - cameraPTZActionCBs[option.action] ? option.action : "other" - ]; - return { ...option, callback: () => cb(option) }; - }); - - // Voluntarily disabling eslint, since length of `cameraPTZ` is constant and - // hence shall not cause issues. (https://news.ycombinator.com/item?id=24363703) - for (const option of cameraPTZ) { - if (!option.shortcutKey) continue; - // eslint-disable-next-line react-hooks/rules-of-hooks - useKeyboardShortcut(option.shortcutKey, option.callback); - } - - return ( -
- {toDelete && ( - -

- Preset: {toDelete.meta.preset_name} -

-

- Bed: {toDelete.bed_object.name} -

- - } - action="Delete" - variant="danger" - onClose={() => setToDelete(null)} - onConfirm={() => deletePreset(toDelete.id)} - /> - )} - {toUpdate && ( - setToUpdate(null)} - onConfirm={() => updatePreset(toUpdate)} - > -
- Bed - setBed(selected as BedModel)} - selected={bed} - error="" - multiple={false} - location={cameraAsset.location_id} - facility={cameraAsset.facility_id} - /> -
-
- )} -
-
-
- {/* ADD VIDEO PLAYER HERE */} -
- - - {streamStatus === StreamStatus.Playing && - calculateVideoLiveDelay() > 3 && ( -
- - Slow Network Detected -
- )} - - {loading && ( -
-
-
-

{loading}

-
-
- )} - {/* { streamStatus > 0 && */} -
- {streamStatus === StreamStatus.Offline && ( -
-

- STATUS: OFFLINE -

-

- Feed is currently not live. -

-

- Click refresh button to try again. -

-
- )} - {streamStatus === StreamStatus.Stop && ( -
-

- STATUS: STOPPED -

-

Feed is Stooped.

-

- Click refresh button to start feed. -

-
- )} - {streamStatus === StreamStatus.Loading && ( -
-

- STATUS: LOADING -

-

- Fetching latest feed. -

-
- )} -
-
-
- {cameraPTZ.map((option) => { - const shortcutKeyDescription = - option.shortcutKey && - option.shortcutKey - .join(" + ") - .replace("Control", "Ctrl") - .replace("ArrowUp", "↑") - .replace("ArrowDown", "↓") - .replace("ArrowLeft", "←") - .replace("ArrowRight", "→"); - - return ( - - ); - })} -
- -
-
-
- -
- -
-
- {showDefaultPresets ? ( - <> - {viewOptions(presetsPage)?.map((option: any, i) => ( - - ))} - - ) : ( - <> - {bedPresets?.map((preset: any, index: number) => ( -
- -
- - -
-
- ))} - - )} -
- {/* Page Number Next and Prev buttons */} - {showDefaultPresets ? ( -
- - -
- ) : ( -
- - -
- )} - {props?.showRefreshButton && ( - - )} -
-
-
-
-
- ); -}; - -export default LiveFeed; diff --git a/src/Components/VitalsMonitor/WaveformLabels.tsx b/src/Components/VitalsMonitor/WaveformLabels.tsx index 92ebb6b8145..5e3fdf09bef 100644 --- a/src/Components/VitalsMonitor/WaveformLabels.tsx +++ b/src/Components/VitalsMonitor/WaveformLabels.tsx @@ -7,8 +7,11 @@ interface Props { export default function WaveformLabels({ labels }: Props) { return (
- {Object.entries(labels).map(([label, className]) => ( - + {Object.entries(labels).map(([label, className], i) => ( + {label} ))} diff --git a/src/Redux/api.tsx b/src/Redux/api.tsx index 2a64d921792..2a7681975a4 100644 --- a/src/Redux/api.tsx +++ b/src/Redux/api.tsx @@ -996,7 +996,7 @@ const routes = { // Assets endpoints listAssets: { - path: "/api/v1/asset", + path: "/api/v1/asset/", method: "GET", TRes: Type>(), }, diff --git a/src/Routers/routes/FacilityLocationRoutes.tsx b/src/Routers/routes/FacilityLocationRoutes.tsx index c43673b60f5..aa08661da8b 100644 --- a/src/Routers/routes/FacilityLocationRoutes.tsx +++ b/src/Routers/routes/FacilityLocationRoutes.tsx @@ -3,6 +3,7 @@ import { AddInventoryForm } from "../../Components/Facility/AddInventoryForm"; import { AddLocationForm } from "../../Components/Facility/AddLocationForm"; import { BedManagement } from "../../Components/Facility/BedManagement"; import LocationManagement from "../../Components/Facility/LocationManagement"; +import CentralLiveMonitoring from "../../Components/CameraFeed/CentralLiveMonitoring"; export default { "/facility/:facilityId/location": ({ facilityId }: any) => ( @@ -35,4 +36,7 @@ export default { }: any) => ( ), + "/facility/:facilityId/live-monitoring": (props: any) => ( + + ), }; diff --git a/src/Routers/routes/FacilityRoutes.tsx b/src/Routers/routes/FacilityRoutes.tsx index be52ef4e72c..77247df9189 100644 --- a/src/Routers/routes/FacilityRoutes.tsx +++ b/src/Routers/routes/FacilityRoutes.tsx @@ -8,7 +8,6 @@ import ResourceCreate from "../../Components/Resource/ResourceCreate"; import CentralNursingStation from "../../Components/Facility/CentralNursingStation"; import FacilityLocationRoutes from "./FacilityLocationRoutes"; import FacilityInventoryRoutes from "./FacilityInventoryRoutes"; -import LiveFeedScreen from "../../Components/Facility/LiveFeedScreen"; export default { "/facility": () => , @@ -22,9 +21,6 @@ export default { "/facility/:facilityId/cns": ({ facilityId }: any) => ( ), - "/facility/:facilityId/livefeed": ({ facilityId }: any) => ( - - ), "/facility/:facilityId": ({ facilityId }: any) => ( ), diff --git a/src/Utils/utils.ts b/src/Utils/utils.ts index cd75fdba7e7..7e34d027020 100644 --- a/src/Utils/utils.ts +++ b/src/Utils/utils.ts @@ -175,10 +175,15 @@ function _isAppleDevice() { } /** - * `true` if device is iOS, else `false` + * `true` if device is an Apple device, else `false` */ export const isAppleDevice = _isAppleDevice(); +/** + * `true` if device is an iOS device, else `false` + */ +export const isIOS = /iPad|iPhone|iPod/.test(navigator.userAgent); + /** * Conditionally concatenate classes. An alternate replacement for `clsx`. * diff --git a/src/style/CAREUI.css b/src/style/CAREUI.css index 404fcc3c4da..eff9ae91640 100644 --- a/src/style/CAREUI.css +++ b/src/style/CAREUI.css @@ -39,7 +39,7 @@ .tooltip .tooltip-text { visibility: hidden; opacity: 0; - @apply bg-black/75 backdrop-blur text-white text-center p-2 rounded absolute z-50 text-sm block transition-all whitespace-nowrap + @apply bg-black/75 backdrop-blur text-white text-center p-2 rounded absolute z-50 text-sm block transition-opacity whitespace-nowrap pointer-events-none } .tooltip .tooltip-left { diff --git a/src/style/index.css b/src/style/index.css index 787290bab26..8767498e42a 100644 --- a/src/style/index.css +++ b/src/style/index.css @@ -564,12 +564,6 @@ button:disabled, background: rgba(0, 0, 0, 0.4); } -@media (max-width:640px) { - .hideonmobilescreen { - display: none; - } -} - @media (min-width:1000px) { .manualGrid { display: grid !important