import { isValidDate } from '~/lib/date' import { medicalRoles } from '~/const/common/role' export interface EncounterItem { id: string title: string classCode?: string[] unit?: string afterId?: string component?: any props?: Record } export interface EncounterProps { classCode: 'ambulatory' | 'emergency' | 'inpatient' | 'outpatient' subClassCode: 'reg' | 'rehab' | 'chemo' | 'emg' | 'eon' | 'op' | 'icu' | 'hcu' | 'vk' } export interface EncounterListData { encounter?: any status?: any medicalAssessment?: any medicalAssessmentRehab: any functionAssessment?: any protocolTheraphy?: any protocolChemotherapy?: any medicineProtocolChemotherapy?: any consultation?: any letterOfControl?: any } const StatusAsync = defineAsyncComponent(() => import('~/components/content/encounter/status.vue')) const EarlyMedicalRehabAssessmentListAsync = defineAsyncComponent(() => import('~/components/content/soapi/entry.vue')) const EarlyMedicalAssesmentListAsync = defineAsyncComponent(() => import('~/components/content/soapi/entry.vue')) const initialNursesAssessmentAsync = defineAsyncComponent(() => import('~/components/content/initial-nursing/entry.vue')) const AssesmentFunctionListAsync = defineAsyncComponent(() => import('~/components/content/soapi/entry.vue')) const ChemoProtocolListAsync = defineAsyncComponent(() => import('~/components/app/chemotherapy/list.protocol.vue')) const ChemoMedicineProtocolListAsync = defineAsyncComponent(() => import('~/components/app/chemotherapy/list.medicine.vue')) const DeviceOrderAsync = defineAsyncComponent(() => import('~/components/content/device-order/main.vue')) const PrescriptionAsync = defineAsyncComponent(() => import('~/components/content/prescription/main.vue')) const CpLabOrderAsync = defineAsyncComponent(() => import('~/components/content/cp-lab-order/main.vue')) const ProcedureRoomOrderAsync = defineAsyncComponent(() => import('~/components/content/procedure-room-order/main.vue')) const MicroLabOrderAsync = defineAsyncComponent(() => import('~/components/content/micro-lab-order/main.vue')) const ApLabOrderAsync = defineAsyncComponent(() => import('~/components/content/ap-lab-order/main.vue')) const CprjAsync = defineAsyncComponent(() => import('~/components/content/cprj/entry.vue')) const RadiologyAsync = defineAsyncComponent(() => import('~/components/content/radiology-order/main.vue')) const ConsultationAsync = defineAsyncComponent(() => import('~/components/content/consultation/list.vue')) const DocUploadListAsync = defineAsyncComponent(() => import('~/components/content/document-upload/list.vue')) const GeneralConsentListAsync = defineAsyncComponent(() => import('~/components/content/general-consent/entry.vue')) const ResumeListAsync = defineAsyncComponent(() => import('~/components/content/resume/list.vue')) const ControlLetterListAsync = defineAsyncComponent(() => import('~/components/content/control-letter/list.vue')) const KfrListAsync = defineAsyncComponent(() => import('~/components/content/kfr/list.vue')) const PrbListAsync = defineAsyncComponent(() => import('~/components/content/prb/list.vue')) const SurgeryReportListAsync = defineAsyncComponent(() => import('~/components/content/surgery-report/list.vue')) const VaccineDataListAsync = defineAsyncComponent(() => import('~/components/content/vaccine-data/list.vue')) const AssessmentEducationEntryAsync = defineAsyncComponent( () => import('~/components/content/assessment-education/entry.vue'), ) const SummaryMedicAsync = defineAsyncComponent(() => import('~/components/content/summary-medic/entry.vue')) const ActionReportEntryAsync = defineAsyncComponent(() => import('~/components/content/action-report/entry.vue')) const defaultKeys: Record = { status: { id: 'status', title: 'Status Masuk/Keluar', classCode: ['ambulatory', 'emergency', 'inpatient'], unit: 'all', }, earlyMedicalAssessment: { id: 'early-medical-assessment', title: 'Pengkajian Awal Medis', classCode: ['ambulatory', 'emergency', 'inpatient'], unit: 'all', }, earlyMedicalRehabAssessment: { id: 'rehab-medical-assessment', title: 'Pengkajian Awal Medis Rehabilitasi Medis', classCode: ['ambulatory'], unit: 'rehab', afterId: 'early-medical-assessment', }, initialNursesAssessment: { id: 'early-nurse-assessment', title: 'Kajian Awal Keperawatan', classCode: ['ambulatory', 'emergency', 'inpatient'], unit: 'all', }, kfr: { id: 'kfr', title: 'KFR', classCode: ['ambulatory', 'emergency', 'inpatient'], unit: 'all', }, chemotherapyProtocol: { id: 'chemotherapy-protocol', title: 'Protokol Kemoterapi', classCode: ['ambulatory'], unit: 'chemo', afterId: 'early-medical-assessment', }, chemotherapyMedicine: { id: 'chemotherapy-medicine', title: 'Protokol Obat Kemoterapi', classCode: ['ambulatory'], unit: 'chemo', afterId: 'chemotherapy-protocol', }, educationAssessment: { id: 'education-assessment', title: 'Asesmen Kebutuhan Edukasi', classCode: ['ambulatory', 'emergency', 'inpatient'], unit: 'all', }, generalConsent: { id: 'general-consent', title: 'General Consent', classCode: ['ambulatory', 'emergency', 'inpatient'], unit: 'all', }, patientAmbNote: { id: 'patient-amb-note', title: 'CPRJ', classCode: ['ambulatory', 'emergency'], unit: 'all', }, patientDevNote: { id: 'patient-dev-note', title: 'CPP', classCode: ['inpatient'], unit: 'all', }, prescription: { id: 'prescription', title: 'Order Obat', classCode: ['ambulatory', 'emergency', 'inpatient'], unit: 'all', }, deviceOrder: { id: 'device-order', title: 'Order Alkes', classCode: ['ambulatory', 'emergency', 'inpatient'], unit: 'all', }, radiologyOrder: { id: 'radiology-order', title: 'Order Radiologi', classCode: ['ambulatory', 'emergency', 'inpatient'], unit: 'all', }, cpLabOrder: { id: 'cp-lab-order', title: 'Order Lab PK', classCode: ['ambulatory', 'emergency', 'inpatient'], unit: 'all', }, microLabOrder: { id: 'micro-lab-order', title: 'Order Lab Mikro', classCode: ['ambulatory', 'emergency', 'inpatient'], unit: 'all', }, apLabOrder: { id: 'ap-lab-order', title: 'Order Lab PA', classCode: ['ambulatory', 'emergency', 'inpatient'], unit: 'all', }, procedureRoomOrder: { id: 'procedure-room-order', title: 'Order Ruang Tindakan', classCode: ['ambulatory', 'emergency', 'inpatient'], unit: 'all', }, mcuResult: { id: 'mcu-result', title: 'Hasil Penunjang', classCode: ['ambulatory', 'emergency', 'inpatient'], unit: 'all', }, actionReport: { id: 'action-report', title: 'Laporan Tindakan', classCode: ['ambulatory', 'emergency', 'inpatient'], unit: 'all', }, surgeryReport: { id: 'surgery-report', title: 'Laporan Operasi', classCode: ['ambulatory', 'emergency', 'inpatient'], unit: 'all', }, vaccineData: { id: 'vaccine-data', title: 'Data Vaksin', classCode: ['ambulatory', 'emergency', 'inpatient'], unit: 'all', }, consultation: { id: 'consultation', title: 'Konsultasi', classCode: ['ambulatory', 'emergency', 'inpatient'], unit: 'all', }, controlLetter: { id: 'control-letter', title: 'Surat Kontrol', classCode: ['ambulatory', 'emergency', 'inpatient'], unit: 'all', }, inpatientLetter: { id: 'inpatient-letter', title: 'SPRI', classCode: ['ambulatory', 'emergency'], unit: 'all', }, refBack: { id: 'reference-back', title: 'PRB', classCode: ['ambulatory', 'emergency', 'inpatient'], unit: 'all', }, screening: { id: 'screening', title: 'Skrinning MPP', classCode: ['ambulatory', 'emergency', 'inpatient'], unit: 'all', }, supportingDocument: { id: 'supporting-document', title: 'Upload Dokumen Pendukung', classCode: ['ambulatory'], unit: 'all', }, resume: { id: 'resume', title: 'Resume', classCode: ['ambulatory', 'emergency', 'inpatient'], unit: 'all', }, ambResume: { id: 'amb-resume', title: 'Resume Medis Rawat Jalan', classCode: ['ambulatory', 'emergency', 'inpatient'], unit: 'all', }, priceList: { id: 'price-list', title: 'Tarif Tindakan', classCode: ['ambulatory', 'emergency', 'inpatient'], unit: 'all', }, // initialNursesAssessment: { // id: 'initial-nursing-study', // title: 'Kajian Awal Keperawatan', // classCode: ['ambulatory', 'emergency', 'inpatient'], // unit: 'all', // }, summaryMedic: { id: 'summary-medic', title: 'Profil Ringkasan Medis', classCode: ['ambulatory', 'emergency', 'inpatient'], unit: 'all', }, } export function getItemsByClassCode(classCode: string, items: EncounterItem[]) { return items.filter((item) => item.classCode?.includes(classCode)) } export function getItemsByUnit(unit: string, items: EncounterItem[]) { return items.filter((item) => item.unit === unit) } export function getItemsByIds(ids: string[], items: EncounterItem[]) { return items.filter((item) => ids.includes(item.id)) } export function getIndexById(id: string, items: EncounterItem[]) { return items.findIndex((item) => item.id === id) } export const getItemsAll = (classCode: string, unit: string, items: EncounterItem[]) => { const prevItems = [...items] let updateItems = getItemsByClassCode(classCode, prevItems) updateItems = getItemsByUnit(unit, updateItems) return updateItems } export function insertItemByAfterId(id: string, items: EncounterItem[], newItem: EncounterItem) { const index = getIndexById(id, items) if (index > -1) { items.splice(index + 1, 0, newItem) } } export function injectComponents(id: string | number, data: EncounterListData, meta: EncounterListData) { const currentKeys = { ...defaultKeys } if (currentKeys?.status) { currentKeys.status['component'] = StatusAsync currentKeys.status['props'] = { encounter: data?.encounter } } if (currentKeys?.earlyMedicalAssessment) { currentKeys.earlyMedicalAssessment['component'] = EarlyMedicalAssesmentListAsync currentKeys.earlyMedicalAssessment['props'] = { encounter: data?.encounter, type: 'early-medic', label: currentKeys.earlyMedicalAssessment['title'], } } if (currentKeys?.earlyMedicalRehabAssessment) { currentKeys.earlyMedicalRehabAssessment['component'] = EarlyMedicalRehabAssessmentListAsync currentKeys.earlyMedicalRehabAssessment['props'] = { encounter: data?.encounter, type: 'early-rehab', label: currentKeys.earlyMedicalRehabAssessment['title'], } } if (currentKeys?.functionAssessment) { currentKeys.functionAssessment['component'] = AssesmentFunctionListAsync currentKeys.functionAssessment['props'] = { encounter: data?.encounter, type: 'function', label: currentKeys.functionAssessment['title'], } } if (currentKeys?.therapyProtocol) { // TODO: add component for therapyProtocol currentKeys.therapyProtocol['component'] = null currentKeys.therapyProtocol['props'] = { data: data?.encounter, paginationMeta: meta?.protocolTheraphy, } } if (currentKeys?.chemotherapyProtocol) { currentKeys.chemotherapyProtocol['component'] = ChemoProtocolListAsync currentKeys.chemotherapyProtocol['props'] = { data: data?.encounter, paginationMeta: meta?.protocolChemotherapy, } } if (currentKeys?.chemotherapyMedicine) { currentKeys.chemotherapyMedicine['component'] = ChemoMedicineProtocolListAsync currentKeys.chemotherapyMedicine['props'] = { data: data?.encounter, paginationMeta: meta?.medicineProtocolChemotherapy, } } if (currentKeys?.educationAssessment) { currentKeys.educationAssessment['component'] = AssessmentEducationEntryAsync currentKeys.educationAssessment['props'] = { encounter: data?.encounter, type: 'education-assessment', label: currentKeys.educationAssessment['title'], } } if (currentKeys?.generalConsent) { currentKeys.generalConsent['component'] = GeneralConsentListAsync currentKeys.generalConsent['props'] = { encounter_id: id } } if (currentKeys?.patientAmbNote) { currentKeys.patientAmbNote['component'] = CprjAsync currentKeys.patientAmbNote['props'] = { encounter_id: id } } if (currentKeys?.prescription) { currentKeys.prescription['component'] = PrescriptionAsync currentKeys.prescription['props'] = { encounter_id: id } } if (currentKeys?.deviceOrder) { currentKeys.deviceOrder['component'] = DeviceOrderAsync currentKeys.deviceOrder['props'] = { encounter_id: id } } if (currentKeys?.radiologyOrder) { currentKeys.radiologyOrder['component'] = RadiologyAsync currentKeys.radiologyOrder['props'] = { encounter_id: id } } if (currentKeys?.cpLabOrder) { currentKeys.cpLabOrder['component'] = CpLabOrderAsync currentKeys.cpLabOrder['props'] = { encounter_id: id } } if (currentKeys?.microLabOrder) { currentKeys.microLabOrder['component'] = MicroLabOrderAsync currentKeys.microLabOrder['props'] = { encounter_id: id } } if (currentKeys?.apLabOrder) { // TODO: add component for apLabOrder currentKeys.apLabOrder['component'] = ApLabOrderAsync currentKeys.apLabOrder['props'] = { encounter_id: id } } if (currentKeys?.procedureRoomOrder) { currentKeys.procedureRoomOrder['component'] = ProcedureRoomOrderAsync currentKeys.procedureRoomOrder['props'] = { encounter_id: id } } if (currentKeys?.mcuResult) { // TODO: add component for mcuResult currentKeys.mcuResult['component'] = null currentKeys.mcuResult['props'] = { encounter_id: id } } if (currentKeys?.consultation) { currentKeys.consultation['component'] = ConsultationAsync currentKeys.consultation['props'] = { encounter: data?.encounter } } if (currentKeys?.resume) { currentKeys.resume['component'] = ResumeListAsync currentKeys.resume['props'] = { encounter_id: id } } if (currentKeys?.controlLetter) { currentKeys.controlLetter['component'] = ControlLetterListAsync currentKeys.controlLetter['props'] = { encounter: data?.encounter } } if (currentKeys?.refBack) { currentKeys.refBack['component'] = PrbListAsync currentKeys.refBack['props'] = { encounter: data?.encounter } } if (currentKeys?.kfr) { currentKeys.kfr['component'] = KfrListAsync currentKeys.kfr['props'] = { encounter: data?.encounter } } if (currentKeys?.screening) { // TODO: add component for screening currentKeys.screening['component'] = null currentKeys.screening['props'] = { encounter_id: id } } if (currentKeys?.surgeryReport) { currentKeys.surgeryReport['component'] = SurgeryReportListAsync currentKeys.surgeryReport['props'] = { encounter: data?.encounter } } if (currentKeys?.vaccineData) { currentKeys.vaccineData['component'] = VaccineDataListAsync currentKeys.vaccineData['props'] = { encounter: data?.encounter } } if (currentKeys?.supportingDocument) { currentKeys.supportingDocument['component'] = DocUploadListAsync currentKeys.supportingDocument['props'] = { encounter_id: id } } if (currentKeys?.priceList) { // TODO: add component for priceList currentKeys.priceList['component'] = null currentKeys.priceList['props'] = { encounter_id: id } } if (currentKeys?.initialNursesAssessment) { currentKeys.initialNursesAssessment['component'] = initialNursesAssessmentAsync currentKeys.initialNursesAssessment['props'] = { encounter: data?.encounter } } if (currentKeys?.actionReport) { currentKeys.actionReport['component'] = ActionReportEntryAsync currentKeys.actionReport['props'] = { encounter: data?.encounter, type: 'action-report', label: currentKeys.actionReport['title'], } } return currentKeys } export function mergeArrayAt(arraysOne: T[], arraysTwo: T[] | T, deleteCount = 0): T[] { const prevItems = arraysOne.slice() if (!prevItems) return prevItems const nextItems = Array.isArray(arraysTwo) ? arraysTwo : [arraysTwo] if (nextItems.length === 0) return prevItems // determine insertion position using the first item's `id` if available const firstId = (nextItems[0] as any)?.afterId || (prevItems[0] as any)?.id let pos = prevItems.length if (typeof firstId === 'string') { const index = prevItems.findIndex((item: any) => item.id === firstId) pos = index < 0 ? Math.max(prevItems.length + index, 0) : Math.min(index, prevItems.length) } prevItems.splice(pos, deleteCount, ...nextItems) return prevItems } // Function to map API response to Encounter structure export function mapResponseToEncounter(result: any): any { if (!result) return null // Check if patient and patient.person exist (minimal validation) if (!result.patient || !result.patient.person) { return null } const mapped: any = { id: result.id || 0, patient_id: result.patient_id || result.patient?.id || 0, patient: { id: result.patient?.id || 0, number: result.patient?.number || '', person: { id: result.patient?.person?.id || 0, name: result.patient?.person?.name || '', birthDate: result.patient?.person?.birthDate || null, gender_code: result.patient?.person?.gender_code || '', residentIdentityNumber: result.patient?.person?.residentIdentityNumber || null, frontTitle: result.patient?.person?.frontTitle || '', endTitle: result.patient?.person?.endTitle || '', addresses: result.patient?.person?.addresses || [], }, }, registeredAt: result.registeredAt || result.patient?.registeredAt || null, class_code: result.class_code || '', unit_id: result.unit_id || 0, unit: result.unit || null, specialist_id: result.specialist_id || null, subspecialist_id: result.subspecialist_id || null, visitDate: isValidDate(result.visitDate) ? result.visitDate : result.registeredAt || result.patient?.registeredAt || null, adm_employee_id: result.adm_employee_id || 0, adm_employee: result.adm_employee || null, responsible_nurse_id: result.responsible_nurse_id || null, responsible_nurse: result.responsible_nurse || null, appointment_doctor_id: result.appointment_doctor_id || null, responsible_doctor_id: result.responsible_doctor_id || null, appointment_doctor: result.appointment_doctor || null, responsible_doctor: result.responsible_doctor || null, refSource_name: result.refSource_name || null, appointment_id: result.appointment_id || null, earlyEducation: result.earlyEducation || null, medicalDischargeEducation: result.medicalDischargeEducation || '', admDischargeEducation: result.admDischargeEducation || null, discharge_method_code: result.discharge_method_code || null, discharge_reason: result.dischargeReason || result.discharge_reason || null, discharge_date: result.discharge_date || null, status_code: result.status_code || '', // Payment related fields paymentMethod_code: result.paymentMethod_code && result.paymentMethod_code.trim() !== '' ? result.paymentMethod_code : null, trx_number: result.trx_number || null, member_number: result.member_number || null, ref_number: result.ref_number || null, } return mapped } export function getMenuItems(id: string | number, props: any, user: any, data: EncounterListData, meta: any) { // const normalClassCode = props.classCode === 'ambulatory' ? 'outpatient' : props.classCode const normalClassCode = props.classCode === 'ambulatory' ? 'ambulatory' : props.classCode const currentKeys = injectComponents(id, data, meta) const defaultItems: EncounterItem[] = Object.values(currentKeys) const listItemsForOutpatientRehab = mergeArrayAt( getItemsAll('ambulatory', 'all', defaultItems), getItemsAll('ambulatory', 'rehab', defaultItems), ) const listItemsForOutpatientChemo = mergeArrayAt( getItemsAll('ambulatory', 'all', defaultItems), getItemsAll('ambulatory', 'chemo', defaultItems), ) const listItems: Record>> = { 'installation|ambulatory': { 'unit|rehab': { items: listItemsForOutpatientRehab, roles: medicalRoles, }, 'unit|chemo': { items: listItemsForOutpatientChemo, roles: medicalRoles, }, all: getItemsAll('ambulatory', 'all', defaultItems), }, 'installation|emergency': { all: getItemsAll('emergency', 'all', defaultItems), }, 'installation|inpatient': { all: getItemsAll('inpatient', 'all', defaultItems), }, } const currentListItems = listItems[`installation|${normalClassCode}`] if (!currentListItems) return [] const unitCode = user?.unit_code ? `unit|${user.unit_code}` : 'all' const currentUnitItems: any = currentListItems[`${unitCode}`] if (!currentUnitItems) return [] let menus = [] if (currentUnitItems.roles && currentUnitItems.roles && user.activeRole in currentUnitItems.roles) { menus = [...currentUnitItems.items] } else { menus = unitCode !== 'all' && currentUnitItems?.items ? [...currentUnitItems.items] : [...currentUnitItems] } return menus }