diff --git a/src/adapters/encounter-diagnosis-adapter.test.ts b/src/adapters/encounter-diagnosis-adapter.test.ts new file mode 100644 index 000000000..f9dd460e5 --- /dev/null +++ b/src/adapters/encounter-diagnosis-adapter.test.ts @@ -0,0 +1,254 @@ +import { type FormContextProps } from '../provider/form-provider'; +import { type FormField } from '../types'; +import { EncounterDiagnosisAdapter } from './encounter-diagnosis-adapter'; + +const formContext = { + methods: null, + workspaceLayout: 'maximized', + isSubmitting: false, + patient: { + id: '833db896-c1f0-11eb-8529-0242ac130003', + }, + formJson: null, + visit: null, + sessionMode: 'enter', + sessionDate: new Date(), + location: { + uuid: '41e6e516-c1f0-11eb-8529-0242ac130003', + }, + currentProvider: null, + layoutType: 'small-desktop', + domainObjectValue: { + uuid: '873455da-3ec4-453c-b565-7c1fe35426be', + obs: [], + diagnoses: [], + }, + previousDomainObjectValue: null, + processor: null, + formFields: [], + formFieldAdapters: null, + formFieldValidators: null, + customDependencies: { + patientPrograms: [], + }, + getFormField: jest.fn(), + addFormField: jest.fn(), + updateFormField: jest.fn(), + removeFormField: () => {}, + addInvalidField: jest.fn(), + removeInvalidField: jest.fn(), + setInvalidFields: jest.fn(), + setForm: jest.fn(), +} as FormContextProps; + +const field = { + label: 'Test Diagnosis', + id: 'DiagNosIS', + type: 'diagnosis', + questionOptions: { + rendering: 'repeating', + diagnosis: { + rank: 1, + isConfirmed: false, + }, + datasource: { + name: 'problem_datasource', + config: { + class: [ + '8d4918b0-c2cc-11de-8d13-0010c6dffd0f', + '8d492954-c2cc-11de-8d13-0010c6dffd0f', + '8d492b2a-c2cc-11de-8d13-0010c6dffd0f', + ], + }, + }, + }, + meta: { + submission: { + newValue: null, + }, + previousValue: null, + }, + validators: [ + { + type: 'form_field', + }, + { + type: 'default_value', + }, + ], + isHidden: false, + isRequired: false, + isDisabled: false, +} as FormField; + +const diagnoses = [ + { + uuid: '8d975f9e-e9e6-452f-be7c-0e87c047f056', + diagnosis: { + coded: { + uuid: '127133AAAAAAAAAAAAAAAAAAAAAAAAAAAAAA', + display: 'Schistosoma Mansonii Infection', + links: [], + }, + }, + condition: null, + encounter: { + uuid: '9a4b06bd-d655-414f-b9ce-69e940c337ce', + }, + certainty: 'CONFIRMED', + rank: 1, + voided: false, + display: 'Schistosoma Mansonii Infection', + patient: { + uuid: '00affa97-0010-417c-87f5-de48362de915', + display: '1000VKV - Bett Tett', + }, + formFieldNamespace: 'rfe-forms', + formFieldPath: 'rfe-forms-DiagNosIS_1', + links: [], + resourceVersion: '1.8', + }, + { + uuid: 'b2d0e95b-d2f6-49d1-a477-acc7026edbd7', + diagnosis: { + coded: { + uuid: '137329AAAAAAAAAAAAAAAAAAAAAAAAAAAAAA', + display: 'Infection due to Entamoeba Histolytica', + links: [], + }, + }, + condition: null, + encounter: { + uuid: '9a4b06bd-d655-414f-b9ce-69e940c337ce', + }, + certainty: 'PROVISIONAL', + rank: 1, + voided: false, + display: 'Infection due to Entamoeba Histolytica', + patient: { + uuid: '00affa97-0010-417c-87f5-de48362de915', + display: '1000VKV - Bett Tett', + }, + formFieldNamespace: 'rfe-forms', + formFieldPath: 'rfe-forms-DiagNosIS', + links: [], + resourceVersion: '1.8', + }, +]; + +describe('EncounterDiagnosisAdapter', () => { + it('should should handle submission of a diagnosis field', async () => { + const value = '127133AAAAAAAAAAAAAAAAAAAAAAAAAAAAAA'; + EncounterDiagnosisAdapter.transformFieldValue(field, value, formContext); + expect(field.meta.submission.newValue).toEqual({ + patient: '833db896-c1f0-11eb-8529-0242ac130003', + condition: null, + diagnosis: { + coded: '127133AAAAAAAAAAAAAAAAAAAAAAAAAAAAAA', + }, + certainty: 'PROVISIONAL', + rank: 1, + formFieldPath: 'rfe-forms-DiagNosIS', + formFieldNamespace: 'rfe-forms', + }); + }); + + it('should get initial value for the diagnosis', async () => { + formContext.domainObjectValue.diagnoses.push(...diagnoses); + const diagnosis = await EncounterDiagnosisAdapter.getInitialValue(field, null, formContext); + expect(diagnosis).toEqual('137329AAAAAAAAAAAAAAAAAAAAAAAAAAAAAA'); + }); + + it('should return null for getPreviousValue', async () => { + const previousValue = await EncounterDiagnosisAdapter.getPreviousValue(field, null, formContext); + expect(previousValue).toBeNull(); + }); + + it('should execute tearDown without issues', () => { + expect(() => EncounterDiagnosisAdapter.tearDown()).not.toThrow(); + }); + + it('should edit a diagnosis value', () => { + formContext.sessionMode = 'edit'; + + const value = '128138AAAAAAAAAAAAAAAAAAAAAAAAAAAAAA'; + field.meta = { + previousValue: { + patient: '833db896-c1f0-11eb-8529-0242ac130003', + condition: null, + diagnosis: { + coded: '127133AAAAAAAAAAAAAAAAAAAAAAAAAAAAAA', + }, + certainty: 'PROVISIONAL', + rank: 1, + formFieldPath: 'rfe-forms-DiagNosIS', + formFieldNamespace: 'rfe-forms', + uuid: '0e20bb67-5d7f-41e0-96a1-751efc21a96f', + }, + initialValue: { + omrsObject: { + uuid: '0e20bb67-5d7f-41e0-96a1-751efc21a96f', + diagnosis: { + coded: { + uuid: '127133AAAAAAAAAAAAAAAAAAAAAAAAAAAAAA', + display: 'Schistosoma Mansonii Infection', + }, + }, + }, + refinedValue: null, + }, + }; + + EncounterDiagnosisAdapter.transformFieldValue(field, value, formContext); + expect(field.meta.submission.newValue).toEqual({ + patient: '833db896-c1f0-11eb-8529-0242ac130003', + condition: null, + diagnosis: { + coded: '128138AAAAAAAAAAAAAAAAAAAAAAAAAAAAAA', + }, + certainty: 'PROVISIONAL', + rank: 1, + formFieldPath: 'rfe-forms-DiagNosIS', + formFieldNamespace: 'rfe-forms', + uuid: '0e20bb67-5d7f-41e0-96a1-751efc21a96f', + }); + expect(field.meta.submission.voidedValue).toBe(undefined); + }); + + it('should handle deleting a diagnosis', () => { + formContext.sessionMode = 'edit'; + + const value = null; + field.meta = { + previousValue: { + patient: '833db896-c1f0-11eb-8529-0242ac130003', + condition: null, + diagnosis: { + coded: '127133AAAAAAAAAAAAAAAAAAAAAAAAAAAAAA', + }, + certainty: 'PROVISIONAL', + rank: 1, + formFieldPath: 'rfe-forms-DiagNosIS', + formFieldNamespace: 'rfe-forms', + uuid: '0e20bb67-5d7f-41e0-96a1-751efc21a96f', + }, + initialValue: { + omrsObject: { + uuid: '0e20bb67-5d7f-41e0-96a1-751efc21a96f', + diagnosis: { + coded: { + uuid: '127133AAAAAAAAAAAAAAAAAAAAAAAAAAAAAA', + display: 'Schistosoma Mansonii Infection', + }, + }, + }, + refinedValue: null, + }, + repeat: { + wasDeleted: true, + }, + }; + EncounterDiagnosisAdapter.transformFieldValue(field, value, formContext); + expect(field.meta.submission.voidedValue).toEqual({ uuid: '0e20bb67-5d7f-41e0-96a1-751efc21a96f', voided: true }); + }); +}); diff --git a/src/adapters/encounter-diagnosis-adapter.ts b/src/adapters/encounter-diagnosis-adapter.ts new file mode 100644 index 000000000..1719a8317 --- /dev/null +++ b/src/adapters/encounter-diagnosis-adapter.ts @@ -0,0 +1,95 @@ +import { type OpenmrsResource } from '@openmrs/esm-framework'; +import { type DiagnosisPayload, type FormFieldValueAdapter, type FormProcessorContextProps } from '../types'; +import { type FormContextProps } from '../provider/form-provider'; +import { type OpenmrsEncounter, type FormField } from '../types'; +import { clearSubmission, gracefullySetSubmission } from '../utils/common-utils'; + +export let assignedDiagnosesIds: string[] = []; + +export const EncounterDiagnosisAdapter: FormFieldValueAdapter = { + transformFieldValue: function (field: FormField, value: any, context: FormContextProps) { + if (context.sessionMode == 'edit' && field.meta?.previousValue?.uuid) { + return editDiagnosis(value, field, context.patient.id); + } + const newValue = constructNewDiagnosis(value, field, context.patient.id); + gracefullySetSubmission(field, newValue, null); + return newValue; + }, + getInitialValue: function ( + field: FormField, + sourceObject: OpenmrsResource, + context: FormProcessorContextProps, + ): Promise { + const encounter = sourceObject ?? (context.domainObjectValue as OpenmrsEncounter); + const matchedDiagnosis = encounter.diagnoses.find( + (diagnosis) => diagnosis.formFieldPath === `rfe-forms-${field.id}`, + ); + + if (matchedDiagnosis) { + field.meta = { ...(field.meta || {}), previousValue: matchedDiagnosis }; + if (!assignedDiagnosesIds.includes(matchedDiagnosis.diagnosis?.coded?.uuid)) { + assignedDiagnosesIds.push(matchedDiagnosis.diagnosis?.coded?.uuid); + } + return matchedDiagnosis.diagnosis?.coded.uuid; + } + return null; + }, + getPreviousValue: function ( + field: FormField, + sourceObject: OpenmrsResource, + context: FormProcessorContextProps, + ): Promise { + return null; + }, + getDisplayValue: (field: FormField, value: any) => { + return field.questionOptions.answers?.find((option) => option.concept == value)?.label || value; + }, + tearDown: function (): void { + assignedDiagnosesIds = []; + }, +}; + +const constructNewDiagnosis = (value: any, field: FormField, patientUuid: string, uuid?: string) => { + if (!value) { + return null; + } + const diagnosis: DiagnosisPayload = { + patient: patientUuid, + condition: null, + diagnosis: { + coded: value, + }, + certainty: field.questionOptions?.diagnosis?.isConfirmed ? 'CONFIRMED' : 'PROVISIONAL', + rank: field.questionOptions.diagnosis?.rank ?? 1, // rank 1 denotes a diagnosis is primary, else secondary + formFieldPath: `rfe-forms-${field.id}`, + formFieldNamespace: 'rfe-forms', + }; + + if (uuid && uuid.trim() !== '') { + diagnosis.uuid = uuid; + } + + return diagnosis; +}; + +function editDiagnosis(newEncounterDiagnosis: any, field: FormField, patientUuid: string) { + if (newEncounterDiagnosis === field.meta.previousValue?.diagnosis?.coded?.uuid && !field.meta.repeat?.wasDeleted) { + clearSubmission(field); + return null; + } + + //the field has been deleted + if (field.meta.repeat?.wasDeleted) { + const voided = { + uuid: field.meta.previousValue?.uuid, + voided: true, + }; + gracefullySetSubmission(field, constructNewDiagnosis(newEncounterDiagnosis, field, null), voided); + return field.meta.submission.newValue || null; + } else { + const oldDiagnosis = field.meta.initialValue?.omrsObject as OpenmrsResource; + const newValue = constructNewDiagnosis(newEncounterDiagnosis, field, patientUuid, oldDiagnosis.uuid); + gracefullySetSubmission(field, newValue, null); + return newValue; + } +} diff --git a/src/components/repeat/repeat.component.tsx b/src/components/repeat/repeat.component.tsx index 26552f505..f8cbf3984 100644 --- a/src/components/repeat/repeat.component.tsx +++ b/src/components/repeat/repeat.component.tsx @@ -15,6 +15,7 @@ import { useFormFactory } from '../../provider/form-factory-provider'; const renderingByTypeMap: Record = { obsGroup: 'group', testOrder: 'select', + diagnosis: 'ui-select-extended', }; const Repeat: React.FC = ({ field }) => { diff --git a/src/constants.ts b/src/constants.ts index 567569286..88ee7c866 100644 --- a/src/constants.ts +++ b/src/constants.ts @@ -5,6 +5,7 @@ export const encounterRepresentation = 'custom:(uuid,encounterDatetime,encounterType:(uuid,name,description),location:(uuid,name),' + 'patient:(uuid,display),encounterProviders:(uuid,provider:(uuid,name),encounterRole:(uuid,name)),' + 'orders:(uuid,display,concept:(uuid,display),voided),' + + 'diagnoses:(uuid,certainty,condition,formFieldPath,formFieldNamespace,display,rank,voided,diagnosis:(coded:(uuid,display))),' + 'obs:(uuid,obsDatetime,comment,voided,groupMembers,formFieldNamespace,formFieldPath,concept:(uuid,name:(uuid,name)),value:(uuid,name:(uuid,name),' + 'names:(uuid,conceptNameType,name))))'; export const FormsStore = 'forms-engine-store'; diff --git a/src/processors/encounter/encounter-form-processor.ts b/src/processors/encounter/encounter-form-processor.ts index e0b645fbc..75c351673 100644 --- a/src/processors/encounter/encounter-form-processor.ts +++ b/src/processors/encounter/encounter-form-processor.ts @@ -162,11 +162,21 @@ export class EncounterFormProcessor extends FormProcessor { // save encounter try { const { data: savedEncounter } = await saveEncounter(abortController, encounter, encounter.uuid); - const saveOrders = savedEncounter.orders.map((order) => order.orderNumber); - if (saveOrders.length) { + const savedOrders = savedEncounter.orders.map((order) => order.orderNumber); + const savedDiagnoses = savedEncounter.diagnoses.map((diagnosis) => diagnosis.display); + if (savedOrders.length) { showSnackbar({ title: translateFn('ordersSaved', 'Order(s) saved successfully'), - subtitle: saveOrders.join(', '), + subtitle: savedOrders.join(', '), + kind: 'success', + isLowContrast: true, + }); + } + // handle diagnoses + if (savedDiagnoses.length) { + showSnackbar({ + title: translateFn('diagnosisSaved', 'Diagnosis(es) saved successfully'), + subtitle: savedDiagnoses.join(', '), kind: 'success', isLowContrast: true, }); diff --git a/src/processors/encounter/encounter-processor-helper.ts b/src/processors/encounter/encounter-processor-helper.ts index e7dd7e9b7..3d886efc3 100644 --- a/src/processors/encounter/encounter-processor-helper.ts +++ b/src/processors/encounter/encounter-processor-helper.ts @@ -17,6 +17,7 @@ import { DefaultValueValidator } from '../../validators/default-value-validator' import { cloneRepeatField } from '../../components/repeat/helpers'; import { assignedOrderIds } from '../../adapters/orders-adapter'; import { type OpenmrsResource } from '@openmrs/esm-framework'; +import { assignedDiagnosesIds } from '../../adapters/encounter-diagnosis-adapter'; export function prepareEncounter( context: FormContextProps, @@ -29,6 +30,7 @@ export function prepareEncounter( const obsForSubmission = []; prepareObs(obsForSubmission, formFields); const ordersForSubmission = prepareOrders(formFields); + const diagnosesForSubmission = prepareDiagnosis(formFields); let encounterForSubmission: OpenmrsEncounter = {}; if (encounter) { @@ -58,6 +60,7 @@ export function prepareEncounter( } encounterForSubmission.obs = obsForSubmission; encounterForSubmission.orders = ordersForSubmission; + encounterForSubmission.diagnoses = diagnosesForSubmission; } else { encounterForSubmission = { patient: patient.id, @@ -76,6 +79,7 @@ export function prepareEncounter( }, visit: visit?.uuid, orders: ordersForSubmission, + diagnoses: diagnosesForSubmission, }; } return encounterForSubmission; @@ -313,6 +317,43 @@ export async function hydrateRepeatField( }), ); } + + const unMappedDiagnoses = encounter.diagnoses.filter((diagnosis) => { + return ( + !assignedDiagnosesIds.includes(diagnosis?.diagnosis?.coded.uuid) && + diagnosis.formFieldPath.startsWith(`rfe-forms-${field.id}_`) + ); + }); + + const sortedDiagnoses = unMappedDiagnoses + .filter((diagnosis) => !diagnosis.voided) + .sort((a, b) => { + // Extract numeric part of formFieldPath for sorting + const numberA = parseInt(a.formFieldPath.split('_')[1], 10); + const numberB = parseInt(b.formFieldPath.split('_')[1], 10); + return numberA - numberB; // Sort numerically based on formFieldPath + }); + + if (field.type === 'diagnosis') { + return Promise.all( + sortedDiagnoses + .filter((diagnosis) => !diagnosis.voided) + .map(async (diagnosis) => { + const clone = cloneRepeatField(field, diagnosis, counter++); + initialValues[clone.id] = await formFieldAdapters[field.type].getInitialValue( + clone, + { diagnoses: [diagnosis] } as any, + context, + ); + + if (!assignedDiagnosesIds.includes(diagnosis.diagnosis.coded.uuid)) { + assignedDiagnosesIds.push(diagnosis.diagnosis.coded.uuid); + } + + return clone; + }), + ); + } // handle obs groups return Promise.all( unMappedGroups.map(async (group) => { @@ -331,3 +372,10 @@ export async function hydrateRepeatField( }), ).then((results) => results.flat()); } + +function prepareDiagnosis(fields: FormField[]) { + return fields + .filter((field) => field.type === 'diagnosis' && hasSubmission(field)) + .flatMap((field) => [field.meta.submission.newValue, field.meta.submission.voidedValue]) + .filter((d) => d); +} diff --git a/src/registry/inbuilt-components/inbuiltFieldValueAdapters.ts b/src/registry/inbuilt-components/inbuiltFieldValueAdapters.ts index 9a4c3d850..2a5632bf7 100644 --- a/src/registry/inbuilt-components/inbuiltFieldValueAdapters.ts +++ b/src/registry/inbuilt-components/inbuiltFieldValueAdapters.ts @@ -10,6 +10,7 @@ import { ObsCommentAdapter } from '../../adapters/obs-comment-adapter'; import { OrdersAdapter } from '../../adapters/orders-adapter'; import { PatientIdentifierAdapter } from '../../adapters/patient-identifier-adapter'; import { ProgramStateAdapter } from '../../adapters/program-state-adapter'; +import { EncounterDiagnosisAdapter } from '../../adapters/encounter-diagnosis-adapter'; import { type FormFieldValueAdapter } from '../../types'; export const inbuiltFieldValueAdapters: RegistryItem[] = [ @@ -61,4 +62,8 @@ export const inbuiltFieldValueAdapters: RegistryItem[] = type: 'patientIdentifier', component: PatientIdentifierAdapter, }, + { + type: 'diagnosis', + component: EncounterDiagnosisAdapter, + }, ]; diff --git a/src/transformers/default-schema-transformer.ts b/src/transformers/default-schema-transformer.ts index b39510a79..ba81b5b11 100644 --- a/src/transformers/default-schema-transformer.ts +++ b/src/transformers/default-schema-transformer.ts @@ -148,6 +148,9 @@ function transformByType(question: FormField) { ? 'date' : question.questionOptions.rendering; break; + case 'diagnosis': + handleDiagnosis(question); + break; } } @@ -276,3 +279,18 @@ function handleQuestionsWithObsComments(sectionQuestions: Array): Arr return augmentedQuestions; } + +function handleDiagnosis(question: FormField) { + if ( + ('dataSource' in question.questionOptions && question.questionOptions['dataSource'] === 'diagnoses') || + question.type === 'diagnosis' + ) { + question.questionOptions.datasource = { + name: 'problem_datasource', + config: { + class: question.questionOptions.diagnosis?.conceptClasses, + }, + }; + delete question.questionOptions['dataSource']; + } +} diff --git a/src/types/domain.ts b/src/types/domain.ts index b79ffda62..39f12d52c 100644 --- a/src/types/domain.ts +++ b/src/types/domain.ts @@ -12,6 +12,7 @@ export interface OpenmrsEncounter { visit?: OpenmrsResource | string; encounterProviders?: Array>; form?: OpenmrsFormResource; + diagnoses?: Array; } export interface OpenmrsObs extends OpenmrsResource { @@ -191,3 +192,34 @@ export interface PatientIdentifier { location?: string; preferred?: boolean; } + +export interface DiagnosisPayload { + patient: string; + condition: null; + diagnosis: { + coded: string; + }; + certainty: string; + rank: number; + formFieldNamespace?: string; + formFieldPath?: string; + uuid?: string; + encounter?: string; +} + +export interface Diagnosis { + encounter: string; + patient: string; + diagnosis: { + coded: { + uuid: string; + }; + }; + certainty: string; + rank: number; + display: string; + voided: boolean; + uuid: string; + formFieldNamespace?: string; + formFieldPath?: string; +} diff --git a/src/types/schema.ts b/src/types/schema.ts index 1eb14a931..d40e6ea5c 100644 --- a/src/types/schema.ts +++ b/src/types/schema.ts @@ -192,6 +192,11 @@ export interface FormQuestionOptions { comment?: string; orientation?: 'vertical' | 'horizontal'; shownCommentOptions?: { validators?: Array>; hide?: { hideWhenExpression: string } }; + diagnosis?: { + rank?: number; + isConfirmed?: boolean; + conceptClasses?: Array; + }; } export interface QuestionAnswerOption {