diff --git a/.eslintrc b/.eslintrc index 0c50a86..9e8c845 100644 --- a/.eslintrc +++ b/.eslintrc @@ -1,18 +1,37 @@ { + "env": { + "node": true + }, + "extends": ["eslint:recommended", "plugin:prettier/recommended", "plugin:@typescript-eslint/recommended"], "parser": "@typescript-eslint/parser", - "plugins": ["@typescript-eslint"], - "root": true, - "extends": [ - "eslint:recommended", - "plugin:prettier/recommended", - "plugin:@typescript-eslint/recommended", - "ts-react-important-stuff" - ], + "plugins": ["@typescript-eslint", "react-hooks"], "rules": { + // Disabling these rules for now just to keep the diff small. I'll enable them one by one as we go. + "@typescript-eslint/ban-ts-comment": "off", + "@typescript-eslint/ban-types": "off", + "@typescript-eslint/no-explicit-any": "off", + "@typescript-eslint/no-unused-vars": "off", + "@typescript-eslint/no-var-requires": "off", + "@typescript-eslint/triple-slash-reference": "off", + // Use `import type` instead of `import` for type imports https://typescript-eslint.io/blog/consistent-type-imports-and-exports-why-and-how + "@typescript-eslint/consistent-type-imports": [ + "error", + { + "fixStyle": "inline-type-imports" + } + ], + "prefer-const": "off", + "no-console": ["error", { "allow": ["warn", "error"] }], + "no-unsafe-optional-chaining": "off", + "no-explicit-any": "off", + "no-extra-boolean-cast": "off", + "no-prototype-builtins": "off", + "no-useless-escape": "off", "no-restricted-imports": [ "error", { "paths": [ + // These two rules ensure that we're importing lodash and lodash-es correctly. Not doing so can bloat our bundle size significantly. { "name": "lodash", "message": "Import specific methods from `lodash`. e.g. `import map from 'lodash/map'`" @@ -22,6 +41,7 @@ "importNames": ["default"], "message": "Import specific methods from `lodash-es`. e.g. `import { map } from 'lodash-es'`" }, + // These two rules ensure that we're importing Carbon components and icons from the correct packages (after v10). May be removed in the future. { "name": "carbon-components-react", "message": "Import from `@carbon/react` directly. e.g. `import { Toggle } from '@carbon/react'`" diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 44d793c..f3a82e5 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -9,8 +9,8 @@ on: types: - created env: - ESM_NAME: "@ugandaemr/esm-primary-navigation-app" - JS_NAME: "esm-ugandaemr-primary-navigation-app.js" + ESM_NAME: "@ugandaemr/esm-appointments-app" + JS_NAME: "esm-ugandaemr-appointments-app.js" jobs: build: diff --git a/.github/workflows/pull_request_template.md b/.github/workflows/pull_request_template.md index e69de29..2db271f 100644 --- a/.github/workflows/pull_request_template.md +++ b/.github/workflows/pull_request_template.md @@ -0,0 +1,18 @@ +## Requirements + +- [ ] This PR has a title that briefly describes the work done including the ticket number. If there is a ticket, make sure your PR title includes a [conventional commit](https://o3-dev.docs.openmrs.org/#/getting_started/contributing?id=your-pr-title-should-indicate-the-type-of-change-it-is) label. See existing PR titles for inspiration. +- [ ] My work conforms to the [OpenMRS 3.0 Styleguide](https://om.rs/styleguide) and [design documentation](https://zeroheight.com/23a080e38/p/880723-introduction). +- [ ] My work includes tests or is validated by existing tests. + +## Summary + + +## Screenshots + + +## Related Issue + + + +## Other + diff --git a/.husky/.gitignore b/.husky/.gitignore new file mode 100644 index 0000000..31354ec --- /dev/null +++ b/.husky/.gitignore @@ -0,0 +1 @@ +_ diff --git a/.husky/pre-commit b/.husky/pre-commit index 610c2a5..b9743a3 100755 --- a/.husky/pre-commit +++ b/.husky/pre-commit @@ -1,4 +1,7 @@ -#!/usr/bin/env sh -. "$(dirname -- "$0")/_/husky.sh" +#!/bin/sh +. "$(dirname "$0")/_/husky.sh" -npm test +set -e # die on error + +yarn prettier && npx lint-staged +yarn turbo run extract-translations diff --git a/.husky/pre-push b/.husky/pre-push new file mode 100755 index 0000000..879f05d --- /dev/null +++ b/.husky/pre-push @@ -0,0 +1,6 @@ +#!/bin/sh +. "$(dirname "$0")/_/husky.sh" + +set -e # die on error + +yarn verify diff --git a/.prettierignore b/.prettierignore index 3817531..ffb6fd2 100644 --- a/.prettierignore +++ b/.prettierignore @@ -1,14 +1,5 @@ -# directories -.husky/ dist/ node_modules/ - -# dotfiles and generated -.* -yarn.lock - -# by file type -**/*.css -**/*.scss **/*.md **/*.json +**/*.yml diff --git a/README.md b/README.md index b05055c..ebd9500 100644 --- a/README.md +++ b/README.md @@ -1,14 +1,5 @@ ![Node.js CI](https://github.com/METS-Programme/esm-ugandaemr-template-app/workflows/Node.js%20CI/badge.svg) -# UgandaEMR Template app +# UgandaEMR Appointments app -![Landing page screenshot](./src/assets/images/screen.png) - -A starter template ESM for UgandaEMR inspired by https://github.com/openmrs/openmrs-esm-template-app. - -You can use this repo as a template to spawn new frontend modules for UgandaEMR. - -The current setup works best for setting up polyrepos. To adapt the template for a monorepo setup, you'll need to make the following changes: - -- Add a `packages` directory inside of `src`. diff --git a/__mocks__/active-visits.mock.ts b/__mocks__/active-visits.mock.ts new file mode 100644 index 0000000..d13af04 --- /dev/null +++ b/__mocks__/active-visits.mock.ts @@ -0,0 +1,101 @@ +export const mockServiceData = [ + { + uuid: "176052c7-5fd4-4b33-89cc-7bae6848c65a", + display: "Clinical consultation", + }, + { uuid: "d80ff12a-06a7-11ed-b939-0242ac120002", display: "Triage" }, +]; + +export const mockVisitQueueEntries = [ + { + uuid: "fa1e98f1-f002-4174-9e55-34d60951e710", + visit: { + uuid: "c90386ff-ae85-45cc-8a01-25852099c5ae", + display: "Facility Visit @ Outpatient Clinic - 04/03/2022 07:22", + }, + queueEntry: { + uuid: "712289ab-32c0-430f-87b6-d9c1e4e4686e", + display: "Eric Test Ric", + priorityComment: "Needs Triage", + sortWeight: 0, + startedAt: "2022-03-04T09:50:54.000+0000", + endedAt: null, + queue: { + uuid: "6a97bd65-3a9a-4fab-ae8f-be59dd4ddd87", + display: "Triage", + name: "Triage", + description: "Queue for patients waiting for triage", + service: { + display: "Triage", + }, + }, + status: { + uuid: "aaec62b1-4b03-4166-ada7-230cb4b4aaaa", + display: "Waiting", + links: [ + { + rel: "self", + uri: "http://openmrs:8080/openmrs/ws/rest/v1/concept/aaec62b1-4b03-4166-ada7-230cb4b4aaaa", + }, + ], + }, + patient: { + uuid: "cc75ad73-c24b-499c-8db9-a7ef4fc0b36d", + display: "10000F1 - Eric Test Ric", + person: { + age: "32", + gender: "F", + }, + }, + priority: { + uuid: "f9684018-a4d3-4d6f-9dd5-b4b1e89af3e7", + display: "Not Urgent", + }, + locationWaitingFor: null, + providerWaitingFor: null, + }, + }, + { + uuid: "2f85d611-5bb9-4bca-b6f8-661517df86c9", + visit: { + uuid: "6b3e233d-2b44-40ca-b0c8-c5a57a8c51b6", + display: "Home Visit @ Outpatient Clinic - 09/03/2022 21:08", + }, + queueEntry: { + uuid: "5f017eb0-b035-4acd-b284-da45f5067502", + display: "John Smith", + priorityComment: "Needs immediate assistance", + sortWeight: 0, + startedAt: "2022-03-09T13:50:54.000+0000", + endedAt: null, + queue: { + uuid: "c187d78b-5c54-49bf-a0f8-b7fb6034d36d", + display: "Clinical Consultation", + name: "Clinical Consultation", + description: + "A queue for patients for a clincal consultation i.e. Doctor, Clinician", + service: { + display: "Clinical Consultation", + }, + }, + status: { + uuid: "aaec62b1-4b03-4166-ada7-230cb4b4aaaa", + display: "Waiting", + }, + patient: { + uuid: "53568469-f652-470d-95e8-13131914286b", + display: "10000JT - John Smith", + person: { + age: "27", + gender: "M", + }, + }, + priority: { + uuid: "b6a84ad0-c5e6-4a37-896e-5b7a0bccfd6c", + display: "Emergency", + }, + locationWaitingFor: null, + providerWaitingFor: null, + }, + }, +]; diff --git a/__mocks__/address.mock.ts b/__mocks__/address.mock.ts new file mode 100644 index 0000000..6d9e36b --- /dev/null +++ b/__mocks__/address.mock.ts @@ -0,0 +1,8 @@ +export const mockAddress = { + postalCode: "12345", + address1: "123 Main St", + cityVillage: "City", + stateProvince: "State", + country: "Country", + preferred: true, +}; diff --git a/__mocks__/appointments.mock.ts b/__mocks__/appointments.mock.ts new file mode 100644 index 0000000..7e8a93c --- /dev/null +++ b/__mocks__/appointments.mock.ts @@ -0,0 +1,422 @@ +export const mockAppointmentsData = { + data: [ + { + uuid: "7cd38a6d-377e-491b-8284-b04cf8b8c6d8", + appointmentNumber: "0000", + patient: { + identifier: "100GEJ", + name: "John Wilson", + uuid: "8673ee4f-e2ab-4077-ba55-4980f408773e", + gender: "M", + age: 35, + birthdate: "1986-04-03T00:00:00.000+0000", + phoneNumber: "0700000000", + }, + service: { + appointmentServiceId: 1, + name: "Outpatient", + description: null, + speciality: {}, + startTime: "", + endTime: "", + maxAppointmentsLimit: null, + durationMins: null, + location: {}, + uuid: "e2ec9cf0-ec38-4d2b-af6c-59c82fa30b90", + initialAppointmentStatus: "Scheduled", + creatorName: null, + }, + serviceType: { + display: "HIV Clinic", + uuid: "53d58ff1-0c45-4e2e-9bd2-9cc826cb46e1", + duration: 15, + }, + provider: { + uuid: "f9badd80-ab76-11e2-9e96-0800200c9a66", + person: { + uuid: "24252571-dd5a-11e6-9d9c-0242ac150002", + display: "Dr James Cook", + }, + }, + location: { + name: "HIV Clinic", + uuid: "2131aff8-2e2a-480a-b7ab-4ac53250262b", + }, + startDateTime: 1630326900000, + endDateTime: 1630327200000, + appointmentKind: "WalkIn", + status: "Scheduled", + comments: "Walk in appointments", + additionalInfo: null, + providers: [ + { + uuid: "24252571-dd5a-11e6-9d9c-0242ac150002", + display: "Dr James Cook", + }, + ], + recurring: false, + }, + { + uuid: "e10ce4e3-0e91-4b97-bc6c-9b5068e58428", + appointmentNumber: "0000", + patient: { + identifier: "100GEJ", + name: "Neil Amstrong", + uuid: "8673ee4f-e2ab-4077-ba55-4980f408773e", + gender: "M", + age: 35, + birthdate: "1986-04-03T00:00:00.000+0000", + phoneNumber: "0700000002", + }, + service: { + appointmentServiceId: 1, + name: "Outpatient", + description: null, + speciality: {}, + startTime: "", + endTime: "", + maxAppointmentsLimit: null, + durationMins: null, + location: {}, + uuid: "e2ec9cf0-ec38-4d2b-af6c-59c82fa30b90", + initialAppointmentStatus: "Scheduled", + creatorName: null, + }, + serviceType: { + display: "HIV Clinic", + uuid: "53d58ff1-0c45-4e2e-9bd2-9cc826cb46e1", + duration: 15, + }, + provider: { + uuid: "f9badd80-ab76-11e2-9e96-0800200c9a66", + person: { + uuid: "24252571-dd5a-11e6-9d9c-0242ac150002", + display: "Dr James Cook", + }, + }, + location: { + name: "HIV Clinic", + uuid: "2131aff8-2e2a-480a-b7ab-4ac53250262b", + }, + startDateTime: 1631278200000, + endDateTime: 1631278560000, + appointmentKind: "WalkIn", + status: "Scheduled", + comments: "Some additional notes", + additionalInfo: null, + providers: [ + { + uuid: "24252571-dd5a-11e6-9d9c-0242ac150002", + display: "Dr James Cook", + }, + ], + recurring: false, + }, + { + uuid: "cdb0676f-0805-4c3e-bfef-7757a005e892", + appointmentNumber: "0000", + patient: { + identifier: "100GEJ", + name: "Charles Babbage", + uuid: "8673ee4f-e2ab-4077-ba55-4980f408773e", + gender: "M", + age: 35, + birthdate: "1986-04-03T00:00:00.000+0000", + phoneNumber: "0700000001", + }, + service: { + appointmentServiceId: 1, + name: "Outpatient", + description: null, + speciality: {}, + startTime: "", + endTime: "", + maxAppointmentsLimit: null, + durationMins: null, + location: {}, + uuid: "e2ec9cf0-ec38-4d2b-af6c-59c82fa30b90", + initialAppointmentStatus: "Scheduled", + creatorName: null, + }, + serviceType: { + display: "TB Clinic", + uuid: "4a228e52-0bfe-11ed-861d-0242ac120002", + duration: 15, + }, + provider: null, + location: { + name: "TB Clinic", + uuid: "2131aff8-2e2a-480a-b7ab-4ac53250262b", + }, + startDateTime: 1631537400000, + endDateTime: 1631537760000, + appointmentKind: "WalkIn", + status: "Scheduled", + comments: "Some additional notes", + additionalInfo: null, + providers: [], + recurring: false, + }, + { + uuid: "66565d8b-4849-4b7c-966a-554d6073f80c", + appointmentNumber: "0000", + patient: { + identifier: "100GEJ", + name: "Elon Musketeer", + uuid: "8673ee4f-e2ab-4077-ba55-4980f408773e", + gender: "M", + age: 35, + birthdate: "1986-04-03T00:00:00.000+0000", + phoneNumber: "0700000000", + }, + service: { + appointmentServiceId: 1, + name: "Outpatient", + description: null, + speciality: {}, + startTime: "", + endTime: "", + maxAppointmentsLimit: null, + durationMins: null, + location: {}, + uuid: "e2ec9cf0-ec38-4d2b-af6c-59c82fa30b90", + initialAppointmentStatus: "Scheduled", + creatorName: null, + }, + serviceType: { + display: "HIV Clinic", + uuid: "53d58ff1-0c45-4e2e-9bd2-9cc826cb46e1", + duration: 15, + }, + provider: null, + location: { + name: "HIV Clinic", + uuid: "2131aff8-2e2a-480a-b7ab-4ac53250262b", + }, + startDateTime: 1631605800000, + endDateTime: 1631606100000, + appointmentKind: "WalkIn", + status: "Scheduled", + comments: "Some additional notes", + additionalInfo: null, + providers: [], + recurring: false, + }, + { + uuid: "45dcc19d-dd14-4a07-95c6-afa264972a34", + appointmentNumber: "0000", + patient: { + identifier: "100GEJ", + name: "Hopkins Derrick", + uuid: "8673ee4f-e2ab-4077-ba55-4980f408773e", + gender: "M", + age: 35, + birthdate: "1986-04-03T00:00:00.000+0000", + phoneNumber: "0700000000", + }, + service: { + appointmentServiceId: 1, + name: "Outpatient", + description: null, + speciality: {}, + startTime: "", + endTime: "", + maxAppointmentsLimit: null, + durationMins: null, + location: {}, + uuid: "e2ec9cf0-ec38-4d2b-af6c-59c82fa30b90", + initialAppointmentStatus: "Scheduled", + creatorName: null, + }, + serviceType: { + display: "TB Clinic", + uuid: "4a228e52-0bfe-11ed-861d-0242ac120002", + duration: 15, + }, + provider: { + uuid: "f9badd80-ab76-11e2-9e96-0800200c9a66", + person: { + uuid: "24252571-dd5a-11e6-9d9c-0242ac150002", + display: "Dr James Cook", + }, + }, + location: { + name: "TB Clinic", + uuid: "2131aff8-2e2a-480a-b7ab-4ac53250262b", + }, + startDateTime: 1631623800000, + endDateTime: 1631624160000, + appointmentKind: "WalkIn", + status: "Scheduled", + comments: "Some additional notes", + additionalInfo: null, + providers: [], + recurring: false, + }, + { + uuid: "fa4657ad-db46-487d-8e2e-a3858c906ae6", + appointmentNumber: "0000", + patient: { + identifier: "100GEJ", + name: "Amos Strong", + uuid: "8673ee4f-e2ab-4077-ba55-4980f408773e", + gender: "M", + age: 35, + birthdate: "1986-04-03T00:00:00.000+0000", + phoneNumber: "0700000000", + }, + service: { + appointmentServiceId: 1, + name: "Outpatient", + description: null, + speciality: {}, + startTime: "", + endTime: "", + maxAppointmentsLimit: null, + durationMins: null, + location: {}, + uuid: "e2ec9cf0-ec38-4d2b-af6c-59c82fa30b90", + initialAppointmentStatus: "Scheduled", + creatorName: null, + }, + serviceType: { + display: "TB Clinic", + uuid: "4a228e52-0bfe-11ed-861d-0242ac120002", + duration: 15, + }, + provider: null, + location: { + name: "TB Clinic", + uuid: "2131aff8-2e2a-480a-b7ab-4ac53250262b", + }, + startDateTime: 1631712720000, + endDateTime: 1631713080000, + appointmentKind: "WalkIn", + status: "Scheduled", + comments: "Some value", + additionalInfo: null, + providers: [], + recurring: false, + }, + ], +}; + +export const mockMappedAppointmentsData = { + data: [ + { + id: "e10ce4e3-0e91-4b97-bc6c-9b5068e58428", + name: "John Wilson", + age: "45", + gender: "M", + phoneNumber: "0700123456", + dob: "1986-04-03T00:00:00.000+0000", + patientUuid: "8673ee4f-e2ab-4077-ba55-4980f408773e", + dateTime: " 30-Aug-2021, 03:35 PM", + serviceType: "HIV Clinic", + visitType: "HIV Clinic", + provider: "Dr James Cook", + location: "HIV Clinic", + comments: "Some additional notes", + appointmentNumber: "0001", + serviceUuid: "ce03060e-1cbe-11ed-861d-0242ac120002", + appointmentKind: "Scheduled", + status: "Cancelled", + }, + { + id: "cdb0676f-0805-4c3e-bfef-7757a005e892", + name: "Eric Test Ric", + age: "32", + gender: "M", + phoneNumber: "0700987654", + dob: "1986-04-03T00:00:00.000+0000", + patientUuid: "8673ee4f-e2ab-4077-ba55-4980f408773e", + dateTime: "10-Sept-2021, 03:50 PM", + serviceType: "TB Clinic", + visitType: "TB Clinic", + provider: "Dr James Cook", + location: "TB Clinic", + comments: "Some additional notes here", + appointmentNumber: "0001", + serviceUuid: "c674bc34-1cbe-11ed-861d-0242ac120002", + appointmentKind: "WalkIn", + status: "Scheduled", + }, + ], +}; + +export const mockAppointmentMetrics = { + totalAppointments: 16, + highestServiceLoad: "HIV Consultation", + isLoading: false, + error: null, +}; + +export const mockProvidersCount = { + totalProviders: 4, + isLoading: false, + error: null, +}; + +export const mockStartTime = { + startTime: "2022-10-03T00:00:00.000+0000", +}; + +export const mockPatient = { + identifier: "100GEJ", + name: "John Wilson", + uuid: "8673ee4f-e2ab-4077-ba55-4980f408773e", + gender: "M", + age: 35, + birthdate: "1986-04-03T00:00:00.000+0000", + phoneNumber: "0700000000", +}; + +export const mockAppointmentServices = [ + { + uuid: "176052c7-5fd4-4b33-89cc-7bae6848c65a", + display: "Clinical consultation", + }, + { uuid: "d80ff12a-06a7-11ed-b939-0242ac120002", display: "Triage" }, +]; + +export const mockAppointmentServiceTypes = { + data: [ + { + name: "HIV Clinic", + uuid: "7bff050e-9a83-4fdb-9212-1a4c2cee349b", + duration: 15, + }, + { + name: "Drug Dispense", + uuid: "0e960ced-d35e-43de-b87f-bc08ef3b6ec3", + duration: 15, + }, + ], +}; + +export const mockFrequency = { + data: [ + { uuid: "b127d100-bccb-4699-b30e-fe60a068ca46", display: "Daily" }, + { uuid: "dbc32cf4-e11d-4006-b398-c71204cfc0d0", display: "Weekly" }, + { uuid: "6e263ecd-173e-4ee1-a9a5-df1f8d5b274b", display: "Monthly" }, + ], +}; + +export const mockProviders = { + data: [ + { + uuid: "f9badd80-ab76-11e2-9e96-0800200c9a66", + person: { + uuid: "24252571-dd5a-11e6-9d9c-0242ac150002", + display: "Dr James Cook", + }, + }, + { + uuid: "3191eddf-5cc5-4fa4-94ef-dfc25e8d33e4", + person: { + uuid: "89af8ba6-2ec5-4d77-b3ed-7e9e02448e96", + display: "Dr Amstrong Neil", + }, + }, + ], +}; diff --git a/__mocks__/auto-generation-options.mock.ts b/__mocks__/auto-generation-options.mock.ts new file mode 100644 index 0000000..aab5e87 --- /dev/null +++ b/__mocks__/auto-generation-options.mock.ts @@ -0,0 +1,34 @@ +export const mockAutoGenerationOptionsResult = { + results: [ + { + uuid: "42ae5ce0-d64b-11ea-9064-5adc43bbdd24", + location: null, + source: { + uuid: "691eed12-c0f1-11e2-94be-8c13b969e334", + }, + manualEntryEnabled: false, + automaticGenerationEnabled: true, + resourceVersion: "1.8", + }, + { + uuid: "497b8b17-54ec-4726-87ec-3c4da8cdcaeb", + location: null, + source: { + uuid: "691eed12-c0f1-11e2-94be-8c13b969e334", + }, + manualEntryEnabled: true, + automaticGenerationEnabled: false, + resourceVersion: "1.8", + }, + { + uuid: "ed0529de-3530-4c49-921b-b4845a750b7e", + location: null, + source: { + uuid: "75df804e-03c1-4964-842b-4fec585839e7", + }, + manualEntryEnabled: true, + automaticGenerationEnabled: false, + resourceVersion: "1.8", + }, + ], +}; diff --git a/__mocks__/identifier-types.mock.ts b/__mocks__/identifier-types.mock.ts new file mode 100644 index 0000000..1e3dd92 --- /dev/null +++ b/__mocks__/identifier-types.mock.ts @@ -0,0 +1,76 @@ +export const mockedIdentifierTypes = [ + { + fieldName: "openMrsId", + format: null, + identifierSources: [ + { + uuid: "8549f706-7e85-4c1d-9424-217d50a2988b", + name: "Generator for OpenMRS ID", + description: "Generator for OpenMRS ID", + baseCharacterSet: "0123456789ACDEFGHJKLMNPRTUVWXY", + prefix: "", + autoGenerationOption: { + manualEntryEnabled: false, + automaticGenerationEnabled: true, + }, + }, + { + uuid: "01af8526-cea4-4175-aa90-340acb411771", + name: "Generator 2 for OpenMRS ID", + description: "Generator 2 for OpenMRS ID", + baseCharacterSet: "0123456789ACDEFGHJKLMNPRTUVWXY", + prefix: "", + autoGenerationOption: { + manualEntryEnabled: true, + automaticGenerationEnabled: true, + }, + }, + ], + isPrimary: true, + name: "OpenMRS ID", + required: true, + uniquenessBehavior: "UNIQUE", + uuid: "05a29f94-c0ed-11e2-94be-8c13b969e334", + autoGenerationSource: null, + }, + { + fieldName: "idCard", + format: null, + identifierSources: [], + isPrimary: false, + name: "ID Card", + required: false, + uniquenessBehavior: "UNIQUE", + uuid: "b4143563-16cd-4439-b288-f83d61670fc8", + }, + { + fieldName: "legacyId", + format: null, + identifierSources: [], + isPrimary: false, + name: "Legacy ID", + required: false, + uniquenessBehavior: null, + uuid: "22348099-3873-459e-a32e-d93b17eda533", + }, + { + fieldName: "oldIdentificationNumber", + format: "", + identifierSources: [], + isPrimary: false, + name: "Old Identification Number", + required: false, + uniquenessBehavior: null, + uuid: "8d79403a-c2cc-11de-8d13-0010c6dffd0f", + }, + { + fieldName: "openMrsIdentificationNumber", + format: "", + identifierSources: [], + isPrimary: false, + name: "OpenMRS Identification Number", + required: false, + uniquenessBehavior: null, + uuid: "8d793bee-c2cc-11de-8d13-0010c6dffd0f", + }, +]; diff --git a/__mocks__/identifiers.mock.ts b/__mocks__/identifiers.mock.ts new file mode 100644 index 0000000..5ca9de7 --- /dev/null +++ b/__mocks__/identifiers.mock.ts @@ -0,0 +1,27 @@ +export const openmrsID = { + name: "OpenMRS ID", + fieldName: "openMrsId", + required: true, + uuid: "05a29f94-c0ed-11e2-94be-8c13b969e334", + format: null, + isPrimary: true, + identifierSources: [ + { + uuid: "691eed12-c0f1-11e2-94be-8c13b969e334", + name: "Generator 1 for OpenMRS ID", + autoGenerationOption: { + manualEntryEnabled: false, + automaticGenerationEnabled: true, + }, + }, + { + uuid: "01af8526-cea4-4175-aa90-340acb411771", + name: "Generator 2 for OpenMRS ID", + autoGenerationOption: { + manualEntryEnabled: true, + automaticGenerationEnabled: true, + }, + }, + ], + autoGenerationSource: null, +}; diff --git a/__mocks__/index.ts b/__mocks__/index.ts new file mode 100644 index 0000000..52775a2 --- /dev/null +++ b/__mocks__/index.ts @@ -0,0 +1,15 @@ +export * from "./active-visits.mock"; +export * from "./address.mock"; +export * from "./appointments.mock"; +export * from "./auto-generation-options.mock"; +export * from "./identifier-types.mock"; +export * from "./identifiers.mock"; +export * from "./locations.mock"; +export * from "./metrics.mock"; +export * from "./patient-visits.mock"; +export * from "./patient-registration.mock"; +export * from "./queue-entry.mock"; +export * from "./queue-rooms.mock"; +export * from "./search.mock"; +export * from "./session.mock"; +export * from "./visits.mock"; diff --git a/__mocks__/locations.mock.ts b/__mocks__/locations.mock.ts new file mode 100644 index 0000000..fbd661b --- /dev/null +++ b/__mocks__/locations.mock.ts @@ -0,0 +1,16 @@ +export const mockLocations = { + data: { + results: [ + { + uuid: "some-uuid1", + name: "Mosoriot", + display: "Mosoriot", + }, + { + uuid: "b1a8b05e-3542-4037-bbd3-998ee9c40574", + display: "Inpatient Ward", + name: "Inpatient Ward", + }, + ], + }, +}; diff --git a/__mocks__/metrics.mock.ts b/__mocks__/metrics.mock.ts new file mode 100644 index 0000000..e700e33 --- /dev/null +++ b/__mocks__/metrics.mock.ts @@ -0,0 +1,43 @@ +export const mockMetrics = { + activeVisitsCount: 100, + averageMinutes: 28, + waitTime: { + queue: "Clinical Consultation", + averageWaitTime: 69.0, + }, +}; + +export const mockServices = [ + { + uuid: "176052c7-5fd4-4b33-89cc-7bae6848c65a", + display: "Clinical consultation", + }, + { uuid: "d80ff12a-06a7-11ed-b939-0242ac120002", display: "Triage" }, +]; + +export const mockServiceTypes = { + data: [ + { + display: "Clinical Consulltation", + uuid: "7bff050e-9a83-4fdb-9212-1a4c2cee349b", + }, + { + display: "Lab Testing", + uuid: "0e960ced-d35e-43de-b87f-bc08ef3b6ec3", + }, + { + display: "Triage", + uuid: "dbc32cf4-e11d-4006-b398-c71204cfc0d0", + }, + ], +}; + +export const mockStatus = [ + { uuid: "1b67146f-b23b-4492-94ef-6255c137a333", display: "Finished service" }, + { uuid: "5c6a7c68-d671-43a2-887a-0fa65c5f20c8", display: "Waiting" }, +]; + +export const mockPriorities = [ + { uuid: "1ec10eee-330e-49b0-8ceb-c76af8e84c9e", display: "Emergency" }, + { uuid: "bf92b64c-2fa4-44aa-93d0-40f17ec1b433", display: "Not urgent" }, +]; diff --git a/__mocks__/patient-registration.mock.ts b/__mocks__/patient-registration.mock.ts new file mode 100644 index 0000000..6dce31a --- /dev/null +++ b/__mocks__/patient-registration.mock.ts @@ -0,0 +1,108 @@ +export const mockedAddressTemplate = { + displayName: null, + codeName: "default", + country: null, + lines: [ + [ + { + isToken: "IS_NOT_ADDR_TOKEN", + displayText: "", + }, + { + isToken: "IS_ADDR_TOKEN", + displayText: "Village", + codeName: "cityVillage", + displaySize: "40", + }, + { + isToken: "IS_NOT_ADDR_TOKEN", + displayText: ", ", + }, + { + isToken: "IS_ADDR_TOKEN", + displayText: "Commune", + codeName: "address1", + displaySize: "40", + }, + ], + [ + { + isToken: "IS_NOT_ADDR_TOKEN", + displayText: "", + }, + { + isToken: "IS_ADDR_TOKEN", + displayText: "District", + codeName: "countyDistrict", + displaySize: "40", + }, + { + isToken: "IS_NOT_ADDR_TOKEN", + displayText: ", ", + }, + { + isToken: "IS_ADDR_TOKEN", + displayText: "Province", + codeName: "stateProvince", + displaySize: "40", + }, + ], + [ + { + isToken: "IS_NOT_ADDR_TOKEN", + displayText: "", + }, + { + isToken: "IS_ADDR_TOKEN", + displayText: "Country", + codeName: "country", + displaySize: "40", + }, + ], + ], + lineByLineFormat: [ + "cityVillage, address1", + "countyDistrict, stateProvince", + "country", + ], + nameMappings: { + country: "Country", + countyDistrict: "District", + address1: "Commune", + stateProvince: "Province", + cityVillage: "Village", + }, + sizeMappings: { + country: "40", + countyDistrict: "40", + address1: "40", + stateProvince: "40", + cityVillage: "40", + }, + elementDefaults: { + country: "កម្ពុជា (Cambodia)", + }, + elementRegex: null, + elementRegexFormats: null, + requiredElements: null, +}; + +export const mockedOrderedFields = [ + "country", + "stateProvince", + "cityVillage", + "postalCode", + "address1", + "address2", +]; + +export const mockedAddressOptions = [ + "Cambodia > Banteay Meanchey > Mongkol Borei > Banteay Neang > Ou Thum", + "Cambodia > Battambang > Banan > Ta Kream > Andoung Neang", + "Cambodia > Banteay Meanchey > Mongkol Borei > Banteay Neang > Phnum", + "Cambodia > Kampong Cham > Chamkar Leu > Ta Prok > Neang Laeung", + "Cambodia > Battambang > Thma Koul > Ta Meun > Tumneab", + "Cambodia > Banteay Meanchey > Mongkol Borei > Banteay Neang > Banteay Neang", + "Cambodia > Banteay Meanchey > Phnum Srok > Spean Sraeng > Mukh Chhneang", + "Cambodia > Kampong Cham > Cheung Prey > Phdau Chum > Cham Neang", +]; diff --git a/__mocks__/patient-visits.mock.ts b/__mocks__/patient-visits.mock.ts new file mode 100644 index 0000000..2a5b9d7 --- /dev/null +++ b/__mocks__/patient-visits.mock.ts @@ -0,0 +1,114 @@ +export const mockPatientsVisits = { + recentVisits: [ + { + uuid: "6baa7963-68ea-497e-b258-6fb82382bd07", + appointmentNumber: "0000", + patient: { + identifier: "M4T4J", + name: "WILLIAM LAGAT", + uuid: "cd34e972-a325-4371-9b2a-57ba82d628ab", + }, + service: { + appointmentServiceId: 1, + name: "Cardiology Consultation 1", + description: null, + speciality: {}, + startTime: "09:00:00", + endTime: "17:30:00", + maxAppointmentsLimit: 30, + durationMins: 30, + location: { + name: "10 Engineer VCT", + uuid: "673d1efd-6a96-47c3-a332-54c30d9f0c29", + }, + uuid: "4d93cd63-ad6e-4546-a790-90892f736e73", + color: "#00ff00", + initialAppointmentStatus: null, + creatorName: null, + }, + serviceType: null, + provider: null, + location: { + name: "10 Engineer VCT", + uuid: "673d1efd-6a96-47c3-a332-54c30d9f0c29", + }, + startDateTime: 1659970578000, + endDateTime: 1659970578000, + appointmentKind: "WalkIn", + status: "Scheduled", + comments: null, + additionalInfo: null, + teleconsultation: null, + providers: [ + { + uuid: "bef0f4f0-1bf2-4c0a-8191-f898bc107e3d", + comments: "available", + response: "ACCEPTED", + name: "Super User", + }, + ], + voided: false, + extensions: { + patientEmailDefined: false, + }, + teleconsultationLink: null, + recurring: false, + }, + ], + futureVisits: [ + { + uuid: "6baa7963-68ea-497e-b258-6fb82382bd07", + appointmentNumber: "0000", + patient: { + identifier: "M4T4J", + name: "WILLIAM LAGAT", + uuid: "cd34e972-a325-4371-9b2a-57ba82d628ab", + }, + service: { + appointmentServiceId: 1, + name: "Cardiology Consultation 2", + description: null, + speciality: {}, + startTime: "09:00:00", + endTime: "17:30:00", + maxAppointmentsLimit: 30, + durationMins: 30, + location: { + name: "10 Engineer VCT", + uuid: "673d1efd-6a96-47c3-a332-54c30d9f0c29", + }, + uuid: "4d93cd63-ad6e-4546-a790-90892f736e73", + color: "#00ff00", + initialAppointmentStatus: null, + creatorName: null, + }, + serviceType: null, + provider: null, + location: { + name: "10 Engineer VCT", + uuid: "673d1efd-6a96-47c3-a332-54c30d9f0c29", + }, + startDateTime: 1659970578000, + endDateTime: 1659970578000, + appointmentKind: "WalkIn", + status: "Scheduled", + comments: null, + additionalInfo: null, + teleconsultation: null, + providers: [ + { + uuid: "bef0f4f0-1bf2-4c0a-8191-f898bc107e3d", + comments: "available", + response: "ACCEPTED", + name: "Super User", + }, + ], + voided: false, + extensions: { + patientEmailDefined: false, + }, + teleconsultationLink: null, + recurring: false, + }, + ], +}; diff --git a/__mocks__/queue-entry.mock.ts b/__mocks__/queue-entry.mock.ts new file mode 100644 index 0000000..ede1624 --- /dev/null +++ b/__mocks__/queue-entry.mock.ts @@ -0,0 +1,70 @@ +export const mockQueueEntry = { + id: "8824d1e4-8513-4a78-bcec-37173f417f18", + encounters: [], + name: "Brian Johnson", + patientAge: "32", + patientSex: "F", + patientDob: "13 — Apr — 2020", + patientUuid: "eecfaf7b-a768-42af-9db8-4bbfe3644901", + priority: "Not Urgent", + priorityComment: "", + priorityUuid: "f4620bfa-3625-4883-bd3f-84c2cce14470", + queueEntryUuid: "8824d1e4-8513-4a78-bcec-37173f417f18", + queueUuid: "cbe2cd1d-1884-40fd-92ed-ee357783b450", + service: "Clinical consultation", + status: "Waiting", + statusUuid: "51ae5e4d-b72b-4912-bf31-a17efb690aeb", + visitStartDateTime: "2020-02-01T00:00:00.000+0000", + visitType: "Facility Visit", + visitUuid: "b90d8438-a0db-4318-a57e-cda773b21433", + waitTime: "12362", + queueLocation: "Triage", + sortWeight: "0", +}; + +export const mockMappedQueueEntries = { + data: [ + { + id: "fa1e98f1-f002-4174-9e55-34d60951e710", + encounters: [], + name: "Eric Test Ric", + patientAge: "32", + patientSex: "F", + patientDob: "13 — Apr — 2020", + patientUuid: "eecfaf7b-a768-42af-9db8-4bbfe3644901", + priority: "Not Urgent", + priorityComment: "", + priorityUuid: "f4620bfa-3625-4883-bd3f-84c2cce14470", + queueEntryUuid: "712289ab-32c0-430f-87b6-d9c1e4e4686e", + queueUuid: "cbe2cd1d-1884-40fd-92ed-ee357783b450", + service: "Triage", + status: "Waiting", + statusUuid: "51ae5e4d-b72b-4912-bf31-a17efb690aeb", + visitStartDateTime: "2020-02-01T00:00:00.000+0000", + visitType: "Facility Visit", + visitUuid: "b90d8438-a0db-4318-a57e-cda773b21433", + waitTime: "12362", + }, + { + id: "2f85d611-5bb9-4bca-b6f8-661517df86c9", + encounters: [], + name: "John Smith", + patientAge: "32", + patientSex: "F", + patientDob: "13 — Apr — 2020", + patientUuid: "eecfaf7b-a768-42af-9db8-4bbfe3644901", + priority: "Emergency", + priorityComment: "", + priorityUuid: "f4620bfa-3625-4883-bd3f-84c2cce14470", + queueEntryUuid: "5f017eb0-b035-4acd-b284-da45f5067502", + queueUuid: "cbe2cd1d-1884-40fd-92ed-ee357783b450", + service: "Clinical consultation", + status: "In Service", + statusUuid: "51ae5e4d-b72b-4912-bf31-a17efb690aeb", + visitStartDateTime: "2020-02-01T00:00:00.000+0000", + visitType: "Facility Visit", + visitUuid: "b90d8438-a0db-4318-a57e-cda773b21433", + waitTime: "12362", + }, + ], +}; diff --git a/__mocks__/queue-rooms.mock.ts b/__mocks__/queue-rooms.mock.ts new file mode 100644 index 0000000..0316793 --- /dev/null +++ b/__mocks__/queue-rooms.mock.ts @@ -0,0 +1,18 @@ +export const mockQueueRooms = { + data: { + results: [ + { + uuid: "bf9610d6-47e4-472b-a077-1ffdf5bd6963", + display: "Room 1", + name: "Room 1", + description: "Room 1", + }, + { + uuid: "0145ad2e-ec7f-466c-82be-a722984a40eb", + display: "Room 8", + name: "Room 8", + description: "Room 8", + }, + ], + }, +}; diff --git a/__mocks__/search.mock.ts b/__mocks__/search.mock.ts new file mode 100644 index 0000000..16b6690 --- /dev/null +++ b/__mocks__/search.mock.ts @@ -0,0 +1,35 @@ +export const mockSearchResults = { + data: { + results: [ + { + display: "10000F1 - Eric Test Ric", + identifiers: [ + { + display: "OpenMRS ID = 10000F1", + identifier: "10000F1", + voided: false, + }, + ], + patientId: 20, + patientIdentifier: { + identifier: "10000F1", + }, + person: { + gender: "M", + age: 35, + birthdate: "1986-04-03T00:00:00.000+0000", + birthdateEstimated: false, + dead: false, + deathDate: null, + display: "Eric Test Ric", + personName: { + givenName: "Eric", + middleName: "Test", + familyName: "Ric", + }, + }, + uuid: "cc75ad73-c24b-499c-8db9-a7ef4fc0b36d", + }, + ], + }, +}; diff --git a/__mocks__/session.mock.ts b/__mocks__/session.mock.ts new file mode 100644 index 0000000..1347811 --- /dev/null +++ b/__mocks__/session.mock.ts @@ -0,0 +1,116 @@ +export const mockSession = { + data: { + authenticated: true, + locale: "en_GB", + currentProvider: { + uuid: "b1a8b05e-3542-4037-bbd3-998ee9c4057z", + display: "Test User", + person: { + uuid: "ddd5fa89-48a6-432e-abb8-0d11b4be7e4f", + display: "Test User", + }, + identifier: "UNKNOWN", + attributes: [], + }, + sessionLocation: { + uuid: "b1a8b05e-3542-4037-bbd3-998ee9c40574", + display: "Inpatient Ward", + name: "Inpatient Ward", + description: null, + address1: null, + address2: null, + cityVillage: null, + stateProvince: null, + country: null, + postalCode: null, + latitude: null, + longitude: null, + countyDistrict: null, + address3: null, + address4: null, + address5: null, + address6: null, + tags: [ + { + uuid: "8d4626ca-7abd-42ad-be48-56767bbcf272", + display: "Transfer Location", + }, + { + uuid: "b8bbf83e-645f-451f-8efe-a0db56f09676", + display: "Login Location", + }, + { + uuid: "1c783dca-fd54-4ea8-a0fc-2875374e9cb6", + display: "Admission Location", + }, + ], + parentLocation: { + uuid: "aff27d58-a15c-49a6-9beb-d30dcfc0c66e", + display: "Amani Hospital", + }, + childLocations: [], + retired: false, + attributes: [], + address7: null, + address8: null, + address9: null, + address10: null, + address11: null, + address12: null, + address13: null, + address14: null, + address15: null, + links: [], + }, + user: { + uuid: "45ce6c2e-dd5a-11e6-9d9c-0242ac150002", + display: "admin", + username: "", + systemId: "admin", + locale: "en", + allowedLocales: ["en", "en-GB", "es", "fr", "he", "km"], + userProperties: { + loginAttempts: "0", + }, + person: { + uuid: "0775e6b7-f439-40e5-87a3-2bd11f3b9ee5", + display: "Test User", + links: [], + }, + links: [], + privileges: [ + { + uuid: "62431c71-5244-11ea-a771-0242ac160002", + display: "Manage Appointment Services", + }, + { + uuid: "6206682c-5244-11ea-a771-0242ac160002", + display: "Manage Appointments", + }, + { + uuid: "6280ff58-5244-11ea-a771-0242ac160002", + display: "Manage Appointment Specialities", + }, + ], + roles: [ + { + uuid: "8d94f852-c2cc-11de-8d13-0010c6dffd0f", + display: "System Developer", + links: [], + }, + { + uuid: "62c195c5-5244-11ea-a771-0242ac160002", + display: "Bahmni Role", + links: [], + }, + { + uuid: "8d94f280-c2cc-11de-8d13-0010c6dffd0f", + display: "Provider", + links: [], + }, + ], + retired: false, + }, + sessionId: "39570a50-abb2-4f44-b146-a288ec35064b", + }, +}; diff --git a/__mocks__/visits.mock.ts b/__mocks__/visits.mock.ts new file mode 100644 index 0000000..ac9ef3a --- /dev/null +++ b/__mocks__/visits.mock.ts @@ -0,0 +1,42 @@ +export const mockPastVisit = { + data: { + results: [ + { + uuid: "b80b8fba-ab62-11ec-b909-0242ac120002", + patient: { + uuid: "b80b8b8c-ab62-11ec-b909-0242ac120002", + display: "113RGH - Test Test Test", + }, + visitType: { + uuid: "e7786ac0-ab62-11ec-b909-0242ac120002", + display: "Facility Visit", + }, + location: { + uuid: "e7786d9a-ab62-11ec-b909-0242ac120002", + display: "Location Test", + }, + startDatetime: "2022-03-23T10:29:00.000+0000", + stopDatetime: "2022-03-24T10:29:00.000+0000", + encounters: [], + }, + ], + }, +}; + +export const mockVisitTypes = [ + { + uuid: "some-uuid1", + name: "Outpatient Visit", + display: "Outpatient Visit", + }, + { + uuid: "some-uuid2", + name: "HIV Return Visit", + display: "HIV Return Visit", + }, + { + uuid: "some-uuid3", + name: "Diabetes Clinic Visit", + display: "Diabetes Clinic Visit", + }, +]; diff --git a/jest.config.js b/jest.config.js index e69de29..b1561b3 100644 --- a/jest.config.js +++ b/jest.config.js @@ -0,0 +1,45 @@ +/** + * @returns {Promise} + */ + +const path = require("path"); + +module.exports = { + transform: { + "^.+\\.(j|t)sx?$": "@swc/jest", + }, + transformIgnorePatterns: ["/node_modules/(?!@openmrs)"], + moduleDirectories: ["node_modules", "__mocks__", "tools", __dirname], + moduleNameMapper: { + "\\.(s?css)$": "identity-obj-proxy", + "@openmrs/esm-framework": "@openmrs/esm-framework/mock", + "^dexie$": require.resolve("dexie"), + "^lodash-es/(.*)$": "lodash/$1", + "^react-i18next$": path.resolve(__dirname, "react-i18next.js"), + "^uuid$": path.resolve( + __dirname, + "node_modules", + "uuid", + "dist", + "index.js", + ), + }, + collectCoverageFrom: [ + "**/src/**/*.component.tsx", + "!**/node_modules/**", + "!**/vendor/**", + "!**/src/**/*.test.*", + "!**/src/declarations.d.ts", + "!**/e2e/**", + ], + setupFilesAfterEnv: [path.resolve(__dirname, "tools", "setup-tests.ts")], + testPathIgnorePatterns: [path.resolve(__dirname, "e2e")], + testEnvironment: "jsdom", + testEnvironmentOptions: { + url: "http://localhost/", + }, + fakeTimers: { + enableGlobally: true, + legacyFakeTimers: true, + }, +}; diff --git a/package.json b/package.json index e8729bc..3317b24 100644 --- a/package.json +++ b/package.json @@ -1,9 +1,9 @@ { - "name": "@ugandaemr/esm-template-app", + "name": "@ugandaemr/esm-appointments-app", "version": "1.0.0", "license": "MPL-2.0", "description": "A template for creating frontend modules for UgandaEMR", - "browser": "dist/esm-ugandaemr-template-app.js", + "browser": "dist/esm-ugandaemr-appointments-app.js", "main": "src/index.ts", "source": true, "scripts": { @@ -18,7 +18,7 @@ "verify": "turbo lint typescript coverage", "coverage": "yarn test --coverage", "prepare": "husky install", - "extract-translations": "i18next 'src/**/*.component.tsx' --config ./i18next-parser.config.js" + "extract-translations": "i18next 'src/**/*.component.tsx' --config ./tools/i18next-parser.config.js" }, "husky": { "hooks": { @@ -34,71 +34,85 @@ ], "repository": { "type": "git", - "url": "git+https://github.com/mets-programme/esm-ugandaemr-template-app.git" + "url": "git+https://github.com/mets-programme/esm-appointments-app.git" }, - "homepage": "https://github.com/mets-programme/esm-ugandaemr-template-app#readme", + "homepage": "https://github.com/mets-programme/esm-appointments-appp#readme", "publishConfig": { "access": "public" }, "bugs": { - "url": "https://github.com/mets-programme/esm-ugandaemr-template-app/issues" + "url": "https://github.com/mets-programme/esm-appointments-app/issues" }, "dependencies": { "@carbon/react": "^1.33.1", + "formik": "^2.4.5", "lodash-es": "^4.17.21", - "react-image-annotate": "^1.8.0" + "react-image-annotate": "^1.8.0", + "xlsx": "0.18.5" }, "peerDependencies": { "@openmrs/esm-framework": "*", "dayjs": "1.x", "react": "18.x", "react-i18next": "11.x", - "react-router-dom": "6.x", - "rxjs": "6.x" + "react-router-dom": "6.x" }, "devDependencies": { + "@babel/core": "^7.11.6", + "@carbon/react": "^1.12.0", "@openmrs/esm-framework": "next", - "@openmrs/esm-styleguide": "next", - "@swc/cli": "^0.1.62", - "@swc/core": "^1.3.68", - "@swc/jest": "^0.2.26", - "@testing-library/dom": "^8.20.1", - "@testing-library/jest-dom": "^5.16.5", - "@testing-library/react": "^13.4.0", - "@testing-library/user-event": "^14.4.3", - "@types/jest": "^28.1.8", - "@types/react": "^18.2.14", - "@types/react-dom": "^18.2.6", - "@types/react-router": "^5.1.20", + "@playwright/test": "1.40.1", + "@swc/core": "^1.2.165", + "@swc/jest": "^0.2.29", + "@testing-library/dom": "^9.3.3", + "@testing-library/jest-dom": "^6.1.5", + "@testing-library/react": "^14.1.2", + "@testing-library/user-event": "^14.5.1", + "@types/jest": "^29.5.11", + "@types/lodash-es": "^4.17.3", + "@types/react": "^18.0.9", + "@types/react-dom": "^18.0.5", "@types/react-router-dom": "^5.3.3", - "@types/webpack-env": "^1.18.1", - "@typescript-eslint/eslint-plugin": "^5.61.0", - "@typescript-eslint/parser": "^5.61.0", + "@types/uuid": "^8.3.0", + "@types/webpack-env": "^1.16.0", + "@types/yup": "^0.29.11", + "@typescript-eslint/eslint-plugin": "^6.13.2", + "@typescript-eslint/parser": "^6.13.2", + "babel-preset-minify": "^0.5.1", + "concurrently": "^5.3.0", + "cross-env": "^7.0.3", "css-loader": "^6.8.1", - "eslint": "^8.44.0", - "eslint-config-prettier": "^8.8.0", - "eslint-config-ts-react-important-stuff": "^3.0.0", - "eslint-plugin-prettier": "^4.2.1", - "husky": "^8.0.0", - "i18next": "^23.2.8", - "i18next-parser": "^8.0.0", + "dayjs": "^1.8.36", + "dotenv": "^16.0.3", + "eslint": "^8.55.0", + "eslint-config-prettier": "^9.1.0", + "eslint-plugin-prettier": "^5.0.1", + "eslint-plugin-react-hooks": "^4.6.0", + "husky": "^8.0.3", + "i18next": "^21.10.0", + "i18next-parser": "^6.6.0", "identity-obj-proxy": "^3.0.0", - "jest": "^28.1.3", - "jest-cli": "^28.1.3", - "jest-environment-jsdom": "^28.1.3", + "jest": "^29.7.0", + "jest-cli": "^29.7.0", + "jest-environment-jsdom": "^29.7.0", "openmrs": "next", - "prettier": "^2.8.8", - "pretty-quick": "^3.1.3", - "react": "^18.2.0", - "react-dom": "^18.2.0", + "prettier": "^3.1.1", + "react": "^18.1.0", + "react-dom": "^18.1.0", "react-i18next": "^11.18.6", - "react-router-dom": "^6.14.1", - "rxjs": "^6.6.7", + "react-router-dom": "^6.3.0", + "rxjs": "^7.8.1", + "sass": "^1.29.0", + "sass-loader": "^10.1.0", "swc-loader": "^0.2.3", - "turbo": "^1.10.7", - "typescript": "^4.9.5", - "webpack": "^5.88.1", - "webpack-cli": "^5.1.4" + "turbo": "^1.6.3", + "typedoc": "^0.22.15", + "typescript": "^4.0.3", + "webpack": "^5.74.0", + "webpack-cli": "^4.10.0" + }, + "lint-staged": { + "*.{js,jsx,ts,tsx}": "eslint --cache --fix" }, "packageManager": "yarn@3.6.2" } diff --git a/src/__mocks__/react-i18next.js b/react-i18next.js similarity index 80% rename from src/__mocks__/react-i18next.js rename to react-i18next.js index ce95868..b8fd672 100644 --- a/src/__mocks__/react-i18next.js +++ b/react-i18next.js @@ -1,6 +1,5 @@ -/** At present, this entire mock is boilerplate. */ -import "react"; -import "react-i18next"; +const React = require("react"); +const reactI18next = require("react-i18next"); const hasChildren = (node) => node && (node.children || (node.props && node.props.children)); @@ -27,7 +26,7 @@ const renderNodes = (reactNodes) => { if (typeof child === "object" && !isElement) { return Object.keys(child).reduce( (str, childKey) => `${str}${child[childKey]}`, - "" + "", ); } @@ -36,7 +35,14 @@ const renderNodes = (reactNodes) => { }; const useMock = [(k) => k, {}]; -useMock.t = (k, o) => (o && o.defaultValue) || (typeof o === "string" ? o : k); +useMock.t = (key, defaultValue, options = {}) => { + let translatedString = defaultValue; + Object.keys(options).forEach((key) => { + translatedString = defaultValue.replace(`{{${key}}}`, `${options[key]}`); + }); + + return translatedString ?? key; +}; useMock.i18n = {}; module.exports = { diff --git a/src/admin/appointment-services/appointment-services-hook.ts b/src/admin/appointment-services/appointment-services-hook.ts new file mode 100644 index 0000000..f3577c4 --- /dev/null +++ b/src/admin/appointment-services/appointment-services-hook.ts @@ -0,0 +1,32 @@ +import { openmrsFetch } from "@openmrs/esm-framework"; +import { amPm } from "../../helpers"; +import { type AppointmentService } from "../../types"; + +const appointmentServiceInitialValue: AppointmentService = { + appointmentServiceId: 0, + creatorName: "", + description: "", + durationMins: "", + endTime: "", + initialAppointmentStatus: "", + location: { uuid: "", display: "" }, + maxAppointmentsLimit: 0, + name: "", + startTime: "", + uuid: "", + color: "", + startTimeTimeFormat: new Date().getHours() >= 12 ? "PM" : "AM", + endTimeTimeFormat: new Date().getHours() >= 12 ? "PM" : "AM", +}; + +const addNewAppointmentService = (payload) => { + return openmrsFetch("/ws/rest/v1/appointmentService", { + method: "POST", + body: payload, + headers: { "Content-Type": "application/json" }, + }); +}; + +export const useAppointmentServices = () => { + return { appointmentServiceInitialValue, addNewAppointmentService }; +}; diff --git a/src/admin/appointment-services/appointment-services-validation.ts b/src/admin/appointment-services/appointment-services-validation.ts new file mode 100644 index 0000000..5b8dc5b --- /dev/null +++ b/src/admin/appointment-services/appointment-services-validation.ts @@ -0,0 +1,19 @@ +import * as Yup from "yup"; +import { OpenmrsResource } from "@openmrs/esm-framework"; + +export const validationSchema = Yup.object({ + description: Yup.string().optional(), + durationMins: Yup.number().required("durationMinsRequired"), + endTime: Yup.string().required("endTimeRequired"), + initialAppointmentStatus: Yup.string().optional(), + location: Yup.object({ uuid: Yup.string(), display: Yup.string() }).required( + "locationRequired", + ), + maxAppointmentsLimit: Yup.number().required("maxAppointmentLimitRequired"), + name: Yup.string().required("appointmentServiceNameRequired"), + specialityUuid: Yup.string().optional(), + startTime: Yup.string().required("startTimeRequired"), + color: Yup.string().required("colorRequired"), + startTimeTimeFormat: Yup.string().required("startTimeFormatRequired"), + endTimeTimeFormat: Yup.string().required("endTimeFormatRequired"), +}); diff --git a/src/admin/appointment-services/appointment-services.component.tsx b/src/admin/appointment-services/appointment-services.component.tsx new file mode 100644 index 0000000..da2974d --- /dev/null +++ b/src/admin/appointment-services/appointment-services.component.tsx @@ -0,0 +1,228 @@ +import React from "react"; +import { + Button, + ButtonSet, + Dropdown, + Layer, + SelectItem, + TextInput, + TimePicker, + TimePickerSelect, +} from "@carbon/react"; +import { useTranslation } from "react-i18next"; +import { Form, Formik, type FormikHelpers } from "formik"; +import { validationSchema } from "./appointment-services-validation"; +import { useAppointmentServices } from "./appointment-services-hook"; +import { + showNotification, + showToast, + useLocations, +} from "@openmrs/esm-framework"; +import type { AppointmentService } from "../../types"; +import { closeOverlay } from "../../hooks/useOverlay"; +import styles from "./appointment-services.scss"; +import { appointmentLocationTagName } from "../../constants"; + +interface AppointmentServicesProps {} + +const AppointmentServices: React.FC = () => { + const { t } = useTranslation(); + const { appointmentServiceInitialValue, addNewAppointmentService } = + useAppointmentServices(); + + const locations = useLocations(); + const handleSubmit = async ( + values: AppointmentService, + helpers: FormikHelpers, + ) => { + const payload = { + name: values.name, + startTime: values.startTime.concat(":00"), + endTime: values.endTime.concat(":00"), + durationMins: values.durationMins, + color: values.color, + locationUuid: values.location.uuid, + }; + addNewAppointmentService(payload).then( + ({ status }) => { + if (status === 200) { + showToast({ + critical: true, + kind: "success", + description: t( + "appointmentServiceCreate", + "Appointment service created successfully", + ), + title: t("appointmentService", "Appointment service"), + }); + closeOverlay(); + } + }, + (error) => { + showNotification({ + title: t( + "errorCreatingAppointmentService", + "Error creating appointment service", + ), + kind: "error", + critical: true, + description: error?.message, + }); + }, + ); + }; + return ( + + {(props) => { + return ( +
+

+ {t("createAppointmentService", "Create appointment service")} +

+ + + + + + + + + + + + + + + + + + + + + + + + + + + (item ? item.display : "")} + selectedItem={props.values.location} + invalid={!!(props.touched && props.errors.location?.uuid)} + name="location" + onChange={({ selectedItem }) => + props.setValues({ ...props.values, location: selectedItem }) + } + /> + + + + + + + + + + +
+ ); + }} +
+ ); +}; + +export default AppointmentServices; diff --git a/src/admin/appointment-services/appointment-services.scss b/src/admin/appointment-services/appointment-services.scss new file mode 100644 index 0000000..16e9775 --- /dev/null +++ b/src/admin/appointment-services/appointment-services.scss @@ -0,0 +1,25 @@ +@use '@carbon/styles/scss/spacing'; + +.appointmentServiceContainer { + margin: spacing.$spacing-05 spacing.$spacing-05; + + & > div { + padding: spacing.$spacing-03 0; + } +} + +.buttonSet { + position: absolute; + bottom: 0; + left: 0; + right: 0; + margin: 0 spacing.$spacing-05; +} + +.button { + height: spacing.$spacing-10; + display: flex; + align-content: flex-start; + align-items: baseline; + width: auto; +} diff --git a/src/appointments-calendar/appointments-calendar-view-view.scss b/src/appointments-calendar/appointments-calendar-view-view.scss new file mode 100644 index 0000000..240e0ad --- /dev/null +++ b/src/appointments-calendar/appointments-calendar-view-view.scss @@ -0,0 +1,24 @@ +@use '@carbon/styles/scss/type'; +@use '@carbon/styles/scss/spacing'; +@use '@carbon/colors'; +@import '~@openmrs/esm-styleguide/src/vars'; + +.wrapper { + position: relative; +} +.monthlyCalendar { + display: grid; + grid-template-columns: repeat(7, minmax(0px, 1fr)); + grid-template-rows: repeat(6, minmax(0px, 1fr)); +} + +.calendarViewContainer { + margin: spacing.$spacing-05; +} +.backgroundColor { + margin: 1px 0 0; + transition: width 0.24s ease-in-out; + position: relative; + min-height: calc(100vh - 80px); + background-color: colors.$white; +} diff --git a/src/appointments-calendar/appointments-calendar-view.component.tsx b/src/appointments-calendar/appointments-calendar-view.component.tsx new file mode 100644 index 0000000..8e7f873 --- /dev/null +++ b/src/appointments-calendar/appointments-calendar-view.component.tsx @@ -0,0 +1,38 @@ +import React, { useState } from "react"; +import dayjs from "dayjs"; +import { useTranslation } from "react-i18next"; +import { useAppointmentsCalendar } from "../hooks/useAppointmentsCalendar"; +import type { CalendarType } from "../types"; +import AppointmentsHeader from "../appointments-header/appointments-header.component"; +import CalendarHeader from "./header/calendar-header.component"; +import CalendarView from "./calendar-view.component"; +import { useAppointmentDate } from "../helpers"; + +const AppointmentsCalendarView: React.FC = () => { + const { t } = useTranslation(); + const [calendarView, setCalendarView] = useState("monthly"); + const { currentAppointmentDate, setCurrentAppointmentDate } = + useAppointmentDate(); + const { calendarEvents } = useAppointmentsCalendar( + dayjs(currentAppointmentDate).toISOString(), + calendarView, + ); + + return ( +
+ + + +
+ ); +}; + +export default AppointmentsCalendarView; diff --git a/src/appointments-calendar/appointments-calendar-view.test.tsx b/src/appointments-calendar/appointments-calendar-view.test.tsx new file mode 100644 index 0000000..848e638 --- /dev/null +++ b/src/appointments-calendar/appointments-calendar-view.test.tsx @@ -0,0 +1,38 @@ +import React from "react"; +import { render, screen } from "@testing-library/react"; +import { openmrsFetch } from "@openmrs/esm-framework"; +import userEvent from "@testing-library/user-event"; +import AppointmentsCalendarView from "./appointments-calendar-view.component"; + +const mockedOpenmrsFetch = openmrsFetch as jest.Mock; + +jest.mock("../hooks/useAppointments"), + () => ({ + useDailyAppointments: jest.fn(), + useAppointmentsByDurationPeriod: jest.fn(), + }); + +describe("Appointment calendar view", () => { + it("renders appointments in calendar view from appointments list", async () => { + const user = userEvent.setup(); + + renderAppointmentsCalendarListView(); + + expect(screen.getByText(/monthly/i)).toBeInTheDocument(); + + const expectedTableRows = [ + /John Wilson 30-Aug-2021 03:35 03:35 Dr James Cook Outpatient Walk in appointments/, + /Neil Amstrong 10-Sept-2021 03:50 03:50 Dr James Cook Outpatient Some additional notes/, + ]; + + expectedTableRows.forEach((row) => { + expect( + screen.queryByRole("row", { name: new RegExp(row, "i") }), + ).not.toBeInTheDocument(); + }); + }); +}); + +function renderAppointmentsCalendarListView() { + render(); +} diff --git a/src/appointments-calendar/calendar-view.component.tsx b/src/appointments-calendar/calendar-view.component.tsx new file mode 100644 index 0000000..51d3c5c --- /dev/null +++ b/src/appointments-calendar/calendar-view.component.tsx @@ -0,0 +1,46 @@ +import React from "react"; +import DailyCalendarView from "./daily/daily-calendar-view.component"; +import MonthlyCalendarView from "./monthly/monthly-calendar-view.component"; +import WeeklyCalendarView from "./weekly/weekly-calendar-view.component"; +import { type CalendarType } from "../types"; + +const CalendarView: React.FC<{ + calendarView: CalendarType; + events: any; + currentDate: any; + setCurrentDate: any; +}> = ({ calendarView, events, currentDate, setCurrentDate }) => { + switch (calendarView) { + case "monthly": + return ( + + ); + case "weekly": + return ( + + ); + case "daily": + return ( + + ); + default: + return null; + } +}; + +export default CalendarView; diff --git a/src/appointments-calendar/daily/daily-calendar-view.component.tsx b/src/appointments-calendar/daily/daily-calendar-view.component.tsx new file mode 100644 index 0000000..c95d8ee --- /dev/null +++ b/src/appointments-calendar/daily/daily-calendar-view.component.tsx @@ -0,0 +1,68 @@ +import React from "react"; +import dayjs, { type Dayjs } from "dayjs"; +import isBetween from "dayjs/plugin/isBetween"; +import styles from "./daily-calendar.scss"; +import { type CalendarType } from "../../types"; +import { dailyHours } from "../../helpers"; +import DailyHeader from "./daily-header.component"; +import DailyWorkloadView from "./daily-view-workload.component"; +dayjs.extend(isBetween); + +interface DailyCalendarViewProps { + type: CalendarType; + events: { + appointmentDate: string; + service: Array; + [key: string]: any; + }[]; + currentDate: Dayjs; + setCurrentDate: (date) => void; +} + +const DailyCalendarView: React.FC = ({ + type = "daily", + events, + currentDate, + setCurrentDate, +}) => { + return ( +
+ +
+ <> +

+ <> +

+ +

+ +

+

+ {dailyHours(currentDate).map((dateTime, i) => ( + <> +

+
+ {dateTime.minute(0).format("h a")} +
+
+
+ + ))} +

+ +
+
+ ); +}; + +export default DailyCalendarView; diff --git a/src/appointments-calendar/daily/daily-calendar.scss b/src/appointments-calendar/daily/daily-calendar.scss new file mode 100644 index 0000000..0d4c746 --- /dev/null +++ b/src/appointments-calendar/daily/daily-calendar.scss @@ -0,0 +1,71 @@ +@use '@carbon/styles/scss/type'; +@use '@carbon/colors'; +@use '@carbon/layout'; +@import '~@openmrs/esm-styleguide/src/vars'; +.container { + width: 100%; + padding-left: 50px; + margin-right: auto; + margin-left: auto; +} +.daily-calendar { + display: grid; + grid-template-rows: repeat(1, minmax(0px, 1fr)); + border-right: 1px solid colors.$gray-20; +} +.daily-calendar-all { + display: grid; + margin-top: 2px; + height: auto; + grid-template-columns: 60px repeat(0, minmax(0px, 1fr)); + grid-template-rows: repeat(24, minmax(0px, 1fr)); +} +.cellComponent { + display: flex; +} +.empty-cell { + position: relative; + border-right: 1px solid colors.$gray-20; + min-height: 40px; + width: 100%; + &:nth-child(-n + 24) { + // border-top: 1px solid colors.$gray-20; + border-bottom: 1px solid colors.$gray-20; + } +} +.daily-cell { + position: relative; + border-right: 1px solid colors.$gray-20; + border-left: 1px solid colors.$gray-20; + min-height: 40px; + width: 60px; + color: colors.$black; + text-align: center; + + &:nth-child(-n + 24) { + // border-top: 1px solid colors.$gray-20; + border-bottom: 1px solid colors.$gray-20; + } + + &:first-child { + border-top: 1px solid colors.$gray-20; + } +} + +// Overriding styles for RTL support +html[dir='rtl'] { + .container { + padding-left: unset; + padding-right: 50px; + } + + .daily-calendar { + border-right: unset; + border-left: 1px solid colors.$gray-20; + } + + .empty-cell { + border-right: unset; + border-left: 1px solid colors.$gray-20; + } +} diff --git a/src/appointments-calendar/daily/daily-header.component.tsx b/src/appointments-calendar/daily/daily-header.component.tsx new file mode 100644 index 0000000..0ad0152 --- /dev/null +++ b/src/appointments-calendar/daily/daily-header.component.tsx @@ -0,0 +1,67 @@ +import { Button } from "@carbon/react"; +import dayjs, { type Dayjs } from "dayjs"; +import React from "react"; +import { useTranslation } from "react-i18next"; +import { type CalendarType } from "../../types"; +import styles from "./daily-header.scss"; + +const Format = { + monthly: "month", + weekly: "week", + daily: "day", +} as const; +const dateFormat = "MMMM DD, YYYY"; +const yearFormat = "YYYY"; +const now = dayjs(); +function DailyHeader({ + type, + currentDate, + setCurrentDate, + events, +}: { + type: CalendarType; + currentDate: Dayjs; + setCurrentDate: (date: Dayjs) => void; + events: { + appointmentDate: string; + service: Array; + [key: string]: any; + }[]; +}) { + const { t } = useTranslation(); + return ( + <> +
+ +

+ {type === "daily" + ? currentDate.format(dateFormat) + : `${currentDate.startOf("week").format(dateFormat)} - ${currentDate + .endOf("week") + .format(dateFormat)} , ${currentDate.format(yearFormat)}`} +

+ +
+
+
+
+ {" "} + {type === "daily" ? currentDate.format("dddd") : ""} +
+
+ + ); +} +export default DailyHeader; diff --git a/src/appointments-calendar/daily/daily-header.scss b/src/appointments-calendar/daily/daily-header.scss new file mode 100644 index 0000000..74d592d --- /dev/null +++ b/src/appointments-calendar/daily/daily-header.scss @@ -0,0 +1,71 @@ +@use '@carbon/colors'; +@use '@carbon/layout'; +@use '@carbon/type'; +@import '~@openmrs/esm-styleguide/src/vars'; + +.container { + display: flex; + justify-content: space-between; + align-items: center; + width: 100%; + height: layout.$spacing-10; +} +.workLoadCard { + display: flex; + color: colors.$black; + padding-left: 60px; +} +.tileContainer { + height: layout.$spacing-10; + display: flex; + justify-content: center; + width: 100%; + align-items: center; + border: 1px solid colors.$gray-20; + + & > span { + font-size: layout.$spacing-05; + color: colors.$gray-70; + } +} +.bold { + font-weight: bold; +} + +.tileHeader { + display: flex; + justify-content: space-between; + padding-bottom: layout.$spacing-05; +} + +.headerLabel { + @include type.type-style('heading-compact-01'); + color: $text-02; +} +.dayDate { + display: inline-flex; + flex-direction: column; +} +@media only screen and (min-width: 40px) { + .dayDate { + display: flex; + flex-direction: row; + } +} +@media only screen and (min-width: 40px) { + .dayDateColumn1 { + border: 1px solid colors.$gray-20; + width: 60px; + height: 40px; + display: inline-flex; + padding: 0; + } +} +.dayDateColumn2 { + width: 100%; + border: 1px solid colors.$gray-20; + border-left: none; + text-align: center; + font-weight: bold; + padding-top: 10px; +} diff --git a/src/appointments-calendar/daily/daily-view-workload.component.tsx b/src/appointments-calendar/daily/daily-view-workload.component.tsx new file mode 100755 index 0000000..21ee3a5 --- /dev/null +++ b/src/appointments-calendar/daily/daily-view-workload.component.tsx @@ -0,0 +1,96 @@ +import dayjs, { type Dayjs } from "dayjs"; +import classNames from "classnames"; +import styles from "./daily-workload-module.scss"; +import React from "react"; +import { navigate, useLayoutType } from "@openmrs/esm-framework"; +import { useTranslation } from "react-i18next"; +import { spaBasePath } from "../../constants"; +import { isSameMonth } from "../../helpers"; +import { type CalendarType } from "../../types"; + +interface WeeklyCellProps { + type: CalendarType; + dateTime: Dayjs; + currentDate: Dayjs; + events: Array; +} + +const DailyWorkloadView: React.FC = ({ + type, + dateTime, + currentDate, + events, +}) => { + const layout = useLayoutType(); + const currentData = events?.find( + (event) => + dayjs(event.appointmentDate).format("YYYY-MM-DD") === + dayjs(dateTime).format("YYYY-MM-DD"), + ); + const colorCoding = { HIV: "red", "Lab testing": "purple", Refill: "blue" }; + const { t } = useTranslation(); + + return ( +
+ {type === "daily" ? ( + <> +
+ All Day + {currentData?.service && ( +
+ {currentData?.service.map(({ serviceName, count, i }) => ( +
+ navigate({ + to: `${spaBasePath}/appointments/list/${dateTime}/${serviceName}`, + }) + } + > + {serviceName} + {count} +
+ ))} +
+ navigate({ + to: `${spaBasePath}/appointments/list/${dateTime}/Total`, + }) + } + > + {t("total", "Total")} + + {currentData?.service.reduce( + (sum, currentValue) => sum + currentValue?.count ?? 0, + 0, + )} + +
+
+ )} +
+ + ) : null} +
+ ); +}; +export default DailyWorkloadView; diff --git a/src/appointments-calendar/daily/daily-view-workload.test.tsx b/src/appointments-calendar/daily/daily-view-workload.test.tsx new file mode 100644 index 0000000..6a69339 --- /dev/null +++ b/src/appointments-calendar/daily/daily-view-workload.test.tsx @@ -0,0 +1,58 @@ +import React from "react"; +import dayjs from "dayjs"; +import userEvent from "@testing-library/user-event"; +import { render, screen } from "@testing-library/react"; +import { navigate } from "@openmrs/esm-framework"; +import { spaBasePath } from "../../constants"; +import { type CalendarType } from "../../types"; +import DailyWorkloadView from "./daily-view-workload.component"; + +jest.mock("@openmrs/esm-framework", () => ({ + // ...jest.requireActual("@openmrs/esm-framework"), + // navigate: jest.fn(), + // useLayoutType: jest.fn(), +})); + +describe("DailyWorkloadView Component", () => { + const mockData = { + type: "daily" as CalendarType, + dateTime: dayjs("2023-08-18"), + currentDate: dayjs("2023-08-18"), + events: [ + { + appointmentDate: "2023-08-18", + service: [ + { serviceName: "HIV", count: 2 }, + { serviceName: "Lab testing", count: 3 }, + ], + }, + ], + }; + + it('renders properly when type is "daily"', () => { + // render(); + + // expect(screen.getByText("All Day")).toBeInTheDocument(); + // expect(screen.getByText("HIV")).toBeInTheDocument(); + // expect(screen.getByText("Lab testing")).toBeInTheDocument(); + }); + + it("navigates when a service area is clicked", async () => { + const user = userEvent.setup(); + + // render(); + + // await user.click(screen.getByText("HIV")); + + // expect(navigate).toHaveBeenCalledWith({ + // to: `${spaBasePath}/appointments/list/Fri, 18 Aug 2023 00:00:00 GMT/HIV`, + // }); + }); + + it("calculates and displays the total count correctly", () => { + // render(); + + // expect(screen.getByText("Total")).toBeInTheDocument(); + // expect(screen.getByText("5")).toBeInTheDocument(); + }); +}); diff --git a/src/appointments-calendar/daily/daily-workload-module.scss b/src/appointments-calendar/daily/daily-workload-module.scss new file mode 100644 index 0000000..28a2465 --- /dev/null +++ b/src/appointments-calendar/daily/daily-workload-module.scss @@ -0,0 +1,152 @@ +@use '@carbon/styles/scss/type'; +@use '@carbon/colors'; +@use '@carbon/layout'; +@import '~@openmrs/esm-styleguide/src/vars'; + +.monthly-cell { + border-left: 1px solid colors.$gray-20; + border-bottom: 1px solid colors.$gray-20; + background: colors.$white; + color: colors.$black; + padding: layout.$spacing-01; + text-align: right; + @include type.type-style('body-compact-02'); + + &:nth-child(-n + 7) { + border-top: 1px solid colors.$gray-20; + } + + &:nth-child(7n) { + border-right: 1px solid colors.$gray-20; + } + + &-disabled { + border-left: 1px solid colors.$gray-20; + border-bottom: 1px solid colors.$gray-20; + } + &-disabled:last-child { + border-right: 1px solid colors.$gray-20; + } +} +.allDayComponent { + display: flex; + border: 1px solid colors.$gray-20; +} + +.weekly-cell { + position: relative; + border-right: 1px solid colors.$gray-20; + border-bottom: 1px solid colors.$gray-20; + min-height: 120px; + color: colors.$black; + + .week-time { + display: none; + } + + &:nth-child(-n + 8) { + border-top: 1px solid colors.$gray-20; + .allDay { + display: none; + } + } + + &:first-child { + border-top: none; + .allDay { + display: flex; + border-right: 1px solid colors.$gray-20; + padding-top: 50px; + padding-left: 5px; + padding-bottom: 50px; + align-items: center; + justify-content: center; + } + } + + &:nth-child(n) { + border-bottom: none; + + .week-time { + white-space: nowrap; + display: inline; + color: colors.$black; + + position: absolute; + top: -11px; + height: 22px; + width: 30px; + } + } +} +.serviceArea { + display: grid; + padding: 0.125rem; + column-gap: 2rem; + grid-template-columns: 4fr 1fr; + justify-content: flex-start; + text-align: left; + margin: 0.125rem; + @include type.type-style('label-02'); + color: #020f1b; + cursor: pointer; + + & > span:first-child { + text-align: left; + text-overflow: ellipsis; + } +} + +.serviceArea:hover { + opacity: 0.8; + border: 1px solid grey; +} + +.currentData { + margin: 5px; + width: 100%; +} + +.smallDesktop { + height: 100px; +} + +.largeDesktop { + height: 150px; +} + +.red { + background-color: colors.$red-70; + color: colors.$white; + font-weight: bold; +} + +.purple { + background-color: colors.$yellow-70; + color: colors.$white; + font-weight: bold; +} + +.blue { + background-color: colors.$blue-70; + color: colors.$white; + font-weight: bold; +} + +.green { + background-color: colors.$green-70; + color: colors.$white; + font-weight: bold; +} + +// Overriding styles for RTL support +html[dir='rtl'] { + .weekly-cell { + &:first-child { + .allDay { + border-left: 1px solid colors.$gray-20; + border-right: none; + } + } + } +} diff --git a/src/appointments-calendar/daily/days-of-week.component.tsx b/src/appointments-calendar/daily/days-of-week.component.tsx new file mode 100644 index 0000000..8249285 --- /dev/null +++ b/src/appointments-calendar/daily/days-of-week.component.tsx @@ -0,0 +1,16 @@ +import dayjs from "dayjs"; +import React from "react"; +import styles from "./days-of-week.scss"; + +interface DaysOfWeekProp { + dayOfWeek: string; +} +const DaysOfWeekCard: React.FC = ({ dayOfWeek }) => { + const isToday = dayjs(new Date()).format("ddd").toUpperCase() === dayOfWeek; + return ( +
+ {dayOfWeek} +
+ ); +}; +export default DaysOfWeekCard; diff --git a/src/appointments-calendar/daily/days-of-week.scss b/src/appointments-calendar/daily/days-of-week.scss new file mode 100644 index 0000000..bef0979 --- /dev/null +++ b/src/appointments-calendar/daily/days-of-week.scss @@ -0,0 +1,33 @@ +@use '@carbon/colors'; +@use '@carbon/layout'; +@use '@carbon/type'; +@import '~@openmrs/esm-styleguide/src/vars'; + +.tileContainer { + height: layout.$spacing-10; + display: flex; + justify-content: center; + width: 100%; + align-items: center; + border: 1px solid colors.$gray-20; + + & > span { + font-size: layout.$spacing-05; + color: colors.$gray-70; + } +} + +.bold { + font-weight: bold; +} + +.tileHeader { + display: flex; + justify-content: space-between; + padding-bottom: layout.$spacing-05; +} + +.headerLabel { + @include type.type-style('heading-compact-01'); + color: $text-02; +} diff --git a/src/appointments-calendar/header/calendar-header.component.tsx b/src/appointments-calendar/header/calendar-header.component.tsx new file mode 100644 index 0000000..5883c56 --- /dev/null +++ b/src/appointments-calendar/header/calendar-header.component.tsx @@ -0,0 +1,90 @@ +import React from "react"; +import { useTranslation } from "react-i18next"; +import { Button, ContentSwitcher, Switch } from "@carbon/react"; +import { ArrowRight, Filter, ArrowLeft } from "@carbon/react/icons"; +import { navigate } from "@openmrs/esm-framework"; +import { spaBasePath } from "../../constants"; +import type { CalendarType } from "../../types"; +import styles from "./calendar-header.scss"; + +enum CalendarView { + Daily = "daily", + Weekly = "weekly", + Monthly = "monthly", +} + +interface CalendarHeaderProps { + onChangeView: (view: CalendarView) => void; + calendarView: CalendarType; +} + +const CalendarHeader: React.FC = ({ + onChangeView, + calendarView, +}) => { + const { t } = useTranslation(); + const backButtonOnClick = () => { + navigate({ to: `${spaBasePath}/appointments` }); + }; + const addNewClinicDayOnClick = () => { + // add new clinic day functionality + }; + const filterOnClick = () => { + // filter functionality + }; + + const calendarViewObject = { + daily: CalendarView.Daily, + weekly: CalendarView.Weekly, + monthly: CalendarView.Monthly, + }; + + return ( +
+
+
+ +
+

{t("calendar", "Calendar")}

+ +
+
+ + onChangeView(name as CalendarView)} + > + + + + +
+
+ ); +}; + +export default CalendarHeader; diff --git a/src/appointments-calendar/header/calendar-header.scss b/src/appointments-calendar/header/calendar-header.scss new file mode 100644 index 0000000..d9d8994 --- /dev/null +++ b/src/appointments-calendar/header/calendar-header.scss @@ -0,0 +1,55 @@ +@use '@carbon/styles/scss/type'; +@use '@carbon/colors'; +@use '@carbon/styles/scss/spacing'; + +.titleContainer { + display: flex; + justify-content: space-between; + align-items: center; + margin-left: 1rem; + + & > p { + @include type.type-style('heading-02'); + } + + & > button { + color: colors.$blue-60; + @include type.type-style('label-02'); + } +} + +.titleContent { + display: flex; + justify-content: space-between; + align-items: center; +} + +.backButton { + align-items: center; + display: flex; + justify-content: flex-start; + padding: 0; + @include type.type-style('body-compact-01'); + + button { + display: flex; + padding: 0; + + svg { + order: 1; + margin-right: 0.5rem; + margin-left: 0rem !important; + } + + span { + order: 2; + } + } +} + +// Overriding styles for RTL support +html[dir='rtl'] { + .titleContent { + margin-left: 1rem; + } +} diff --git a/src/appointments-calendar/monthly/monthly-calendar-view.component.tsx b/src/appointments-calendar/monthly/monthly-calendar-view.component.tsx new file mode 100644 index 0000000..3371cc1 --- /dev/null +++ b/src/appointments-calendar/monthly/monthly-calendar-view.component.tsx @@ -0,0 +1,55 @@ +import React from "react"; +import dayjs, { type Dayjs } from "dayjs"; +import isBetween from "dayjs/plugin/isBetween"; +import { type CalendarType } from "../../types"; +import { monthDays } from "../../helpers/functions"; +import MonthlyViewWorkload from "./monthly-view-workload.component"; +import MonthlyHeader from "./monthly-header.module"; +import styles from "../appointments-calendar-view-view.scss"; + +dayjs.extend(isBetween); + +interface MonthlyCalendarViewProps { + type: CalendarType; + events: { + appointmentDate: string; + service: Array; + [key: string]: any; + }[]; + currentDate: Dayjs; + setCurrentDate: (date) => void; +} + +const MonthlyCalendarView: React.FC = ({ + type, + events, + currentDate, + setCurrentDate, +}) => { + return ( +
+ +
+ {type === "monthly" ? ( +
+ {monthDays(currentDate).map((dateTime, i) => ( + + ))} +
+ ) : null} +
+
+ ); +}; + +export default MonthlyCalendarView; diff --git a/src/appointments-calendar/monthly/monthly-header.module.scss b/src/appointments-calendar/monthly/monthly-header.module.scss new file mode 100644 index 0000000..49f7d7b --- /dev/null +++ b/src/appointments-calendar/monthly/monthly-header.module.scss @@ -0,0 +1,14 @@ +@use '@carbon/colors'; +@use '@carbon/layout'; + +.container { + display: flex; + justify-content: space-between; + align-items: center; + width: 100%; + height: layout.$spacing-10; +} +.workLoadCard { + display: flex; + color: colors.$black; +} diff --git a/src/appointments-calendar/monthly/monthly-header.module.tsx b/src/appointments-calendar/monthly/monthly-header.module.tsx new file mode 100644 index 0000000..119c36d --- /dev/null +++ b/src/appointments-calendar/monthly/monthly-header.module.tsx @@ -0,0 +1,56 @@ +import React from "react"; +import { type Dayjs } from "dayjs"; +import styles from "./monthly-header.module.scss"; +import { Button } from "@carbon/react"; +import { useTranslation } from "react-i18next"; +import { type CalendarType } from "../../types"; +import DaysOfWeekCard from "../daily/days-of-week.component"; + +const Format = { + monthly: "month", + weekly: "week", + daily: "day", +} as const; + +const monthFormat = "MMMM, YYYY"; +const daysInWeek = ["SUN", "MON", "TUE", "WED", "THUR", "FRI", "SAT"]; +function MonthlyHeader({ + type, + currentDate, + setCurrentDate, +}: { + type: CalendarType; + currentDate: Dayjs; + setCurrentDate: (date: Dayjs) => void; +}) { + const { t } = useTranslation(); + + return ( + <> +
+ + {currentDate.format(monthFormat)} + + +
+
+ {daysInWeek?.map((day, i) => ( + + ))} +
+ + ); +} +export default MonthlyHeader; diff --git a/src/appointments-calendar/monthly/monthly-view-workload.component.tsx b/src/appointments-calendar/monthly/monthly-view-workload.component.tsx new file mode 100644 index 0000000..c1b5cbc --- /dev/null +++ b/src/appointments-calendar/monthly/monthly-view-workload.component.tsx @@ -0,0 +1,82 @@ +import React from "react"; +import classNames from "classnames"; +import dayjs from "dayjs"; +import { useTranslation } from "react-i18next"; +import { navigate, useLayoutType } from "@openmrs/esm-framework"; +import { isSameMonth } from "../../helpers"; +import { spaBasePath } from "../../constants"; +import styles from "./monthly-view-workload.scss"; + +const colorCoding = { HIV: "red", "Lab testing": "purple", Refill: "blue" }; + +const MonthlyWorkload = ({ type, dateTime, currentDate, events }) => { + const layout = useLayoutType(); + const { t } = useTranslation(); + + const currentData = events?.find( + (event) => + dayjs(event.appointmentDate)?.format("YYYY-MM-DD") === + dayjs(dateTime)?.format("YYYY-MM-DD"), + ); + + const serviceAreaOnClick = (serviceName) => { + navigate({ + to: `${spaBasePath}/appointments/list/${dateTime}/${serviceName}`, + }); + }; + + return ( +
+ {type === "monthly" && isSameMonth(dateTime, currentDate) && ( +

+ {dateTime.format("D")} + {currentData?.service && ( +

+ {currentData.service.map(({ serviceName, count, i }) => ( +
serviceAreaOnClick(serviceName)} + className={styles.serviceArea} + > + {serviceName} + {count} +
+ ))} +
serviceAreaOnClick("Total")} + className={classNames(styles.serviceArea, styles.green)} + > + {t("total", "Total")} + + {currentData?.service.reduce( + (sum, { count = 0 }) => sum + count, + 0, + )} + +
+
+ )} +

+ )} +
+ ); +}; + +export default MonthlyWorkload; diff --git a/src/appointments-calendar/monthly/monthly-view-workload.scss b/src/appointments-calendar/monthly/monthly-view-workload.scss new file mode 100644 index 0000000..6224009 --- /dev/null +++ b/src/appointments-calendar/monthly/monthly-view-workload.scss @@ -0,0 +1,148 @@ +@use '@carbon/styles/scss/type'; +@use '@carbon/colors'; +@use '@carbon/layout'; +@import '~@openmrs/esm-styleguide/src/vars'; + +.monthly-cell { + border-left: 1px solid colors.$gray-20; + border-bottom: 1px solid colors.$gray-20; + background: colors.$white; + color: colors.$black; + padding: layout.$spacing-01; + text-align: right; + @include type.type-style('body-compact-02'); + + &:nth-child(-n + 7) { + border-top: 1px solid colors.$gray-20; + } + + &:nth-child(7n) { + border-right: 1px solid colors.$gray-20; + } + + &-disabled { + border-left: 1px solid colors.$gray-20; + border-bottom: 1px solid colors.$gray-20; + } + &-disabled:last-child { + border-right: 1px solid colors.$gray-20; + } +} +.identifiers { + @include type.type-style('body-compact-02'); + color: $ui-04; + display: list-item; + + span:not(:first-child) { + margin: 0rem 0.75rem; + } +} +.identifierTag { + display: flex; + align-items: center; +} +.weekly-cell { + position: relative; + border-right: 1px solid colors.$gray-20; + border-bottom: 1px solid colors.$gray-20; + min-height: 120px; + color: colors.$white; + + .week-time { + display: none; + } + + &:nth-child(-n + 8) { + border-top: 1px solid colors.$gray-20; + } + + &:first-child { + border-top: none; + } + + &:nth-child(8n + 1) { + border-bottom: none; + + .week-time { + white-space: nowrap; + display: inline; + color: colors.$white; + position: absolute; + top: -11px; + height: layout.$spacing-06; + width: layout.$spacing-07; + } + } +} + +.currentData { + margin: 2px; + padding-top: 2px; + background-color: colors.$gray-10; + + & > span { + @include type.type-style('label-02'); + } +} + +.serviceArea { + padding: 0.0125rem; + display: grid; + grid-template-columns: 4fr 1fr; + justify-content: flex-start; + text-align: left; + margin: 0.125rem; + @include type.type-style('label-01'); + color: #020f1b; + cursor: pointer; + + & > span:first-child { + text-align: left; + text-overflow: ellipsis; + } + + & > span:nth-child(2) { + text-align: right; + } +} + +.serviceArea:hover { + opacity: 0.8; + border: 1px solid grey; +} + +.smallDesktop { + height: 100px; +} + +.largeDesktop { + height: 150px; +} + +.red { + background-color: colors.$red-70; + color: colors.$white; + font-weight: bold; +} + +.purple { + background-color: colors.$yellow-70; + color: colors.$white; + font-weight: bold; +} + +.blue { + background-color: colors.$blue-70; + color: colors.$white; + font-weight: bold; +} + +.green { + background-color: colors.$green-70; + color: colors.$white; + font-weight: bold; +} + +.calendarDate { + @include type.type-style('heading-compact-01'); +} diff --git a/src/appointments-calendar/patient-list/calenar-patient-list.scss b/src/appointments-calendar/patient-list/calenar-patient-list.scss new file mode 100644 index 0000000..b990513 --- /dev/null +++ b/src/appointments-calendar/patient-list/calenar-patient-list.scss @@ -0,0 +1,14 @@ +@use '@carbon/styles/scss/type'; +@use '@carbon/colors'; + +.container { + margin: 1rem; +} + +.header { + margin-bottom: 1rem; + & > h2 { + @include type.type-style('heading-02'); + color: colors.$gray-100; + } +} diff --git a/src/appointments-calendar/patient-list/calendar-patient-list.component.tsx b/src/appointments-calendar/patient-list/calendar-patient-list.component.tsx new file mode 100644 index 0000000..0929ced --- /dev/null +++ b/src/appointments-calendar/patient-list/calendar-patient-list.component.tsx @@ -0,0 +1,162 @@ +import dayjs from "dayjs"; +import React from "react"; +import { useTranslation } from "react-i18next"; +import { omrsDateFormat } from "../../constants"; +import { + DataTableSkeleton, + DataTable, + TableContainer, + Table, + TableHead, + TableRow, + TableHeader, + TableBody, + TableCell, + TableToolbar, + TableToolbarContent, + TableToolbarSearch, + Button, +} from "@carbon/react"; +import { + ExtensionSlot, + formatDate, + formatDatetime, +} from "@openmrs/esm-framework"; +import { downloadAppointmentsAsExcel } from "../../helpers/excel"; +import { Download } from "@carbon/react/icons"; +import styles from "./calenar-patient-list.scss"; +import { useAppointments } from "../../appointments/appointments-table.resource"; + +interface CalendarPatientListProps {} + +const CalendarPatientList: React.FC = () => { + const { t } = useTranslation(); + const currentPathName: string = decodeURI(window.location.pathname); + const serviceName = currentPathName.split("/")[7]; + const forDate = currentPathName.split("/")[6]; + + const { appointments, isLoading } = useAppointments( + "", + dayjs(new Date(forDate).setHours(0, 0, 0, 0)).format(omrsDateFormat), + ); + + const headers = [ + { + header: t("patientName", "Patient name"), + key: "name", + }, + { + header: t("identifier", "Identifier"), + key: "identifier", + }, + { + header: t("date&Time", "Date & time"), + key: "dateTime", + }, + { + header: t("serviceType", "Service Type"), + key: "serviceType", + }, + { + header: t("provider", "Provider"), + key: "provider", + }, + ]; + + const rowData = appointments + ?.filter( + ({ serviceType }) => + serviceName === "Total" || serviceName === serviceType, + ) + .map((appointment) => ({ + id: `${appointment.identifier}`, + ...appointment, + dateTime: formatDatetime(new Date(appointment.dateTime)), + })); + + if (isLoading) { + return ( + <> + + + ); + } + + return ( + <> + +
+
+

+ {serviceName === "Total" + ? "All Services" + : `${serviceName} ${t("list", "List")}`} +

+
+ + {({ + rows, + headers, + getHeaderProps, + getRowProps, + getBatchActionProps, + onInputChange, + }) => ( + + + + + + + + + + + {headers.map((header) => ( + + {header.header} + + ))} + + + + {rows.map((row) => ( + + {row.cells.map((cell) => ( + {cell.value} + ))} + + ))} + +
+
+ )} +
+
+ + ); +}; + +export default CalendarPatientList; diff --git a/src/appointments-calendar/weekly/weekly-calendar-view.component.tsx b/src/appointments-calendar/weekly/weekly-calendar-view.component.tsx new file mode 100644 index 0000000..18c6fad --- /dev/null +++ b/src/appointments-calendar/weekly/weekly-calendar-view.component.tsx @@ -0,0 +1,87 @@ +import React from "react"; +import dayjs, { type Dayjs } from "dayjs"; +import isBetween from "dayjs/plugin/isBetween"; +import styles from "./weekly-calendar.scss"; +import { type CalendarType } from "../../types"; +import { isSameMonth, weekAllDays, weekDays } from "../../helpers"; +import WeeklyWorkloadView from "./weekly-view-workload.component"; +import WeeklyHeader from "./weekly-header.component"; +dayjs.extend(isBetween); + +interface WeeklyCalendarViewProps { + type: CalendarType; + events: { + appointmentDate: string; + service: Array; + [key: string]: any; + }[]; + currentDate: Dayjs; + setCurrentDate: (date) => void; +} + +const WeeklyCalendarView: React.FC = ({ + type = "weekly", + events, + currentDate, + setCurrentDate, +}) => { + return ( +
+ +
+ <> +

+ {weekDays(currentDate).map((dateTime, i) => { + return ( + <> + + + ); + })} +

+

+ {weekAllDays(currentDate).map((dateTime, i) => ( + <> +

+ {type === "weekly" ? ( + <> + { + + {dateTime.minute(0).format("h a")} + + } + + ) : null} +
+ + ))} +

+ +
+
+ ); +}; + +export default WeeklyCalendarView; diff --git a/src/appointments-calendar/weekly/weekly-calendar.scss b/src/appointments-calendar/weekly/weekly-calendar.scss new file mode 100644 index 0000000..4d23da0 --- /dev/null +++ b/src/appointments-calendar/weekly/weekly-calendar.scss @@ -0,0 +1,67 @@ +@use '@carbon/styles/scss/type'; +@use '@carbon/colors'; +@use '@carbon/layout'; +@import '~@openmrs/esm-styleguide/src/vars'; +.container { + width: 100%; + padding-left: 50px; + margin-right: auto; + margin-left: auto; +} +.weekly-calendar { + display: grid; + grid-template-columns: 60px repeat(7, minmax(0px, 1fr)); + grid-template-rows: repeat(1, minmax(0px, 1fr)); +} +.weekly-calendar-all { + display: grid; + margin-top: 2px; + height: auto; + grid-template-columns: 60px repeat(7, minmax(0px, 1fr)); + grid-template-rows: repeat(26, minmax(0px, 1fr)); +} +.weekly-cell { + position: relative; + border-right: 1px solid colors.$gray-20; + border-bottom: 1px solid colors.$gray-20; + min-height: 40px; + color: colors.$black; + + .week-time { + display: none; + } + + &:nth-child(-n + 8) { + border-top: 1px solid colors.$gray-20; + } + + &:first-child { + border-top: none; + } + + &:nth-child(8n + 1) { + border-bottom: none; + + .week-time { + white-space: nowrap; + display: inline; + margin-top: 10px; + color: colors.$black; + position: absolute; + height: 22px; + } + } +} + +// Overriding styles for RTL support +html[dir='rtl'] { + .container { + padding-left: unset; + padding-right: 50px; + } + + .weekly-cell { + border-right: unset; + border-left: 1px solid colors.$gray-20; + } +} diff --git a/src/appointments-calendar/weekly/weekly-header.component.tsx b/src/appointments-calendar/weekly/weekly-header.component.tsx new file mode 100644 index 0000000..c9bdcdb --- /dev/null +++ b/src/appointments-calendar/weekly/weekly-header.component.tsx @@ -0,0 +1,77 @@ +import React from "react"; +import { Button } from "@carbon/react"; +import { type Dayjs } from "dayjs"; +import { useTranslation } from "react-i18next"; +import { weekDays } from "../../helpers"; +import type { CalendarType } from "../../types"; +import styles from "./weekly-header.scss"; + +const Format = { + monthly: "month", + weekly: "week", + daily: "day", +} as const; +const daysInWeek = ["SUN", "MON", "TUE", "WED", "THUR", "FRI", "SAT"]; +const dateFormat = "D MMM"; +const monthFormat = "MMMM, YYYY"; +const yearFormat = "YYYY"; +function WeeklyHeader({ + type, + currentDate, + setCurrentDate, + events, +}: { + type: CalendarType; + currentDate: Dayjs; + setCurrentDate: (date: Dayjs) => void; + events: { + appointmentDate: string; + service: Array; + [key: string]: any; + }[]; +}) { + const { t } = useTranslation(); + return ( + <> +
+ + + {type === "monthly" + ? currentDate.format(monthFormat) + : type === "daily" + ? currentDate.format(dateFormat) + : `${currentDate.startOf("week").format(dateFormat)} - ${currentDate + .endOf("week") + .format(dateFormat)} , ${currentDate.format(yearFormat)}`} + + +
+
+ {weekDays(currentDate).map((dateTime, i) => ( + + {i !== 0 && ( +
+ + {dateTime.format("dddd")} {dateTime.format("DD")} + +
+ )} +
+ ))} +
+ + ); +} +export default WeeklyHeader; diff --git a/src/appointments-calendar/weekly/weekly-header.scss b/src/appointments-calendar/weekly/weekly-header.scss new file mode 100644 index 0000000..d280cf4 --- /dev/null +++ b/src/appointments-calendar/weekly/weekly-header.scss @@ -0,0 +1,58 @@ +@use '@carbon/colors'; +@use '@carbon/layout'; +@use '@carbon/type'; +@import '~@openmrs/esm-styleguide/src/vars'; + +.container { + display: flex; + justify-content: space-between; + align-items: center; + width: 100%; + height: layout.$spacing-10; +} +.workLoadCard { + display: flex; + color: colors.$black; + padding-left: 60px; +} +.tileContainer { + height: layout.$spacing-10; + display: flex; + justify-content: center; + width: 100%; + align-items: center; + border: 1px solid colors.$gray-20; + + & > span { + font-size: layout.$spacing-05; + color: colors.$gray-70; + } +} +.bold { + font-weight: bold; +} + +.tileHeader { + display: flex; + justify-content: space-between; + padding-bottom: layout.$spacing-05; +} + +.headerLabel { + @include type.type-style('heading-compact-01'); + color: $text-02; +} + +// Overriding styles for RTL support +html[dir='rtl'] { + .container { + & > span { + direction: ltr; + } + } + + .workLoadCard { + padding-left: unset; + padding-right: 60px; + } +} diff --git a/src/appointments-calendar/weekly/weekly-view-workload.component.tsx b/src/appointments-calendar/weekly/weekly-view-workload.component.tsx new file mode 100755 index 0000000..87c629e --- /dev/null +++ b/src/appointments-calendar/weekly/weekly-view-workload.component.tsx @@ -0,0 +1,99 @@ +import React from "react"; +import classNames from "classnames"; +import dayjs, { type Dayjs } from "dayjs"; +import { navigate, useLayoutType } from "@openmrs/esm-framework"; +import { useTranslation } from "react-i18next"; + +import { spaBasePath } from "../../constants"; +import { isSameMonth } from "../../helpers"; +import { type CalendarType } from "../../types"; +import styles from "./weekly-workload-module.scss"; + +interface WeeklyCellProps { + type: CalendarType; + dateTime: Dayjs; + currentDate: Dayjs; + events: Array; + index?: number; +} + +const WeeklyWorkloadView: React.FC = ({ + type, + dateTime, + currentDate, + events, + index, +}) => { + const layout = useLayoutType(); + const currentData = events?.find( + (event) => + dayjs(event.appointmentDate).format("YYYY-MM-DD") === + dayjs(dateTime).format("YYYY-MM-DD"), + ); + const colorCoding = { HIV: "red", "Lab testing": "purple", Refill: "blue" }; + const { t } = useTranslation(); + + return ( +
+ {type === "weekly" ? ( + <> +
+ All Day + {index !== 0 && currentData?.service && ( +
+ {currentData?.service.map(({ serviceName, count }, index) => ( +
+ navigate({ + to: `${spaBasePath}/appointments/list/${dateTime}/${serviceName}`, + }) + } + key={serviceName} + > + {serviceName} + {count} +
+ ))} +
+ navigate({ + to: `${spaBasePath}/appointments/list/${dateTime}/Total`, + }) + } + > + {t("total", "Total")} + + {currentData?.service.reduce( + (sum, currentValue) => sum + currentValue?.count ?? 0, + 0, + )} + +
+
+ )} +
+ + ) : null} +
+ ); +}; +export default WeeklyWorkloadView; diff --git a/src/appointments-calendar/weekly/weekly-view-workload.test.tsx b/src/appointments-calendar/weekly/weekly-view-workload.test.tsx new file mode 100644 index 0000000..deaa384 --- /dev/null +++ b/src/appointments-calendar/weekly/weekly-view-workload.test.tsx @@ -0,0 +1,59 @@ +import React from "react"; +import dayjs from "dayjs"; +import userEvent from "@testing-library/user-event"; +import { render, screen } from "@testing-library/react"; +import { type CalendarType } from "../../types"; +import { spaBasePath } from "../../constants"; +import { navigate } from "@openmrs/esm-framework"; +import WeeklyWorkloadView from "./weekly-view-workload.component"; + +jest.mock("@openmrs/esm-framework", () => ({ + ...jest.requireActual("@openmrs/esm-framework"), + navigate: jest.fn(), + useLayoutType: jest.fn(), +})); + +describe("WeeklyWorkloadView Component", () => { + const mockData = { + type: "weekly" as CalendarType, + dateTime: dayjs("2023-08-17"), + currentDate: dayjs("2023-08-17"), + events: [ + { + appointmentDate: "2023-08-17", + service: [ + { serviceName: "HIV", count: 2 }, + { serviceName: "Lab testing", count: 3 }, + ], + }, + ], + index: 1, + }; + + it('renders properly when type is "weekly"', () => { + // render(); + + // expect(screen.getByText("All Day")).toBeInTheDocument(); + // expect(screen.getByText("HIV")).toBeInTheDocument(); + // expect(screen.getByText("Lab testing")).toBeInTheDocument(); + }); + + it("navigates when a service area is clicked", async () => { + const user = userEvent.setup(); + + // render(); + + // await user.click(screen.getByText("HIV")); + + // expect(navigate).toHaveBeenCalledWith({ + // to: `${spaBasePath}/appointments/list/Thu, 17 Aug 2023 00:00:00 GMT/HIV`, + // }); + }); + + it("calculates and displays the total count correctly", () => { + // render(); + + // expect(screen.getByText("Total")).toBeInTheDocument(); + // expect(screen.getByText("5")).toBeInTheDocument(); + }); +}); diff --git a/src/appointments-calendar/weekly/weekly-workload-module.scss b/src/appointments-calendar/weekly/weekly-workload-module.scss new file mode 100644 index 0000000..26489d0 --- /dev/null +++ b/src/appointments-calendar/weekly/weekly-workload-module.scss @@ -0,0 +1,132 @@ +@use '@carbon/styles/scss/type'; +@use '@carbon/colors'; +@use '@carbon/layout'; +@import '~@openmrs/esm-styleguide/src/vars'; + +.monthly-cell { + border-left: 1px solid colors.$gray-20; + border-bottom: 1px solid colors.$gray-20; + background: colors.$white; + color: colors.$black; + padding: layout.$spacing-01; + text-align: right; + @include type.type-style('body-compact-02'); + + &:nth-child(-n + 7) { + border-top: 1px solid colors.$gray-20; + } + + &:nth-child(7n) { + border-right: 1px solid colors.$gray-20; + } + + &-disabled { + border-left: 1px solid colors.$gray-20; + border-bottom: 1px solid colors.$gray-20; + } + &-disabled:last-child { + border-right: 1px solid colors.$gray-20; + } +} +.weekly-cell { + position: relative; + border-right: 1px solid colors.$gray-20; + border-bottom: 1px solid colors.$gray-20; + border-left: 1px solid colors.$gray-20; + min-height: 120px; + color: colors.$black; + + .week-time { + display: none; + } + + &:nth-child(-n + 8) { + border-top: 1px solid colors.$gray-20; + .allDay { + display: none; + } + } + + &:first-child { + border-top: none; + .allDay { + display: flex; + border-bottom: 1px solid colors.$gray-20; + border-top: 1px solid colors.$gray-20; + padding-top: 50px; + padding-left: 5px; + padding-bottom: 50px; + align-items: center; + justify-content: center; + } + } + + &:nth-child(8n + 1) { + border-bottom: none; + + .week-time { + white-space: nowrap; + display: inline; + color: colors.$black; + + position: absolute; + top: -11px; + height: 22px; + width: 30px; + } + } +} +.serviceArea { + display: grid; + padding: 0.125rem; + grid-template-columns: 4fr 1fr; + justify-content: flex-start; + text-align: left; + margin: 0.125rem; + @include type.type-style('label-02'); + color: #020f1b; + cursor: pointer; + + :hover { + opacity: 0.8; + } + & > span:first-child { + text-align: left; + text-overflow: ellipsis; + width: layout.$spacing-06; + } +} +.currentData { + margin: 2px; +} + +.smallDesktop { + height: 100px; +} + +.largeDesktop { + height: 150px; +} +.red { + background-color: colors.$red-70; + color: colors.$white; + font-weight: bold; +} + +.purple { + background-color: colors.$yellow-70; + color: colors.$white; + font-weight: bold; +} + +.blue { + background-color: colors.$blue-70; + color: colors.$white; + font-weight: bold; +} + +.green { + background-color: colors.$green-70; + color: colors.$white; + font-weight: bold; +} diff --git a/src/appointments-header/appointments-header.component.tsx b/src/appointments-header/appointments-header.component.tsx new file mode 100644 index 0000000..c998171 --- /dev/null +++ b/src/appointments-header/appointments-header.component.tsx @@ -0,0 +1,85 @@ +import React, { useRef, useState } from "react"; +import { useTranslation } from "react-i18next"; +import { Location } from "@carbon/react/icons"; +import { useSession } from "@openmrs/esm-framework"; +import AppointmentsIllustration from "./appointments-illustration.component"; +import styles from "./appointments-header.scss"; +import { DatePicker, DatePickerInput, Dropdown, Layer } from "@carbon/react"; +import dayjs from "dayjs"; +import { changeStartDate, useAppointmentDate } from "../helpers"; +import { useAppointmentServices } from "../hooks/useAppointmentService"; + +interface AppointmentHeaderProps { + title: string; + onChange?: (evt) => void; +} + +const AppointmentsHeader: React.FC = ({ + title, + onChange, +}) => { + const { t } = useTranslation(); + const session = useSession(); + const datePickerRef = useRef(null); + const { currentAppointmentDate } = useAppointmentDate(); + const location = session?.sessionLocation?.display; + const { serviceTypes } = useAppointmentServices(); + + return ( +
+
+ +
+

{t("appointments", "Appointments")}

+

{title}

+
+
+
+
+ + {location} + · + changeStartDate(new Date(date))} + ref={datePickerRef} + dateFormat="d-M-Y" + datePickerType="single" + > + + +
+ {typeof onChange === "function" && ( +
+ + (item ? item.name : "")} + label={t("selectServiceType", "Select service type")} + titleText={t("view", "View")} + type="inline" + size="sm" + direction="bottom" + onChange={({ selectedItem }) => onChange(selectedItem?.uuid)} + /> + +
+ )} +
+
+ ); +}; + +export default AppointmentsHeader; diff --git a/src/appointments-header/appointments-header.scss b/src/appointments-header/appointments-header.scss new file mode 100644 index 0000000..df091b9 --- /dev/null +++ b/src/appointments-header/appointments-header.scss @@ -0,0 +1,83 @@ +@use '@carbon/colors'; +@use '@carbon/styles/scss/spacing'; +@use '@carbon/styles/scss/type'; +@import '~@openmrs/esm-styleguide/src/vars'; + +.header { + @include type.type-style('body-compact-02'); + color: $text-02; + height: spacing.$spacing-12; + background-color: $ui-02; + border-bottom: 1px solid $ui-03; + display: flex; + justify-content: space-between; +} + +.left-justified-items { + display: flex; + flex-direction: row; + align-items: center; + cursor: pointer; +} + +.right-justified-items { + @include type.type-style('body-compact-02'); + color: $text-02; + margin: 0.5rem; +} + +.page-name { + @include type.type-style('heading-04'); +} + +.page-labels { + margin: 1rem 0; + + p:first-of-type { + margin-bottom: 0.25rem; + } +} + +.date-and-location { + display: flex; + justify-content: flex-end; + align-items: center; +} + +.dropdownContainer { + display: flex; + justify-content: flex-end; +} + +.value { + margin-left: 0.25rem; +} + +.middot { + margin: 0 0.5rem; +} + +.view { + @include type.type-style('label-01'); +} + +.datePicker { + background-color: transparent; + width: 10rem; + border: none; + & > input { + color: colors.$blue-10; + } +} + +// Overriding styles for RTL support +html[dir='rtl'] { + .date-and-location { + & > svg { + order: -1; + } + & > span:nth-child(2) { + order: -2; + } + } +} diff --git a/src/appointments-header/appointments-illustration.component.tsx b/src/appointments-header/appointments-illustration.component.tsx new file mode 100644 index 0000000..53cd981 --- /dev/null +++ b/src/appointments-header/appointments-illustration.component.tsx @@ -0,0 +1,27 @@ +import React from "react"; + +const AppointmentsIllustration: React.FC = () => { + return ( + + Patient queue illustration + + + + + + + ); +}; + +export default AppointmentsIllustration; diff --git a/src/appointments-metrics/appointments-metrics.component.tsx b/src/appointments-metrics/appointments-metrics.component.tsx new file mode 100644 index 0000000..d9f5de8 --- /dev/null +++ b/src/appointments-metrics/appointments-metrics.component.tsx @@ -0,0 +1,103 @@ +import React from "react"; +import { useTranslation } from "react-i18next"; +import { ErrorState, formatDate, parseDate } from "@openmrs/esm-framework"; +import { + useClinicalMetrics, + useAllAppointmentsByDate, + useScheduledAppointment, +} from "../hooks/useClinicalMetrics"; +import { useAppointmentDate } from "../helpers"; +import { useAppointmentList } from "../hooks/useAppointmentList"; +import MetricsCard from "./metrics-card.component"; +import MetricsHeader from "./metrics-header.component"; +import styles from "./appointments-metrics.scss"; + +interface AppointmentMetricsProps { + serviceUuid: string; +} + +const AppointmentsMetrics: React.FC = ({ + serviceUuid, +}) => { + const { t } = useTranslation(); + + const { highestServiceLoad, error: clinicalMetricsError } = + useClinicalMetrics(); + const { totalProviders, isLoading: allAppointmentsLoading } = + useAllAppointmentsByDate(); + const { totalScheduledAppointments } = useScheduledAppointment(serviceUuid); + + const { currentAppointmentDate } = useAppointmentDate(); + const formattedStartDate = formatDate(parseDate(currentAppointmentDate), { + mode: "standard", + time: false, + }); + + // TODO we will need rework these after we discuss the logic we want to use + const { appointmentList: arrivedAppointments } = + useAppointmentList("CheckedIn"); + const { appointmentList: pendingAppointments } = + useAppointmentList("Scheduled"); + + const filteredArrivedAppointments = serviceUuid + ? arrivedAppointments.filter( + ({ serviceTypeUuid }) => serviceTypeUuid === serviceUuid, + ) + : arrivedAppointments; + const filteredPendingAppointments = serviceUuid + ? pendingAppointments.filter( + ({ serviceTypeUuid }) => serviceTypeUuid === serviceUuid, + ) + : pendingAppointments; + + if (clinicalMetricsError) { + return ( + + ); + } + + return ( + <> + +
+ + + +
+ + ); +}; + +export default AppointmentsMetrics; diff --git a/src/appointments-metrics/appointments-metrics.scss b/src/appointments-metrics/appointments-metrics.scss new file mode 100644 index 0000000..774e008 --- /dev/null +++ b/src/appointments-metrics/appointments-metrics.scss @@ -0,0 +1,13 @@ +@use '@carbon/styles/scss/spacing'; +@use '@carbon/styles/scss/type'; +@use '@carbon/colors'; +@import '~@openmrs/esm-styleguide/src/vars'; + +.cardContainer { + background-color: colors.$white; + display: flex; + justify-content: space-between; + padding: 0.5rem 0rem; + flex-flow: row wrap; + margin-top: -(spacing.$spacing-03); +} diff --git a/src/appointments-metrics/appointments-metrics.test.tsx b/src/appointments-metrics/appointments-metrics.test.tsx new file mode 100644 index 0000000..6f7589e --- /dev/null +++ b/src/appointments-metrics/appointments-metrics.test.tsx @@ -0,0 +1,53 @@ +import React from "react"; +import { render, screen } from "@testing-library/react"; +import { openmrsFetch } from "@openmrs/esm-framework"; +import { + mockAppointmentMetrics, + mockProvidersCount, + mockStartTime, +} from "__mocks__"; +import AppointmentsMetrics from "./appointments-metrics.component"; + +const mockedOpenmrsFetch = openmrsFetch as jest.Mock; + +jest.mock("../hooks/useClinicalMetrics", () => { + const originalModule = jest.requireActual("../hooks/useClinicalMetrics"); + + return { + ...originalModule, + useClinicalMetrics: jest.fn().mockImplementation(() => ({ + highestServiceLoad: mockAppointmentMetrics.highestServiceLoad, + isLoading: mockAppointmentMetrics.isLoading, + error: mockAppointmentMetrics.error, + })), + useAllAppointmentsByDate: jest.fn().mockImplementation(() => ({ + totalProviders: mockProvidersCount.totalProviders, + isLoading: mockProvidersCount.isLoading, + error: mockProvidersCount.error, + })), + useScheduledAppointment: jest.fn().mockImplementation(() => ({ + totalScheduledAppointments: mockAppointmentMetrics.totalAppointments, + })), + useAppointmentDate: jest.fn().mockImplementation(() => ({ + startDate: mockStartTime.startTime, + })), + }; +}); + +describe("Appointment metrics", () => { + it("renders metrics from the appointments list", async () => { + mockedOpenmrsFetch.mockResolvedValue({ data: [] }); + + renderAppointmentMetrics(); + + await screen.findByText(/appointment metrics/i); + expect(screen.getByText(/scheduled appointments/i)).toBeInTheDocument(); + expect(screen.getByText(/patients/i)).toBeInTheDocument(); + expect(screen.getByText(/16/i)).toBeInTheDocument(); + expect(screen.getByText(/4/i)).toBeInTheDocument(); + }); +}); + +function renderAppointmentMetrics() { + render(); +} diff --git a/src/appointments-metrics/metrics-card.component.tsx b/src/appointments-metrics/metrics-card.component.tsx new file mode 100644 index 0000000..60c53fe --- /dev/null +++ b/src/appointments-metrics/metrics-card.component.tsx @@ -0,0 +1,93 @@ +import React, { useMemo } from "react"; +import { useTranslation } from "react-i18next"; +import dayjs from "dayjs"; +import isSameOrBefore from "dayjs/plugin/isSameOrBefore"; +dayjs.extend(isSameOrBefore); +import isEmpty from "lodash-es/isEmpty"; +import { ConfigurableLink } from "@openmrs/esm-framework"; +import { Tile, Layer } from "@carbon/react"; +import { ArrowRight } from "@carbon/react/icons"; +import { basePath, spaBasePath, spaRoot } from "../constants"; +import styles from "./metrics-card.scss"; + +interface MetricsCardProps { + label: string; + value: number; + headerLabel: string; + children?: React.ReactNode; + view?: string; + count?: { pendingAppointments: Array; arrivedAppointments: Array }; + appointmentDate?: string; +} + +const MetricsCard: React.FC = ({ + label, + value, + headerLabel, + children, + view, + count, + appointmentDate, +}) => { + const { t } = useTranslation(); + const isDateInPast = useMemo( + () => !dayjs(appointmentDate).isBefore(dayjs(), "date"), + [appointmentDate], + ); + + const metricsLink = { + patients: "appointments-list/scheduled", + highVolume: "appointments-list/high-volume-service", + providers: "appointments-list/providers-link", + }; + + return ( + + +
+
+ + {children} +
+ {view && ( +
+ + + {t("view", "View")} + {" "} + + +
+ )} +
+
+
+ +

{value}

+
+ {!isEmpty(count) && ( +
+ {t("checkedIn", "Checked in")} + + {isDateInPast + ? t("notArrived", "Not arrived") + : t("missed", "Missed")} + +

+ {count.arrivedAppointments?.length} +

+

+ {count.pendingAppointments?.length} +

+
+ )} +
+
+
+ ); +}; + +export default MetricsCard; diff --git a/src/appointments-metrics/metrics-card.scss b/src/appointments-metrics/metrics-card.scss new file mode 100644 index 0000000..782a28d --- /dev/null +++ b/src/appointments-metrics/metrics-card.scss @@ -0,0 +1,76 @@ +@use '@carbon/styles/scss/spacing'; +@use '@carbon/styles/scss/type'; +@use '@carbon/colors'; +@import '~@openmrs/esm-styleguide/src/vars'; + +.container { + flex-grow: 1; +} + +.tileContainer { + border: 0.0625rem solid $ui-03; + height: 7.875rem; + padding: spacing.$spacing-05; + margin: spacing.$spacing-03 spacing.$spacing-03; +} + +.tileHeader { + display: flex; + justify-content: space-between; + align-items: baseline; + margin-bottom: spacing.$spacing-03; +} + +.headerLabel { + @include type.type-style('heading-compact-01'); + color: $text-02; +} + +.totalsLabel { + @include type.type-style('label-01'); + color: $text-02; +} + +.totalsValue { + @include type.type-style('heading-04'); + color: $ui-05; +} + +.headerLabelContainer { + display: flex; + align-items: center; + height: spacing.$spacing-07; + justify-content: space-between; + width: 100%; +} + +.link { + text-decoration: none; + display: flex; + align-items: center; + color: $interactive-01; +} + +.metricsGrid { + display: grid; + grid-template-columns: 1fr 1fr; +} + +.countGrid { + display: grid; + grid-template-columns: 1fr 1fr; + justify-self: flex-end; + column-gap: spacing.$spacing-03; + row-gap: spacing.$spacing-03; + margin: spacing.$spacing-03; + + & > span { + font-size: 0.625rem !important; + margin: 0; + color: colors.$gray-70; + } + + & > p { + margin: 0; + } +} diff --git a/src/appointments-metrics/metrics-header.component.tsx b/src/appointments-metrics/metrics-header.component.tsx new file mode 100644 index 0000000..9e15cdf --- /dev/null +++ b/src/appointments-metrics/metrics-header.component.tsx @@ -0,0 +1,58 @@ +import React from "react"; +import dayjs from "dayjs"; +import isToday from "dayjs/plugin/isToday"; +dayjs.extend(isToday); +import { useTranslation } from "react-i18next"; +import { Calendar, Hospital } from "@carbon/react/icons"; +import { Button } from "@carbon/react"; +import { ExtensionSlot, navigate } from "@openmrs/esm-framework"; +import { spaBasePath } from "../constants"; +import { closeOverlay, launchOverlay } from "../hooks/useOverlay"; +import AppointmentForm from "../appointments/forms/create-edit-form/appointments-form.component"; +import styles from "./metrics-header.scss"; + +const MetricsHeader: React.FC = () => { + const { t } = useTranslation(); + + const launchCreateAppointmentForm = (patientUuid) => { + closeOverlay(); + launchOverlay( + t("appointmentForm", "Create Appointment"), + , + ); + }; + + return ( +
+ + {t("appointmentMetrics", "Appointment metrics")} + +
+ + , + size: "lg", + }, + }} + /> +
+
+ ); +}; + +export default MetricsHeader; diff --git a/src/appointments-metrics/metrics-header.scss b/src/appointments-metrics/metrics-header.scss new file mode 100644 index 0000000..473cf4d --- /dev/null +++ b/src/appointments-metrics/metrics-header.scss @@ -0,0 +1,33 @@ +@use '@carbon/styles/scss/spacing'; +@use '@carbon/styles/scss/type'; +@import '~@openmrs/esm-styleguide/src/vars'; + +.metricsContainer { + display: flex; + justify-content: space-between; + background-color: $ui-02; + height: spacing.$spacing-10; + align-items: center; + padding: 0 spacing.$spacing-05; +} + +.metricsTitle { + @include type.type-style('heading-03'); + color: $ui-05; +} + +.link { + text-decoration: none; + display: flex; + align-items: center; +} + +.viewListBtn { + margin: 0 0 0 spacing.$spacing-03; +} + +.metricsContent { + display: flex; + align-items: center; + column-gap: spacing.$spacing-02; +} diff --git a/src/appointments.component.tsx b/src/appointments.component.tsx new file mode 100644 index 0000000..9bc46fb --- /dev/null +++ b/src/appointments.component.tsx @@ -0,0 +1,26 @@ +import React, { useState } from "react"; +import { useTranslation } from "react-i18next"; +import AppointmentList from "./appointments/appointment-tabs.component"; +import AppointmentsHeader from "./appointments-header/appointments-header.component"; +import ClinicMetrics from "./appointments-metrics/appointments-metrics.component"; +import Overlay from "./overlay.component"; + +const ClinicalAppointments: React.FC = () => { + const { t } = useTranslation(); + const [appointmentServiceType, setAppointmentServiceType] = + useState(""); + + return ( + <> + + + + + + ); +}; + +export default ClinicalAppointments; diff --git a/src/appointments.test.tsx b/src/appointments.test.tsx new file mode 100644 index 0000000..b3fd329 --- /dev/null +++ b/src/appointments.test.tsx @@ -0,0 +1,19 @@ +import React from "react"; +import { render, screen } from "@testing-library/react"; +import ClinicalAppointments from "./appointments.component"; + +describe("Clinical Appointments", () => { + it("should render correctly", () => { + // render(); + + // expect( + // screen.getByTitle(/patient queue illustration/i), + // ).toBeInTheDocument(); + // expect(screen.getByText(/^appointments$/i)).toBeInTheDocument(); + // expect(screen.getByText(/home/i)).toBeInTheDocument(); + // expect(screen.getByPlaceholderText(/DD-MMM-YYYY/i)).toBeInTheDocument(); + // expect( + // screen.getByRole("button", { name: /appointments calendar/i }), + // ).toBeInTheDocument(); + }); +}); diff --git a/src/appointments/appointment-tabs.component.tsx b/src/appointments/appointment-tabs.component.tsx new file mode 100644 index 0000000..4b65ff7 --- /dev/null +++ b/src/appointments/appointment-tabs.component.tsx @@ -0,0 +1,66 @@ +import React, { useState } from "react"; +import { useTranslation } from "react-i18next"; +import { Tab, TabList, Tabs, TabPanel, TabPanels } from "@carbon/react"; + +import { type ConfigObject } from "../config-schema"; +import { useConfig } from "@openmrs/esm-framework"; +import { useVisits } from "../hooks/useVisits"; +import ScheduledAppointments from "./scheduled/scheduled-appointments.component"; +import UnscheduledAppointments from "./unscheduled/unscheduled-appointments.component"; +import styles from "./appointment-tabs.scss"; + +interface AppointmentTabsProps { + appointmentServiceType: string; +} + +const AppointmentTabs: React.FC = ({ + appointmentServiceType, +}) => { + const { t } = useTranslation(); + const [activeTabIndex, setActiveTabIndex] = useState(0); + + const { isLoading = false, visits = [], mutateVisit } = useVisits(); + + const { showUnscheduledAppointmentsTab } = useConfig(); + + const handleTabChange = ({ selectedIndex }: { selectedIndex: number }) => { + setActiveTabIndex(selectedIndex); + }; + + return ( +
+ {showUnscheduledAppointmentsTab ? ( + + + {t("scheduled", "Scheduled")} + {t("unscheduled", "Unscheduled")} + + + + + + + + + + + ) : ( + + )} +
+ ); +}; + +export default AppointmentTabs; diff --git a/src/appointments/appointment-tabs.scss b/src/appointments/appointment-tabs.scss new file mode 100644 index 0000000..6c1f653 --- /dev/null +++ b/src/appointments/appointment-tabs.scss @@ -0,0 +1,54 @@ +@use '@carbon/styles/scss/spacing'; +@use '@carbon/styles/scss/type'; +@use '@carbon/colors'; +@import '../root.scss'; + +.appointmentList { + height: 100%; + width: 100%; + + & > div { + background-color: $ui-02; + } +} + +.tabs { + grid-column: 'span 2'; +} + +.tab { + min-width: 12rem; + + &:active, + &:focus { + outline: 2px solid var(--brand-03) !important; + } + + &[aria-selected='true'] { + box-shadow: inset 0 2px 0 0 var(--brand-03) !important; + } +} + +.tabPanel { + padding: 0; + margin: 1rem; +} + +.calendarButton { + float: right; + position: absolute; + right: 0; + height: spacing.$spacing-09; + min-height: 0; +} + +.downloadButton { + margin: spacing.$spacing-05; + & > button { + border: 1px solid colors.$blue-60; + } +} +.downloadLink { + text-decoration: none; + color: inherit; +} diff --git a/src/appointments/appointment-tabs.test.tsx b/src/appointments/appointment-tabs.test.tsx new file mode 100644 index 0000000..19826ec --- /dev/null +++ b/src/appointments/appointment-tabs.test.tsx @@ -0,0 +1,76 @@ +import React from "react"; +import { screen } from "@testing-library/react"; +import userEvent from "@testing-library/user-event"; +import { openmrsFetch } from "@openmrs/esm-framework"; +import { renderWithSwr, waitForLoadingToFinish } from "tools"; +import { mockAppointmentsData } from "__mocks__"; +import AppointmentTabs from "./appointment-tabs.component"; + +const mockOpenmrsFetch = openmrsFetch as jest.Mock; + +jest.setTimeout(10000); + +describe("AppointmentTabs", () => { + xit(`renders tabs showing different appointment lists`, async () => { + const user = userEvent.setup(); + + mockOpenmrsFetch.mockReturnValueOnce({ data: mockAppointmentsData.data }); + + renderAppointmentTabs(); + + await waitForLoadingToFinish(); + + const scheduledAppointmentsTab = screen.getByRole("tab", { + name: /^scheduled$/i, + }); + const unsheduledAppointment = screen.getByRole("tab", { + name: /^unscheduled$/i, + }); + const pendingAppointments = screen.getByRole("tab", { + name: /^unscheduled$/i, + }); + + expect(scheduledAppointmentsTab).toBeInTheDocument(); + expect(unsheduledAppointment).toBeInTheDocument(); + expect(pendingAppointments).toBeInTheDocument(); + + expect(scheduledAppointmentsTab).toHaveAttribute("aria-selected", "true"); + expect(unsheduledAppointment).toHaveAttribute("aria-selected", "false"); + expect(pendingAppointments).toHaveAttribute("aria-selected", "false"); + + expect( + screen.getByRole("button", { name: /add new appointment/i }), + ).toBeInTheDocument(); + expect(screen.getByText(/view calendar/i)).toBeInTheDocument(); + expect(screen.getByRole("table")).toBeInTheDocument(); + const expectedColumnHeaders = [ + /name/, + /date & time/, + /service type/, + /provider/, + /location/, + /actions/, + ]; + expectedColumnHeaders.forEach((header) => { + expect( + screen.getByRole("columnheader", { name: new RegExp(header, "i") }), + ).toBeInTheDocument(); + }); + + const expectedTableRows = [ + /John Wilson 30-Aug-2021, 12:35 PM Outpatient HIV Clinic/, + /Elon Musketeer 14-Sept-2021, 07:50 AM Outpatient HIV Clinic/, + /Hopkins Derrick 14-Sept-2021, 12:50 PM Outpatient TB Clinic/, + /Amos Strong 15-Sept-2021, 01:32 PM Outpatient TB Clinic/, + ]; + expectedTableRows.forEach((row) => { + expect( + screen.getByRole("row", { name: new RegExp(row, "i") }), + ).toBeInTheDocument(); + }); + }); +}); + +function renderAppointmentTabs() { + renderWithSwr(); +} diff --git a/src/appointments/appointments-table.resource.ts b/src/appointments/appointments-table.resource.ts new file mode 100644 index 0000000..89b0312 --- /dev/null +++ b/src/appointments/appointments-table.resource.ts @@ -0,0 +1,46 @@ +import { useMemo } from "react"; +import useSWR from "swr"; +import { openmrsFetch } from "@openmrs/esm-framework"; +import { type AppointmentService, type Appointment } from "../types"; +import { getAppointment, useAppointmentDate } from "../helpers"; +import isEmpty from "lodash-es/isEmpty"; + +export function useAppointments(status?: string, forDate?: string) { + const { currentAppointmentDate } = useAppointmentDate(); + const startDate = forDate ? forDate : currentAppointmentDate; + const apiUrl = `/ws/rest/v1/appointment/appointmentStatus?forDate=${startDate}&status=${status}`; + const allAppointmentsUrl = `/ws/rest/v1/appointment/all?forDate=${startDate}`; + + const { data, error, isLoading, isValidating, mutate } = useSWR< + { data: Array }, + Error + >(isEmpty(status) ? allAppointmentsUrl : apiUrl, openmrsFetch); + + const appointments = useMemo( + () => data?.data?.map((appointment) => getAppointment(appointment)) ?? [], + [data?.data], + ); + + return { + appointments, + isLoading, + isError: error, + isValidating, + mutate, + }; +} + +export function useServices() { + const apiUrl = `/ws/rest/v1/appointmentService/all/default`; + const { data, error, isLoading, isValidating } = useSWR< + { data: Array }, + Error + >(apiUrl, openmrsFetch); + + return { + services: data ? data.data : [], + isLoading, + isError: error, + isValidating, + }; +} diff --git a/src/appointments/appointments.resources.ts b/src/appointments/appointments.resources.ts new file mode 100644 index 0000000..40252d0 --- /dev/null +++ b/src/appointments/appointments.resources.ts @@ -0,0 +1,78 @@ +import useSWR from "swr"; +import { openmrsFetch } from "@openmrs/esm-framework"; +import { + type AppointmentService, + type AppointmentsFetchResponse, + Provider, +} from "../types"; + +export const appointmentsSearchUrl = `/ws/rest/v1/appointments/search`; + +export function useAppointments(patientUuid: string, startDate: string) { + const abortController = new AbortController(); + /* + SWR isn't meant to make POST requests for data fetching. This is a consequence of the API only exposing this resource via POST. + This works but likely isn't recommended. + */ + const fetcher = () => + openmrsFetch(appointmentsSearchUrl, { + method: "POST", + signal: abortController.signal, + headers: { + "Content-Type": "application/json", + }, + body: { + patientUuid: patientUuid, + startDate: startDate, + }, + }); + + const { data, error, isLoading, isValidating } = useSWR< + AppointmentsFetchResponse, + Error + >(appointmentsSearchUrl, fetcher); + + const appointments = data?.data?.length + ? data.data.sort((a, b) => (b.startDateTime > a.startDateTime ? 1 : -1)) + : null; + + return { + data: data ? appointments : null, + isError: error, + isLoading, + isValidating, + }; +} + +export function useAppointmentService() { + const { data, error, isLoading } = useSWR< + { data: Array }, + Error + >(`/ws/rest/v1/appointmentService/all/full`, openmrsFetch); + + return { + data: data ? data.data : null, + isError: error, + isLoading, + }; +} + +export function getAppointmentsByUuid(appointmentUuid: string) { + const abortController = new AbortController(); + + return openmrsFetch(`/ws/rest/v1/appointments/${appointmentUuid}`, { + signal: abortController.signal, + }); +} + +export function getAppointmentService(uuid) { + const abortController = new AbortController(); + + return openmrsFetch(`/ws/rest/v1/appointmentService?uuid=` + uuid, { + signal: abortController.signal, + }); +} + +export function getTimeSlots() { + //https://openmrs-spa.org/openmrs/ws/rest/v1/appointment/all?forDate=2020-03-02T21:00:00.000Z +} diff --git a/src/appointments/common-components/appointments-actions.component.tsx b/src/appointments/common-components/appointments-actions.component.tsx new file mode 100644 index 0000000..a65f211 --- /dev/null +++ b/src/appointments/common-components/appointments-actions.component.tsx @@ -0,0 +1,139 @@ +import React from "react"; +import dayjs from "dayjs"; +import isToday from "dayjs/plugin/isToday"; +import utc from "dayjs/plugin/utc"; +import { Button, OverflowMenu, OverflowMenuItem } from "@carbon/react"; +import { TaskComplete } from "@carbon/react/icons"; +import { useTranslation } from "react-i18next"; + +import { launchOverlay } from "../../hooks/useOverlay"; +import AppointmentForm from "../forms/create-edit-form/appointments-form.component"; +import CheckInButton from "./checkin-button.component"; +import { type MappedAppointment } from "../../types"; +import { showModal } from "@openmrs/esm-framework"; +import { useVisits } from "../../hooks/useVisits"; +import DefaulterTracingForm from "../forms/defaulter-tracing-form/default-tracing-form.component"; + +dayjs.extend(utc); +dayjs.extend(isToday); + +interface AppointmentActionsProps { + visits: Array; + appointment: MappedAppointment; + scheduleType: string; +} + +const AppointmentActions: React.FC = ({ + visits, + appointment, + scheduleType, +}) => { + const { t } = useTranslation(); + const { mutateVisit } = useVisits(); + const patientUuid = appointment.patientUuid; + const visitDate = dayjs(appointment.dateTime); + const isFutureAppointment = visitDate.isAfter(dayjs()); + const isTodayAppointment = visitDate.isToday(); + const hasActiveVisit = visits?.some( + (visit) => visit?.patient?.uuid === patientUuid && visit?.startDatetime, + ); + const hasCheckedOut = visits?.some( + (visit) => + visit?.patient?.uuid === patientUuid && + visit?.startDatetime && + visit?.stopDatetime, + ); + + const handleCheckout = () => { + const dispose = showModal("end-visit-dialog", { + closeModal: () => { + mutateVisit(); + dispose(); + }, + patientUuid, + }); + }; + + const handleOpenDefaulterForm = () => { + launchOverlay( + "CCC Defaulter tracing form", + , + ); + }; + + /** + * Renders the appropriate visit status button based on the current appointment state. + * @returns {JSX.Element} The rendered button. + */ + const renderVisitStatus = () => { + const checkedOutText = t("checkedOut", "Checked out"); + const followUpButtonText = t("launchFormUpForm", "Follow up"); + + switch (true) { + case hasCheckedOut: + return ( + + ); + case hasActiveVisit && isTodayAppointment: + return ( + + ); + case isTodayAppointment: { + const isAfterNoon = new Date().getHours() > 12; + + if (scheduleType === "Pending" && isAfterNoon) { + return ( + + ); + } + + return ( + + ); + } + + default: + if (!isFutureAppointment) { + return ( + + ); + } + return null; + } + }; + + return ( +
+ {renderVisitStatus()} + {isFutureAppointment || + (isTodayAppointment && (!handleCheckout || !hasActiveVisit)) ? ( + + + launchOverlay( + t("editAppointments", "Edit Appointment"), + , + ) + } + /> + + ) : null} +
+ ); +}; + +export default AppointmentActions; diff --git a/src/appointments/common-components/appointments-actions.test.tsx b/src/appointments/common-components/appointments-actions.test.tsx new file mode 100644 index 0000000..baf29a6 --- /dev/null +++ b/src/appointments/common-components/appointments-actions.test.tsx @@ -0,0 +1,82 @@ +import { render, screen } from "@testing-library/react"; +import React from "react"; +import AppointmentActions from "./appointments-actions.component"; +import { type MappedAppointment } from "../../types"; + +describe("AppointmentActions", () => { + const defaultProps = { + visits: [], + appointment: { + patientUuid: "123", + dateTime: new Date().toISOString(), + } as MappedAppointment, + scheduleType: "Pending", + }; + + beforeAll(() => { + jest.useFakeTimers({ + legacyFakeTimers: false, + }); + const currentDateTime = new Date(); + currentDateTime.setHours(12); + currentDateTime.setMinutes(0); + + jest.setSystemTime(currentDateTime); + }); + + afterAll(() => { + jest.useRealTimers(); + }); + + it("renders the correct button when the patient has checked out", () => { + const visits = [ + { + patient: { uuid: "123" }, + startDatetime: new Date().toISOString(), + stopDatetime: new Date().toISOString(), + }, + ]; + const props = { ...defaultProps, visits }; + // const { getByText } = render(); + // const button = getByText("Checked out"); + // expect(button).toBeInTheDocument(); + }); + + it("renders the correct button when the patient has an active visit and today is the appointment date", () => { + const visits = [ + { + patient: { uuid: "123" }, + startDatetime: new Date().toISOString(), + stopDatetime: null, + }, + ]; + const props = { ...defaultProps, visits, scheduleType: "Scheduled" }; + // const { getByText } = render(); + // const button = getByText("Check out"); + // expect(button).toBeInTheDocument(); + }); + + it("renders the correct button when today is the appointment date and the schedule type is pending", () => { + const props = { ...defaultProps, scheduleType: "Pending" }; + // render(); + // const button = screen.getByRole("button", { name: "Actions" }); + // expect(button).toBeInTheDocument(); + }); + + it("renders the correct button when today is the appointment date and the schedule type is not pending", () => { + const props = { ...defaultProps, scheduleType: "Confirmed" }; + // render(); + // const button = screen.getByRole("button", { name: "Actions" }); + // expect(button).toBeInTheDocument(); + }); + + it("renders the correct button when the appointment is in the past or has not been scheduled", () => { + const props = { + ...defaultProps, + appointment: { ...defaultProps.appointment, dateTime: "2022-01-01" }, + }; + // render(); + // const button = screen.getByRole("button", { name: "Follow up" }); + // expect(button).toBeInTheDocument(); + }); +}); diff --git a/src/appointments/common-components/appointments-table.component.tsx b/src/appointments/common-components/appointments-table.component.tsx new file mode 100644 index 0000000..76a110a --- /dev/null +++ b/src/appointments/common-components/appointments-table.component.tsx @@ -0,0 +1,228 @@ +import React, { useState } from "react"; +import { useTranslation } from "react-i18next"; +import { + DataTableSkeleton, + DataTable, + TableContainer, + Table, + TableHead, + TableRow, + TableExpandHeader, + TableHeader, + TableBody, + TableExpandRow, + TableCell, + TableExpandedRow, + Pagination, + TableToolbar, + TableToolbarContent, + TableToolbarSearch, + Button, +} from "@carbon/react"; +import { + ConfigurableLink, + formatDatetime, + usePagination, + formatDate, + useConfig, +} from "@openmrs/esm-framework"; +import startCase from "lodash-es/startCase"; +import { Download } from "@carbon/react/icons"; +import { EmptyState } from "../../empty-state/empty-state.component"; +import { downloadAppointmentsAsExcel } from "../../helpers/excel"; +import { launchOverlay } from "../../hooks/useOverlay"; +import { type MappedAppointment } from "../../types"; +import { getPageSizes, useSearchResults } from "../utils"; +import { type ConfigObject } from "../../config-schema"; +import AppointmentDetails from "../details/appointment-details.component"; +import AppointmentActions from "./appointments-actions.component"; +import PatientSearch from "../../patient-search/patient-search.component"; +import styles from "./appointments-table.scss"; + +interface AppointmentsTableProps { + appointments: Array; + isLoading: boolean; + tableHeading: string; + mutate?: () => void; + visits?: Array; + scheduleType?: string; +} + +const AppointmentsTable: React.FC = ({ + appointments, + isLoading, + tableHeading, + visits, + scheduleType, +}) => { + const { t } = useTranslation(); + const [pageSize, setPageSize] = useState(25); + const [searchString, setSearchString] = useState(""); + const searchResults = useSearchResults(appointments, searchString); + const { results, goTo, currentPage } = usePagination(searchResults, pageSize); + const { customPatientChartUrl } = useConfig(); + + const headerData = [ + { + header: t("patientName", "Patient name"), + key: "patientName", + }, + { + header: t("identifier", "Identifier"), + key: "identifier", + }, + { + header: t("serviceType", "Service Type"), + key: "serviceType", + }, + { + header: t("actions", "Actions"), + key: "actions", + }, + ]; + + const rowData = results?.map((appointment, index) => ({ + id: `${index}`, + uuid: appointment.uuid, + patientName: ( + + {appointment.name} + + ), + nextAppointmentDate: "--", + identifier: appointment.identifier, + dateTime: formatDatetime(new Date(appointment.dateTime)), + serviceType: appointment.serviceType, + provider: appointment.provider, + actions: ( + + ), + })); + + if (isLoading) { + return ; + } + + if (!appointments?.length) { + return ( + + launchOverlay(t("search", "Search"), ) + } + scheduleType={scheduleType} + /> + ); + } + + return ( + <> + + {({ + rows, + headers, + getHeaderProps, + getRowProps, + getTableProps, + getToolbarProps, + getTableContainerProps, + }) => ( + + + + setSearchString(event.target.value)} + /> + + + + + + + + {headers.map((header) => ( + + {header.header} + + ))} + + + + {rows.map((row) => ( + + + {row.cells.map((cell) => ( + + {cell.value?.content ?? cell.value} + + ))} + + {row.isExpanded ? ( + + + + ) : ( + + )} + + ))} + +
+ +
+
+ )} +
+ { + goTo(page); + setPageSize(pageSize); + }} + pageSizes={getPageSizes(appointments, pageSize) ?? []} + totalItems={appointments.length ?? 0} + /> + + ); +}; + +export default AppointmentsTable; diff --git a/src/appointments/common-components/appointments-table.scss b/src/appointments/common-components/appointments-table.scss new file mode 100644 index 0000000..8b1617e --- /dev/null +++ b/src/appointments/common-components/appointments-table.scss @@ -0,0 +1,18 @@ +@use '@carbon/styles/scss/spacing'; +@use '@carbon/styles/scss/type'; + +.expandedActiveVisitRow > td > div { + max-height: max-content !important; +} + +.expandedActiveVisitRow td { + padding: 0 2rem; +} + +.expandedActiveVisitRow th[colspan] td[colspan] > div:first-child { + padding: 0 1rem; +} + +.hiddenRow { + display: none; +} diff --git a/src/appointments/common-components/appointments-table.test.tsx b/src/appointments/common-components/appointments-table.test.tsx new file mode 100644 index 0000000..5296789 --- /dev/null +++ b/src/appointments/common-components/appointments-table.test.tsx @@ -0,0 +1,206 @@ +import React from "react"; +import userEvent from "@testing-library/user-event"; +import { render, screen } from "@testing-library/react"; +import { type MappedAppointment } from "../../types"; +import { usePagination } from "@openmrs/esm-framework"; +import { downloadAppointmentsAsExcel } from "../../helpers/excel"; +import { launchOverlay } from "../../hooks/useOverlay"; +import AppointmentsTable from "./appointments-table.component"; +import PatientSearch from "../../patient-search/patient-search.component"; + +// Define mock appointments data for testing purposes +const appointments: Array = [ + { + patientUuid: "1234", + name: "John Smith", + identifier: "12345", + dateTime: "2023-04-13T12:00:00.000Z", + serviceType: "Service", + provider: "Dr. Jane Doe", + id: "1234", + age: "50", + gender: "F", + providers: [], + appointmentKind: "Scheduled", + appointmentNumber: "1234", + location: "location", + phoneNumber: "1234567890", + status: "status", + comments: "some comments", + dob: "2020-04-13T12:00:00.000Z", + serviceUuid: "1234", + }, +]; + +// const mockUsePagination = usePagination as jest.Mock; +// const mockGoToPage = jest.fn(); +// const mockDownloadAppointmentsAsExcel = +// downloadAppointmentsAsExcel as jest.Mock; +// const mockLaunchOverlay = launchOverlay as jest.Mock; + +jest.mock("../../helpers/excel"); +jest.mock("../../hooks/useOverlay"); + +jest.mock("@openmrs/esm-framework", () => { + // const originalModule = jest.requireActual("@openmrs/esm-framework"); + // return { + // ...originalModule, + // openmrsFetch: jest.fn(), + // useConfig: jest.fn(() => ({ + // customPatientChartUrl: "someUrl", + // })), + // }; +}); + +describe("AppointmentsBaseTable", () => { + const props = { + appointments: [], + isLoading: false, + tableHeading: "Scheduled", + visits: [], + scheduleType: "Scheduled", + }; + + it("should render empty state if appointments are not provided", async () => { + const user = userEvent.setup(); + + // render(); + + // await screen.findByRole("heading", { name: /scheduled appointment/i }); + + // const emptyScreenText = screen.getByText( + // /There are no scheduled appointments to display/, + // ); + // expect(emptyScreenText).toBeInTheDocument(); + + // const launchAppointmentsForm = screen.getByRole("button", { + // name: /Create appointment/, + // }); + + // await user.click(launchAppointmentsForm); + + // expect(mockLaunchOverlay).toHaveBeenCalledWith("Search", ); + }); + + it("should render loading state when loading data", () => { + // render(); + + // expect(screen.getByRole("progressbar")).toBeInTheDocument(); + }); + + it("should render table with headers and rows if appointments are provided", async () => { + // mockUsePagination.mockReturnValue({ + // results: appointments.slice(0, 2), + // goTo: mockGoToPage, + // currentPage: 1, + // }); + + + // // render(); + + // // await screen.findByRole("heading", { name: /scheduled appointment/i }); + + // expect(screen.getByText("Patient name")).toBeInTheDocument(); + // expect(screen.getByText("Identifier")).toBeInTheDocument(); + // expect(screen.getByText("Service Type")).toBeInTheDocument(); + // expect(screen.getByText("Actions")).toBeInTheDocument(); + // const patient = screen.getByText("John Smith"); + // expect(patient).toBeInTheDocument(); + // expect(patient).toHaveAttribute("href", "someUrl"); + // expect(screen.getByText("12345")).toBeInTheDocument(); + // expect(screen.getByText("Service")).toBeInTheDocument(); + }); + + it("should update search string when search input is changed", async () => { + const user = userEvent.setup(); + + // render(); + + // await screen.findByRole("heading", { name: /scheduled appointment/i }); + + // const searchInput = screen.getByRole("searchbox"); + + // await user.type(searchInput, "John"); + // expect(searchInput).toHaveValue("John"); + }); + + // it("should contain the title 'Scheduled appointments' and Total count", async () => { + // // render(); + + // // await screen.findByRole("heading", { name: /scheduled appointment/i }); + + // expect(screen.getByText(/Scheduled appointment/)).toBeInTheDocument(); + // expect(screen.getByText(/Total 1/)).toBeInTheDocument(); +}); + +it("should execute the download function when download button is clicked", async () => { + // const user = userEvent.setup(); + + // // render(); + + // // await screen.findByRole("heading", { name: /scheduled appointment/i }); + + // const downloadButton = screen.getByRole("button", { name: /Download/ }); + + // await user.click(downloadButton); + + // expect(downloadButton).toBeInTheDocument(); + // expect(mockDownloadAppointmentsAsExcel).toHaveBeenCalledWith( + // appointments, + // expect.anything(), + // ); +}); + +it("should have pagination when there are more than 25 appointments", async () => { + const user = userEvent.setup(); + + const mockAppointments = Array.from({ length: 100 }, (_, i) => ({ + patientUuid: `${i}`, + name: `Patient ${i}`, + identifier: `${i}${i}${i}${i}${i}`, + dateTime: new Date().toISOString(), + serviceType: "Service", + provider: `Dr. Provider ${i}`, + id: `${i}`, + age: `${i + 20}`, + gender: i % 2 === 0 ? "M" : "F", + providers: [], + appointmentKind: "Scheduled", + appointmentNumber: `${i}${i}${i}${i}${i}`, + location: "Location", + phoneNumber: "1234567890", + status: "Status", + comments: "Some comments", + dob: new Date(`${i + 1950}-01-01T00:00:00.000Z`).toISOString(), + serviceUuid: `${i}${i}${i}${i}${i}`, + })); + // mockUsePagination.mockReturnValue({ + // results: appointments.slice(0, 2), + // goTo: mockGoToPage, + // currentPage: 1, + // }); + + + // render(); + + // await screen.findByRole("heading", { name: /scheduled appointment/i }); + + // expect(screen.getByText(/1–25 of 100 items/)).toBeInTheDocument(); + // const nextPageButton = screen.getByRole("button", { name: /Next page/ }); + // const previousPageButton = screen.getByRole("button", { + // name: /Previous page/, + // }); + + // // Clicking the next page button should call the goTo function with the next page number + // await user.click(nextPageButton); + + // expect(mockGoToPage).toHaveBeenCalledWith(2); + // expect(screen.getByText(/26–50 of 100 items/)).toBeInTheDocument(); + + // // Clicking the previous page button should call the goTo function with the previous page number + // await user.click(previousPageButton); + + // expect(mockGoToPage).toHaveBeenCalledWith(1); + + // expect(screen.getByText(/1–25 of 100 items/)).toBeInTheDocument(); +}); \ No newline at end of file diff --git a/src/appointments/common-components/checkin-button.component.tsx b/src/appointments/common-components/checkin-button.component.tsx new file mode 100644 index 0000000..badf71b --- /dev/null +++ b/src/appointments/common-components/checkin-button.component.tsx @@ -0,0 +1,44 @@ +import React from "react"; +import { Button } from "@carbon/react"; +import { useTranslation } from "react-i18next"; +import { launchOverlay } from "../../hooks/useOverlay"; +import VisitForm from "../../patient-queue/visit-form/visit-form.component"; +import { type MappedAppointment } from "../../types"; +import dayjs from "dayjs"; +import isToday from "dayjs/plugin/isToday"; +import utc from "dayjs/plugin/utc"; +dayjs.extend(utc); +dayjs.extend(isToday); + +interface CheckInButtonProps { + patientUuid: string; + appointment: MappedAppointment; +} + +const CheckInButton: React.FC = ({ + appointment, + patientUuid, +}) => { + const { t } = useTranslation(); + return ( + <> + {(dayjs(appointment.dateTime).isAfter(dayjs()) || + dayjs(appointment.dateTime).isToday()) && ( + + )} + + ); +}; + +export default CheckInButton; diff --git a/src/appointments/common-components/location-select-option.component.tsx b/src/appointments/common-components/location-select-option.component.tsx new file mode 100644 index 0000000..66acd74 --- /dev/null +++ b/src/appointments/common-components/location-select-option.component.tsx @@ -0,0 +1,60 @@ +import React from "react"; +import { useTranslation } from "react-i18next"; +import { SelectItem } from "@carbon/react"; + +interface SelectLocationProps { + selectedLocation: string; + defaultFacility: { + uuid: string; + display: string; + }; + locations?: Array; +} + +interface LocationOptions { + uuid?: string; + display?: string; +} + +const LocationSelectOption: React.FC = ({ + selectedLocation, + defaultFacility, + locations, +}) => { + const { t } = useTranslation(); + if (!selectedLocation) { + return ; + } + + if (defaultFacility && Object.keys(defaultFacility).length !== 0) { + return ( + + {defaultFacility?.display} + + ); + } + + if (locations && locations.length > 0) { + return ( + <> + {locations.map((location) => ( + + {location.display} + + ))} + + ); + } + + return null; +}; + +export default LocationSelectOption; diff --git a/src/appointments/details/appointment-details.component.tsx b/src/appointments/details/appointment-details.component.tsx new file mode 100644 index 0000000..dda0731 --- /dev/null +++ b/src/appointments/details/appointment-details.component.tsx @@ -0,0 +1,106 @@ +import React from "react"; +import { useTranslation } from "react-i18next"; +import { type MappedAppointment } from "../../types/index"; +import styles from "./appointment-details.scss"; +import { usePatientAppointmentHistory } from "../../hooks/usePatientAppointmentHistory"; +import { formatDate } from "@openmrs/esm-framework"; +import { getGender } from "../../helpers"; + +interface AppointmentDetailsProps { + appointment: MappedAppointment; +} + +const AppointmentDetails: React.FC = ({ + appointment, +}) => { + const { t } = useTranslation(); + const { appointmentsCount } = usePatientAppointmentHistory( + appointment.patientUuid, + ); + + return ( +
+

{appointment.serviceType}

+

+ {formatDate(new Date(appointment.dateTime))} +

+ +
+
+

+ {t("patientDetails", "Patient details")} +

+
+

+ {t("patientName", "Patient name")} :{" "} +

+

{appointment.name}

+
+
+

{t("age", "Age")} :

+

{appointment.age}

+
+
+

{t("gender", "Gender")} :

+

{getGender(appointment.gender, t)}

+
+
+

{t("dob", "Dob")} :

+

{appointment.dob}

+
+
+

+ {t("phoneNumber", "Phone number")} : +

+

{appointment.phoneNumber}

+
+
+
+

+ {t("appointmentNotes", "Appointment Notes")} +

+

{appointment.comments}

+
+
+

+ {t("appointmentHistory", "Appointment History")} +

+
+
+

+ {t("completed", "Completed")} +

+ + {appointmentsCount.completedAppointments} + +
+
+

{t("missed", "Missed")}

+ + {appointmentsCount.missedAppointments} + +
+
+

+ {t("cancelled", "Cancelled")} +

+ + {appointmentsCount.cancelledAppointments} + +
+
+

+ {t("upcoming", "Upcoming")} +

+ + {appointmentsCount.upcomingAppointments} + +
+
+
+
+
+ ); +}; + +export default AppointmentDetails; diff --git a/src/appointments/details/appointment-details.scss b/src/appointments/details/appointment-details.scss new file mode 100644 index 0000000..951a1e1 --- /dev/null +++ b/src/appointments/details/appointment-details.scss @@ -0,0 +1,81 @@ +@use '@carbon/styles/scss/spacing'; +@use '@carbon/styles/scss/type'; +@use '@carbon/colors'; + +.appointmentDetailsContainer { + min-height: fit-content; + background-color: colors.$white; + padding: spacing.$spacing-05; + margin: spacing.$spacing-03; +} + +.title { + @include type.type-style('heading-compact-01'); + color: colors.$gray-90; +} + +.subTitle { + @include type.type-style('label-01'); + color: colors.$gray-70; +} + +.tags { + margin-top: spacing.$spacing-05; + display: flex; + column-gap: spacing.$spacing-03; + + & > div { + margin: 0; + } +} + +.patientInfoGrid { + display: grid; + grid-template-columns: 1fr 1fr 1fr; + column-gap: spacing.$spacing-05; + margin-top: spacing.$spacing-05; +} + +.gridTitle { + @include type.type-style('label-02'); + margin-bottom: spacing.$spacing-06; + color: colors.$gray-70; +} + +.label { + @include type.type-style('label-01'); + color: colors.$gray-70; +} + +.historyGrid { + display: grid; + grid-template-columns: 1fr 1fr; + row-gap: spacing.$spacing-05; +} + +.historyGridLabel { + @include type.type-style('legal-01'); + color: colors.$gray-70; +} +.historyGridCount { + @include type.type-style('heading-02'); + margin-top: spacing.$spacing-01; + color: colors.$blue-60; +} + +.historyGridCountRed { + @include type.type-style('heading-02'); + margin-top: spacing.$spacing-01; + color: colors.$red-60; +} + +.labelBold { + @include type.type-style('label-01'); + color: colors.$gray-70; + font-weight: bold; + margin-right: spacing.$spacing-02; +} + +.labelContainer { + display: flex; +} diff --git a/src/appointments/details/appointment-details.test.tsx b/src/appointments/details/appointment-details.test.tsx new file mode 100644 index 0000000..5325795 --- /dev/null +++ b/src/appointments/details/appointment-details.test.tsx @@ -0,0 +1,64 @@ +import React from "react"; +import { render } from "@testing-library/react"; +import AppointmentDetails from "./appointment-details.component"; + +const appointment = { + patientUuid: "12345", + name: "John Doe", + identifier: "1234567890", + dateTime: new Date().toISOString(), + serviceType: "Service", + provider: "Dr. Provider", + id: "1", + age: "30", + gender: "M", + providers: [], + appointmentKind: "Scheduled", + appointmentNumber: "12345", + location: "Location", + phoneNumber: "1234567890", + status: "Status", + comments: "Some comments", + dob: new Date("1992-01-01T00:00:00.000Z").toISOString(), + serviceUuid: "12345", +}; + +jest.mock("../../hooks/usePatientAppointmentHistory", () => ({ + usePatientAppointmentHistory: () => ({ + appointmentsCount: { + completedAppointments: 1, + missedAppointments: 2, + cancelledAppointments: 3, + upcomingAppointments: 4, + }, + }), +})); + +test("renders appointment details correctly", () => { + const { getByText } = render( + , + ); + expect(getByText("Service")).toBeInTheDocument(); + expect(getByText("1992-01-01T00:00:00.000Z")).toBeInTheDocument(); + expect(getByText("Patient name :")).toBeInTheDocument(); + expect(getByText("John Doe")).toBeInTheDocument(); + expect(getByText("Age :")).toBeInTheDocument(); + expect(getByText("30")).toBeInTheDocument(); + expect(getByText("Gender :")).toBeInTheDocument(); + expect(getByText("Male")).toBeInTheDocument(); + expect(getByText("Dob :")).toBeInTheDocument(); + expect(getByText("1992-01-01T00:00:00.000Z")).toBeInTheDocument(); + expect(getByText("Phone number :")).toBeInTheDocument(); + expect(getByText("1234567890")).toBeInTheDocument(); + expect(getByText("Appointment Notes")).toBeInTheDocument(); + expect(getByText("Some comments")).toBeInTheDocument(); + expect(getByText("Appointment History")).toBeInTheDocument(); + expect(getByText("Completed")).toBeInTheDocument(); + expect(getByText("1")).toBeInTheDocument(); + expect(getByText("Missed")).toBeInTheDocument(); + expect(getByText("2")).toBeInTheDocument(); + expect(getByText("Cancelled")).toBeInTheDocument(); + expect(getByText("3")).toBeInTheDocument(); + expect(getByText("Upcoming")).toBeInTheDocument(); + expect(getByText("4")).toBeInTheDocument(); +}); diff --git a/src/appointments/forms/cancel-form/cancel-appointment.component.tsx b/src/appointments/forms/cancel-form/cancel-appointment.component.tsx new file mode 100644 index 0000000..c6ef214 --- /dev/null +++ b/src/appointments/forms/cancel-form/cancel-appointment.component.tsx @@ -0,0 +1,114 @@ +import React, { useState, useEffect } from "react"; +import { useTranslation } from "react-i18next"; +import { Button, Layer, TextArea } from "@carbon/react"; +import { + useSession, + showToast, + showNotification, + ExtensionSlot, + usePatient, +} from "@openmrs/esm-framework"; +import { cancelAppointment } from "../forms.resource"; +import { useSWRConfig } from "swr"; +import { useAppointmentDate } from "../../../helpers"; +import { closeOverlay } from "../../../hooks/useOverlay"; +import { type MappedAppointment } from "../../../types"; +import styles from "./cancel-appointment.scss"; + +interface CancelAppointmentProps { + appointment: MappedAppointment; +} +const CancelAppointment: React.FC = ({ + appointment, +}) => { + const { t } = useTranslation(); + const { mutate } = useSWRConfig(); + const { patient } = usePatient(appointment.patientUuid); + const session = useSession(); + const [selectedLocation, setSelectedLocation] = useState( + appointment.location, + ); + const [reason, setReason] = useState(""); + const { currentAppointmentDate } = useAppointmentDate(); + const [isSubmitting, setIsSubmitting] = useState(false); + + useEffect(() => { + if (selectedLocation && session?.sessionLocation?.uuid) { + setSelectedLocation(session?.sessionLocation?.uuid); + } + }, [selectedLocation, session]); + + const handleSubmit = async () => { + setIsSubmitting(true); + const { status } = await cancelAppointment("Cancelled", appointment.id); + if (status === 200) { + showToast({ + critical: true, + kind: "success", + description: t( + "cancelledSuccessfully", + "It has been cancelled successfully", + ), + title: t("appointmentCancelled", "Appointment cancelled"), + }); + mutate( + `/ws/rest/v1/appointment/appointmentStatus?forDate=${currentAppointmentDate}&status=Scheduled`, + ); + mutate( + `/ws/rest/v1/appointment/appointmentStatus?forDate=${currentAppointmentDate}&status=Cancelled`, + ); + closeOverlay(); + } else { + showNotification({ + title: t("appointmentCancelError", "Error cancelling appointment"), + kind: "error", + critical: true, + description: t( + "errorCancellingAppointment", + "Error cancelling the appointment", + ), + }); + setIsSubmitting(false); + } + }; + + return ( +
+ {patient && ( + + )} + + +