@thrustdevs/esm-procedure-orders-app 1.0.2-pre.6

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (207) hide show
  1. package/.turbo/turbo-build.log +41 -0
  2. package/README.md +7 -0
  3. package/dist/122.js +1 -0
  4. package/dist/122.js.map +1 -0
  5. package/dist/144.js +2 -0
  6. package/dist/144.js.LICENSE.txt +19 -0
  7. package/dist/144.js.map +1 -0
  8. package/dist/182.js +1 -0
  9. package/dist/182.js.map +1 -0
  10. package/dist/205.js +1 -0
  11. package/dist/205.js.map +1 -0
  12. package/dist/216.js +2 -0
  13. package/dist/216.js.LICENSE.txt +9 -0
  14. package/dist/216.js.map +1 -0
  15. package/dist/290.js +2 -0
  16. package/dist/290.js.LICENSE.txt +5 -0
  17. package/dist/290.js.map +1 -0
  18. package/dist/300.js +1 -0
  19. package/dist/341.js +2 -0
  20. package/dist/341.js.LICENSE.txt +29 -0
  21. package/dist/341.js.map +1 -0
  22. package/dist/41.js +2 -0
  23. package/dist/41.js.LICENSE.txt +9 -0
  24. package/dist/41.js.map +1 -0
  25. package/dist/470.js +1 -0
  26. package/dist/470.js.map +1 -0
  27. package/dist/495.js +1 -0
  28. package/dist/495.js.map +1 -0
  29. package/dist/506.js +2 -0
  30. package/dist/506.js.LICENSE.txt +39 -0
  31. package/dist/506.js.map +1 -0
  32. package/dist/537.js +1 -0
  33. package/dist/537.js.map +1 -0
  34. package/dist/647.js +2 -0
  35. package/dist/647.js.LICENSE.txt +5 -0
  36. package/dist/647.js.map +1 -0
  37. package/dist/7.js +1 -0
  38. package/dist/7.js.map +1 -0
  39. package/dist/719.js +2 -0
  40. package/dist/719.js.LICENSE.txt +5 -0
  41. package/dist/719.js.map +1 -0
  42. package/dist/720.js +1 -0
  43. package/dist/720.js.map +1 -0
  44. package/dist/876.js +1 -0
  45. package/dist/876.js.map +1 -0
  46. package/dist/883.js +1 -0
  47. package/dist/883.js.map +1 -0
  48. package/dist/89.js +1 -0
  49. package/dist/89.js.map +1 -0
  50. package/dist/892.js +1 -0
  51. package/dist/892.js.map +1 -0
  52. package/dist/895.js +1 -0
  53. package/dist/895.js.map +1 -0
  54. package/dist/913.js +2 -0
  55. package/dist/913.js.LICENSE.txt +32 -0
  56. package/dist/913.js.map +1 -0
  57. package/dist/924.js +1 -0
  58. package/dist/924.js.map +1 -0
  59. package/dist/943.js +1 -0
  60. package/dist/943.js.map +1 -0
  61. package/dist/99.js +2 -0
  62. package/dist/99.js.LICENSE.txt +5 -0
  63. package/dist/99.js.map +1 -0
  64. package/dist/kenyaemr-esm-procedure-orders-app.js +1 -0
  65. package/dist/kenyaemr-esm-procedure-orders-app.js.buildmanifest.json +786 -0
  66. package/dist/kenyaemr-esm-procedure-orders-app.js.map +1 -0
  67. package/dist/main.js +2 -0
  68. package/dist/main.js.LICENSE.txt +35 -0
  69. package/dist/main.js.map +1 -0
  70. package/dist/routes.json +1 -0
  71. package/jest.config.js +8 -0
  72. package/package.json +55 -0
  73. package/src/completed-list/completed-list.component.tsx +40 -0
  74. package/src/completed-list/completed-list.resource.ts +0 -0
  75. package/src/completed-list/completed-list.scss +223 -0
  76. package/src/components/create-dashboard-link.component.tsx +35 -0
  77. package/src/components/overlay/hook.ts +47 -0
  78. package/src/components/overlay/overlay.component.tsx +42 -0
  79. package/src/components/overlay/overlay.scss +92 -0
  80. package/src/config-schema.ts +78 -0
  81. package/src/constants.ts +5 -0
  82. package/src/declarations.d.ts +6 -0
  83. package/src/empty-state/empty-state-component.tsx +21 -0
  84. package/src/empty-state/empty-state.scss +23 -0
  85. package/src/form/post-procedures/post-procedure-form.component.tsx +468 -0
  86. package/src/form/post-procedures/post-procedure-form.scss +189 -0
  87. package/src/form/post-procedures/post-procedure.resource.tsx +71 -0
  88. package/src/form/procedures-orders/add-procedures-order/add-procedures-order.scss +44 -0
  89. package/src/form/procedures-orders/add-procedures-order/add-procedures-order.workspace.tsx +93 -0
  90. package/src/form/procedures-orders/add-procedures-order/procedures-order-form.component.tsx +476 -0
  91. package/src/form/procedures-orders/add-procedures-order/procedures-order-form.scss +80 -0
  92. package/src/form/procedures-orders/add-procedures-order/procedures-order.ts +17 -0
  93. package/src/form/procedures-orders/add-procedures-order/procedures-type-search.scss +115 -0
  94. package/src/form/procedures-orders/add-procedures-order/procedures-type-search.tsx +236 -0
  95. package/src/form/procedures-orders/add-procedures-order/useProceduresTypes.ts +93 -0
  96. package/src/form/procedures-orders/api.ts +282 -0
  97. package/src/form/procedures-orders/order-config.ts +48 -0
  98. package/src/form/procedures-orders/procedures-order-basket-panel/procedures-icon.component.tsx +39 -0
  99. package/src/form/procedures-orders/procedures-order-basket-panel/procedures-order-basket-item-tile.component.tsx +100 -0
  100. package/src/form/procedures-orders/procedures-order-basket-panel/procedures-order-basket-item-tile.scss +72 -0
  101. package/src/form/procedures-orders/procedures-order-basket-panel/procedures-order-basket-panel.extension.tsx +190 -0
  102. package/src/form/procedures-orders/procedures-order-basket-panel/procedures-order-basket-panel.scss +74 -0
  103. package/src/form/procedures-orders/procedures-order-basket-panel/procedures-order-basket.scss +55 -0
  104. package/src/header/procedure-header.component.tsx +32 -0
  105. package/src/header/procedure-header.scss +70 -0
  106. package/src/header/procedure-illustration.component.tsx +52 -0
  107. package/src/hooks/useOrdersWorklist.ts +70 -0
  108. package/src/hooks/useSearchGroupedResults.ts +22 -0
  109. package/src/hooks/useSearchResults.ts +39 -0
  110. package/src/index.ts +59 -0
  111. package/src/left-panel-link.tsx +40 -0
  112. package/src/not-done-list/not-done-list.component.tsx +44 -0
  113. package/src/not-done-list/not-done.scss +207 -0
  114. package/src/patient-chart/patient-procedure-order-results-table.resource.ts +43 -0
  115. package/src/patient-chart/patient-procedure-order-results.component.tsx +12 -0
  116. package/src/patient-chart/patient-procedure-order-results.resource.ts +485 -0
  117. package/src/patient-chart/patient-procedure-results.component.tsx +30 -0
  118. package/src/patient-chart/procedure-active-order/procedure-active-order-results.component.tsx +390 -0
  119. package/src/patient-chart/procedure-active-order/procedure-active-order-results.scss +78 -0
  120. package/src/patient-chart/procedure-order-referals/procedure-order-referals.component.tsx +394 -0
  121. package/src/patient-chart/procedure-order-referals/procedure-order-referals.resource.tsx +0 -0
  122. package/src/patient-chart/procedure-order-referals/procedure-order-referals.scss +78 -0
  123. package/src/patient-chart/procedure-past-test/laboratory-past-test-order-results.component.tsx +366 -0
  124. package/src/patient-chart/procedure-past-test/laboratory-past-test-order-results.scss +74 -0
  125. package/src/patient-chart/procedure-tabs/laboratory-order-tabs.component.tsx +44 -0
  126. package/src/patient-chart/procedure-tabs/laboratory-order-tabs.scss +7 -0
  127. package/src/patient-chart/procedure-workspaces/laboratory-referral.workspace.component.tsx +11 -0
  128. package/src/patient-chart/procedure-workspaces/laboratory-referral.workspace.scss +0 -0
  129. package/src/patient-chart/results-summary/print-results-summary.component.tsx +152 -0
  130. package/src/patient-chart/results-summary/print-results-summary.scss +80 -0
  131. package/src/patient-chart/results-summary/print-results-table.component.tsx +134 -0
  132. package/src/patient-chart/results-summary/results-summary.resource.tsx +174 -0
  133. package/src/patient-chart/results-summary/results-summary.scss +158 -0
  134. package/src/patient-chart/results-summary/send-email-dialog.component.tsx +59 -0
  135. package/src/patient-chart/results-summary/test-children-results.component.tsx +177 -0
  136. package/src/patient-chart/results-summary/test-print-results-table.component.tsx +105 -0
  137. package/src/patient-chart/results-summary/test-results-table.component.tsx +103 -0
  138. package/src/print/print-procedure-results.component.tsx +49 -0
  139. package/src/print/print-procedure.component.tsx +105 -0
  140. package/src/print/print-procedure.scss +98 -0
  141. package/src/procedure-tabs/completed-tab.component.tsx +12 -0
  142. package/src/procedure-tabs/not-done-tab.component.tsx +12 -0
  143. package/src/procedure-tabs/referred-tab.component.tsx +12 -0
  144. package/src/procedure-tabs/work-list-tab.component.tsx +13 -0
  145. package/src/procedure.component.tsx +24 -0
  146. package/src/procedures-ordered/_pick-procedure-request-menu.component.tsx +33 -0
  147. package/src/procedures-ordered/pick-procedure-order/add-to-worklist-dialog.component.tsx +105 -0
  148. package/src/procedures-ordered/pick-procedure-order/add-to-worklist-dialog.resource.ts +106 -0
  149. package/src/procedures-ordered/pick-procedure-order/add-to-worklist-dialog.scss +38 -0
  150. package/src/procedures-ordered/pick-procedure-request-menu.component.tsx +32 -0
  151. package/src/procedures-ordered/procedure-dialogs/add-to-worklist-dialog.component.tsx +300 -0
  152. package/src/procedures-ordered/procedure-dialogs/add-to-worklist-dialog.resource.ts +153 -0
  153. package/src/procedures-ordered/procedure-dialogs/add-to-worklist-dialog.scss +38 -0
  154. package/src/procedures-ordered/procedure-instructions/instructions.scss +24 -0
  155. package/src/procedures-ordered/procedure-instructions/procedure-instructions-menu.component.tsx +32 -0
  156. package/src/procedures-ordered/procedure-instructions/procedure-instructions.component.tsx +78 -0
  157. package/src/procedures-ordered/procedure-instructions/procedure-instructions.scss +24 -0
  158. package/src/procedures-ordered/procedure-queue.scss +211 -0
  159. package/src/procedures-ordered/procedure-tabs.component.tsx +104 -0
  160. package/src/procedures-ordered/procedure-tests/procedure-tests.component.tsx +83 -0
  161. package/src/procedures-ordered/procedure-tests/procedure-tests.resource.ts +14 -0
  162. package/src/procedures-ordered/procedure-tests/procedure-tests.scss +12 -0
  163. package/src/procedures-ordered/procedures-ordered-list.component.tsx +38 -0
  164. package/src/procedures-ordered/reject-order-dialog/reject-order-dialog.scss +14 -0
  165. package/src/procedures-ordered/reject-order-dialog/reject-procedure-order-dialog.component.tsx +98 -0
  166. package/src/procedures-ordered/reject-reason/procedure-reject-reason-menu.component.tsx +32 -0
  167. package/src/procedures-ordered/reject-reason/procedure-reject-reason.component.tsx +40 -0
  168. package/src/procedures-ordered/transition-patient-new-queue/transition-latest-queue-entry-button.component.tsx +42 -0
  169. package/src/procedures-ordered/transition-patient-new-queue/transition-latest-queue-entry-button.scss +14 -0
  170. package/src/procedures-ordered/transition-patient-new-queue/transition-latest-queue-entry-button.test.tsx +67 -0
  171. package/src/referred-procedures/referred-procedures.component.tsx +37 -0
  172. package/src/results/result-form-field.component.tsx +141 -0
  173. package/src/results/result-form.component.tsx +120 -0
  174. package/src/results/result-form.resource.ts +361 -0
  175. package/src/results/result-form.scss +22 -0
  176. package/src/root.component.tsx +16 -0
  177. package/src/routes.json +152 -0
  178. package/src/setup-tests.ts +7 -0
  179. package/src/shared/ui/common/action-button/action-button.component.tsx +68 -0
  180. package/src/shared/ui/common/action-button/action-button.scss +12 -0
  181. package/src/shared/ui/common/action-button/order-action-extension.component.tsx +21 -0
  182. package/src/shared/ui/common/grouped-orders-table.component.tsx +176 -0
  183. package/src/shared/ui/common/grouped-orders-table.scss +30 -0
  184. package/src/shared/ui/common/grouped-procedure-types.ts +47 -0
  185. package/src/shared/ui/common/list-order-details.component.tsx +171 -0
  186. package/src/shared/ui/common/list-order-details.resource.ts +41 -0
  187. package/src/shared/ui/common/list-order-details.scss +118 -0
  188. package/src/shared/ui/common/orders-date-range-picker.scss +15 -0
  189. package/src/shared/ui/common/orders-date-range-picker.tsx +38 -0
  190. package/src/summary-tiles/procedure-summary-tiles.component.tsx +36 -0
  191. package/src/summary-tiles/procedure-summary-tiles.scss +11 -0
  192. package/src/summary-tiles/procedure-summary.resource.tsx +79 -0
  193. package/src/summary-tiles/summary-tile.component.tsx +41 -0
  194. package/src/summary-tiles/summary-tile.scss +53 -0
  195. package/src/types/index.ts +661 -0
  196. package/src/types/patient-queue.ts +77 -0
  197. package/src/ui-components/overflow-menu.component.tsx +74 -0
  198. package/src/ui-components/overflow-menu.scss +39 -0
  199. package/src/utils/functions.ts +236 -0
  200. package/src/utils/orders-table/orders-data-table.component.tsx +129 -0
  201. package/src/utils/orders-table/orders-data-table.scss +50 -0
  202. package/src/work-list/work-list.component.tsx +38 -0
  203. package/src/work-list/work-list.resource.ts +26 -0
  204. package/src/work-list/work-list.scss +207 -0
  205. package/translations/en.json +141 -0
  206. package/tsconfig.json +5 -0
  207. package/webpack.config.js +1 -0
@@ -0,0 +1,366 @@
1
+ import React, { useCallback, useEffect, useMemo, useRef, useState } from 'react';
2
+ import { useTranslation } from 'react-i18next';
3
+ import styles from './laboratory-past-test-order-results.scss';
4
+ import { formatDate, parseDate, ErrorState, showModal, useConfig, usePagination } from '@openmrs/esm-framework';
5
+
6
+ import {
7
+ DataTable,
8
+ DataTableSkeleton,
9
+ Table,
10
+ TableBody,
11
+ TableCell,
12
+ TableContainer,
13
+ TableHead,
14
+ TableHeader,
15
+ TableRow,
16
+ TableToolbar,
17
+ TableToolbarContent,
18
+ TableToolbarSearch,
19
+ Layer,
20
+ Tag,
21
+ Tile,
22
+ Tooltip,
23
+ Pagination,
24
+ TableExpandHeader,
25
+ TableExpandRow,
26
+ TableExpandedRow,
27
+ Button,
28
+ IconButton,
29
+ InlineLoading,
30
+ } from '@carbon/react';
31
+
32
+ import { Printer, MailAll, Add, Checkmark, SendAlt, NotSent } from '@carbon/react/icons';
33
+
34
+ import TestsResults from '../results-summary/test-results-table.component';
35
+ import { useReactToPrint } from 'react-to-print';
36
+ import PrintResultsSummary from '../results-summary/print-results-summary.component';
37
+ import { useGetPatientByUuid } from '../../utils/functions';
38
+ import { ResourceRepresentation, type Result, getOrderColor } from '../patient-procedure-order-results.resource';
39
+ import { useLaboratoryOrderResultsPages } from '../patient-procedure-order-results-table.resource';
40
+ import { CardHeader } from '@openmrs/esm-patient-common-lib';
41
+
42
+ interface LaboratoryPastTestOrderResultsProps {
43
+ patientUuid: string;
44
+ }
45
+
46
+ interface PrintProps {
47
+ encounter: Result;
48
+ }
49
+
50
+ const LaboratoryPastTestOrderResults: React.FC<LaboratoryPastTestOrderResultsProps> = ({ patientUuid }) => {
51
+ const { t } = useTranslation();
52
+
53
+ const { enableSendingLabTestsByEmail, laboratoryEncounterTypeUuid } = useConfig();
54
+
55
+ const displayText = t('pastLaboratoryTestsDisplayTextTitle', 'Past Laboratory Tests');
56
+ const { items, tableHeaders, isLoading, isError } = useLaboratoryOrderResultsPages({
57
+ v: ResourceRepresentation.Full,
58
+ totalCount: true,
59
+ patientUuid: patientUuid,
60
+ laboratoryEncounterTypeUuid: laboratoryEncounterTypeUuid,
61
+ });
62
+ const pageSizes = [10, 20, 30, 40, 50];
63
+ const [currentPageSize, setPageSize] = useState(10);
64
+
65
+ const sortedLabRequests = useMemo(() => {
66
+ return [...items]
67
+ ?.filter((item) => item?.encounterType?.uuid === laboratoryEncounterTypeUuid)
68
+ ?.sort((a, b) => {
69
+ const dateA = new Date(a.encounterDatetime);
70
+ const dateB = new Date(b.encounterDatetime);
71
+ return dateB.getTime() - dateA.getTime();
72
+ });
73
+ }, [items, laboratoryEncounterTypeUuid]);
74
+
75
+ const [searchTerm, setSearchTerm] = useState('');
76
+ const [laboratoryOrders, setLaboratoryOrders] = useState(sortedLabRequests);
77
+ const [initialTests, setInitialTests] = useState(sortedLabRequests);
78
+
79
+ const handleChange = useCallback((event) => {
80
+ const searchText = event?.target?.value?.trim().toLowerCase();
81
+ setSearchTerm(searchText);
82
+ }, []);
83
+
84
+ useEffect(() => {
85
+ if (!searchTerm) {
86
+ setLaboratoryOrders(initialTests);
87
+ } else {
88
+ const filteredItems = initialTests.filter((item) =>
89
+ item?.orders?.some((order) => order?.concept?.display.toLowerCase().includes(searchTerm)),
90
+ );
91
+ setLaboratoryOrders(filteredItems);
92
+ }
93
+ }, [searchTerm, initialTests]);
94
+
95
+ useEffect(() => {
96
+ setInitialTests(sortedLabRequests);
97
+ }, [sortedLabRequests]);
98
+
99
+ const oneDayBeforeDate = new Date();
100
+ oneDayBeforeDate.setDate(oneDayBeforeDate.getDate() - 1);
101
+
102
+ const EmailButtonAction: React.FC = () => {
103
+ const launchSendEmailModal = useCallback(() => {
104
+ const dispose = showModal('send-email-dialog', {
105
+ closeModal: () => dispose(),
106
+ });
107
+ }, []);
108
+
109
+ return (
110
+ <Button
111
+ kind="ghost"
112
+ size="sm"
113
+ onClick={(e) => launchSendEmailModal()}
114
+ renderIcon={(props) => <MailAll size={16} {...props} />}
115
+ />
116
+ );
117
+ };
118
+
119
+ const LaunchLabRequestForm: React.FC = () => {
120
+ return (
121
+ <IconButton label="Add">
122
+ <Add />
123
+ </IconButton>
124
+ );
125
+ };
126
+ const PrintButtonAction: React.FC<PrintProps> = ({ encounter }) => {
127
+ const { patient } = useGetPatientByUuid(encounter.patient.uuid);
128
+
129
+ const [isPrinting, setIsPrinting] = useState(false);
130
+
131
+ const contentToPrintRef = useRef(null);
132
+
133
+ const onBeforeGetContentResolve = useRef(null);
134
+
135
+ useEffect(() => {
136
+ if (onBeforeGetContentResolve.current) {
137
+ onBeforeGetContentResolve.current();
138
+ }
139
+ }, [isPrinting]);
140
+
141
+ const handlePrint = useReactToPrint({
142
+ content: () => contentToPrintRef.current,
143
+ onBeforeGetContent: () =>
144
+ new Promise((resolve) => {
145
+ onBeforeGetContentResolve.current = resolve;
146
+ setIsPrinting(true);
147
+ }),
148
+ onAfterPrint: () => {
149
+ onBeforeGetContentResolve.current = null;
150
+ setIsPrinting(false);
151
+ },
152
+ });
153
+
154
+ return (
155
+ <div>
156
+ <div ref={contentToPrintRef}>
157
+ <PrintResultsSummary encounterResponse={encounter} patient={patient} />
158
+ </div>
159
+ <Tooltip align="bottom" label="Print out results">
160
+ <Button
161
+ kind="ghost"
162
+ size="sm"
163
+ onClick={handlePrint}
164
+ renderIcon={(props) => <Printer size={16} {...props} />}
165
+ />
166
+ </Tooltip>
167
+ </div>
168
+ );
169
+ };
170
+
171
+ const currentDateTime = new Date().getTime();
172
+ const twentyFourHoursAgo = currentDateTime - 24 * 60 * 60 * 1000;
173
+
174
+ const filteredPastTestOrderResults = useMemo(() => {
175
+ return laboratoryOrders?.filter((entry) => {
176
+ const entryDate = new Date(entry.encounterDatetime).getTime();
177
+ return entryDate < twentyFourHoursAgo;
178
+ });
179
+ }, [laboratoryOrders, twentyFourHoursAgo]);
180
+ const {
181
+ goTo,
182
+ results: paginatedPastTestOrderResults,
183
+ currentPage,
184
+ } = usePagination(filteredPastTestOrderResults, currentPageSize);
185
+
186
+ const tableRows = useMemo(() => {
187
+ return paginatedPastTestOrderResults?.map((entry, index) => ({
188
+ ...entry,
189
+ id: entry?.uuid,
190
+ orderDate: formatDate(parseDate(entry.encounterDatetime), {
191
+ mode: 'standard',
192
+ time: true,
193
+ }),
194
+ orders: (
195
+ <>
196
+ {entry?.orders
197
+ ?.filter((order) => order?.type === 'testorder' && order?.action === 'NEW')
198
+ .map((order) => (
199
+ <Tag
200
+ style={{
201
+ background: `${getOrderColor(order?.dateActivated, order?.dateStopped)}`,
202
+ color: 'white',
203
+ }}
204
+ role="tooltip"
205
+ key={order?.uuid}>
206
+ {order?.display}
207
+ </Tag>
208
+ ))}
209
+ </>
210
+ ),
211
+ location: entry?.location?.display,
212
+ status: '--',
213
+ actions: (
214
+ <div style={{ display: 'flex' }}>
215
+ <PrintButtonAction encounter={entry} />
216
+ {enableSendingLabTestsByEmail && <EmailButtonAction />}
217
+ </div>
218
+ ),
219
+ }));
220
+ }, [enableSendingLabTestsByEmail, paginatedPastTestOrderResults]);
221
+
222
+ if (isLoading) {
223
+ return <DataTableSkeleton role="progressbar" />;
224
+ }
225
+
226
+ if (isError) {
227
+ return <ErrorState error={isError} headerTitle={'Error'} />;
228
+ }
229
+
230
+ if (filteredPastTestOrderResults?.length >= 0) {
231
+ return (
232
+ <div className={styles.widgetCard}>
233
+ <CardHeader title={displayText}>
234
+ {isLoading ? (
235
+ <span>
236
+ <InlineLoading />
237
+ </span>
238
+ ) : null}
239
+ </CardHeader>
240
+ <DataTable rows={tableRows} headers={tableHeaders} useZebraStyles>
241
+ {({ rows, headers, getHeaderProps, getTableProps, getRowProps }) => (
242
+ <TableContainer className={styles.tableContainer}>
243
+ <TableToolbar
244
+ style={{
245
+ position: 'static',
246
+ height: '3rem',
247
+ overflow: 'visible',
248
+ backgroundColor: 'color',
249
+ }}>
250
+ <TableToolbarContent>
251
+ <div
252
+ style={{
253
+ fontSize: '10px',
254
+ margin: '5px',
255
+ display: 'flex',
256
+ flexDirection: 'row',
257
+ alignItems: 'center',
258
+ }}>
259
+ Key:
260
+ <Tag
261
+ size="sm"
262
+ style={{
263
+ background: '#6F6F6F',
264
+ color: 'white',
265
+ }}
266
+ title="Result Requested"
267
+ renderIcon={() => <SendAlt />}>
268
+ {'Requested'}
269
+ </Tag>
270
+ <Tag
271
+ size="sm"
272
+ style={{
273
+ background: 'green',
274
+ color: 'white',
275
+ }}
276
+ title="Result Complete"
277
+ renderIcon={() => <Checkmark />}>
278
+ {'Completed'}
279
+ </Tag>
280
+ <Tag
281
+ size="sm"
282
+ style={{
283
+ background: 'red',
284
+ color: 'white',
285
+ }}
286
+ title="Result Rejected"
287
+ renderIcon={() => <NotSent />}>
288
+ {'Rejected'}
289
+ </Tag>
290
+ </div>
291
+ <Layer>
292
+ <TableToolbarSearch
293
+ expanded={true}
294
+ value={searchTerm}
295
+ onChange={handleChange}
296
+ placeholder={t('searchThisList', 'Search this list')}
297
+ size="sm"
298
+ />
299
+ </Layer>
300
+ </TableToolbarContent>
301
+ </TableToolbar>
302
+ <Table {...getTableProps()} className={styles.activePatientsTable}>
303
+ <TableHead>
304
+ <TableRow>
305
+ <TableExpandHeader />
306
+ {headers.map((header) => (
307
+ <TableHeader {...getHeaderProps({ header })}>{header.header}</TableHeader>
308
+ ))}
309
+ </TableRow>
310
+ </TableHead>
311
+ <TableBody>
312
+ {rows.map((row, index) => {
313
+ return (
314
+ <React.Fragment key={row.id}>
315
+ <TableExpandRow {...getRowProps({ row })}>
316
+ {row.cells.map((cell) => (
317
+ <TableCell key={cell.id}>{cell.value?.content ?? cell.value}</TableCell>
318
+ ))}
319
+ </TableExpandRow>
320
+ {row.isExpanded ? (
321
+ <TableExpandedRow className={styles.expandedActiveVisitRow} colSpan={headers.length + 2}>
322
+ {sortedLabRequests[index]?.obs !== null && sortedLabRequests[index]?.obs?.length > 0 && (
323
+ <TestsResults obs={sortedLabRequests[index]?.obs} />
324
+ )}{' '}
325
+ </TableExpandedRow>
326
+ ) : (
327
+ <TableExpandedRow className={styles.hiddenRow} colSpan={headers.length + 2} />
328
+ )}
329
+ </React.Fragment>
330
+ );
331
+ })}
332
+ </TableBody>
333
+ </Table>
334
+ {rows.length === 0 ? (
335
+ <div className={styles.tileContainer}>
336
+ <Tile className={styles.tile}>
337
+ <div className={styles.tileContent}>
338
+ <p className={styles.content}>{t('noTestOrdersToDisplay', 'No test orders to display')}</p>
339
+ </div>
340
+ </Tile>
341
+ </div>
342
+ ) : null}
343
+ <Pagination
344
+ page={currentPage}
345
+ pageSize={currentPageSize}
346
+ pageSizes={pageSizes}
347
+ totalItems={filteredPastTestOrderResults?.length}
348
+ onChange={({ pageSize, page }) => {
349
+ if (pageSize !== currentPageSize) {
350
+ setPageSize(pageSize);
351
+ }
352
+ if (page !== currentPage) {
353
+ goTo(page);
354
+ }
355
+ }}
356
+ className={styles.paginationOverride}
357
+ />
358
+ </TableContainer>
359
+ )}
360
+ </DataTable>
361
+ </div>
362
+ );
363
+ }
364
+ };
365
+
366
+ export default LaboratoryPastTestOrderResults;
@@ -0,0 +1,74 @@
1
+ @use '@carbon/styles/scss/spacing';
2
+ @use '@carbon/styles/scss/colors';
3
+ @use '@carbon/styles/scss/type';
4
+ @use '@openmrs/esm-styleguide/src/vars' as vars;
5
+ @import '../../root.scss';
6
+ .widgetCard {
7
+ border: 1px solid colors.$gray-20;
8
+ border-bottom: none;
9
+ }
10
+
11
+ .headerBtnContainer {
12
+ background-color: vars.$ui-background;
13
+ padding: spacing.$spacing-05;
14
+ text-align: right;
15
+ }
16
+
17
+ .tileContainer {
18
+ background-color: vars.$ui-02;
19
+ border-top: 1px solid vars.$ui-03;
20
+ padding: 5rem 0;
21
+ }
22
+
23
+ .tile {
24
+ margin: auto;
25
+ width: fit-content;
26
+ }
27
+
28
+ .tileContent {
29
+ display: flex;
30
+ flex-direction: column;
31
+ align-items: center;
32
+ }
33
+
34
+ .content {
35
+ @include type.type-style('heading-compact-02');
36
+ color: vars.$text-02;
37
+ margin-bottom: 0.5rem;
38
+ }
39
+
40
+ .helper {
41
+ @include type.type-style('body-compact-01');
42
+ color: vars.$text-02;
43
+ }
44
+
45
+ .container {
46
+ background-color: vars.$ui-01;
47
+ }
48
+
49
+ .separator {
50
+ @include type.type-style('body-compact-02');
51
+ color: vars.$text-02;
52
+ width: 80%;
53
+ margin: 1.5rem auto;
54
+ overflow: hidden;
55
+ text-align: center;
56
+ &::before,
57
+ &::after {
58
+ background-color: vars.$text-03;
59
+ content: '';
60
+ display: inline-block;
61
+ height: 1px;
62
+ position: relative;
63
+ vertical-align: middle;
64
+ width: 50%;
65
+ }
66
+ &::before {
67
+ right: 0.5rem;
68
+ margin-left: -50%;
69
+ }
70
+ &::after {
71
+ left: 0.5rem;
72
+ margin-right: -50%;
73
+ }
74
+ }
@@ -0,0 +1,44 @@
1
+ import React, { useState } from 'react';
2
+ import { Tab, TabList, TabPanel, TabPanels, Tabs } from '@carbon/react';
3
+ import styles from './laboratory-order-tabs.scss';
4
+ import { useTranslation } from 'react-i18next';
5
+ import PatientLaboratoryReferalResults from '../procedure-order-referals/procedure-order-referals.component';
6
+ import PatientLaboratoryResults from '../patient-procedure-results.component';
7
+
8
+ interface LaboratoryResultsTabsProps {
9
+ patientUuid: string;
10
+ }
11
+
12
+ const LaboratoryResultsTabs: React.FC<LaboratoryResultsTabsProps> = ({ patientUuid }) => {
13
+ const { t } = useTranslation();
14
+
15
+ const [selectedTab, setSelectedTab] = useState(0);
16
+
17
+ return (
18
+ <>
19
+ <div className={styles.container}>
20
+ <Tabs
21
+ selectedIndex={selectedTab}
22
+ onChange={({ selectedIndex }) => setSelectedTab(selectedIndex)}
23
+ className={styles.tabs}>
24
+ <TabList style={{ paddingLeft: '1rem' }} aria-label="laboratory results tabs" contained>
25
+ <Tab style={{ width: '150px' }}>{t('pending', 'Routine Tests')}</Tab>
26
+ <Tab style={{ width: '150px' }}>{t('referals', 'Referrals')}</Tab>
27
+ </TabList>
28
+ <TabPanels>
29
+ <TabPanel style={{ padding: 0 }}>
30
+ <PatientLaboratoryResults patientUuid={patientUuid} />
31
+ </TabPanel>
32
+ <TabPanel style={{ padding: 0 }}>
33
+ <div style={{ margin: '10px' }}>
34
+ <PatientLaboratoryReferalResults patientUuid={patientUuid} />
35
+ </div>
36
+ </TabPanel>
37
+ </TabPanels>
38
+ </Tabs>
39
+ </div>
40
+ </>
41
+ );
42
+ };
43
+
44
+ export default LaboratoryResultsTabs;
@@ -0,0 +1,7 @@
1
+ @use '@carbon/styles/scss/spacing';
2
+ @use '@carbon/styles/scss/type';
3
+ @use '@openmrs/esm-styleguide/src/vars' as vars;
4
+
5
+ .container {
6
+ background-color: vars.$ui-01;
7
+ }
@@ -0,0 +1,11 @@
1
+ import React from 'react';
2
+ import { type DefaultWorkspaceProps } from '@openmrs/esm-framework';
3
+ export const LaboratoryWorkspace: React.FC<DefaultWorkspaceProps> = ({ closeWorkspace, promptBeforeClosing }) => {
4
+ return (
5
+ <>
6
+ <span>Coming Soon...</span>
7
+ </>
8
+ );
9
+ };
10
+
11
+ export default LaboratoryWorkspace;
@@ -0,0 +1,152 @@
1
+ import React, { useEffect, useMemo, useRef, useState } from 'react';
2
+ import styles from './print-results-summary.scss';
3
+ import { formatDate, parseDate } from '@openmrs/esm-framework';
4
+ import logoImg from '../../../assets/logo/moh_logo_without_word.png';
5
+ import { type Identifier, type PatientResource } from '../../utils/functions';
6
+ import { useTranslation } from 'react-i18next';
7
+ import PrintResultsTable from './print-results-table.component';
8
+ import { type Result } from '../patient-procedure-order-results.resource';
9
+
10
+ interface PrintResultsSummaryProps {
11
+ encounterResponse: Result;
12
+ patient: PatientResource;
13
+ }
14
+
15
+ const PrintResultsSummary: React.FC<PrintResultsSummaryProps> = ({ encounterResponse, patient }) => {
16
+ const filteredItems = encounterResponse.obs.filter((ob) => ob?.order?.type === 'testorder');
17
+
18
+ const results = useMemo(() => {
19
+ const groupedResults = [];
20
+
21
+ filteredItems.forEach((element) => {
22
+ groupedResults[element?.order?.display] = element;
23
+ });
24
+ return groupedResults;
25
+ }, [filteredItems]);
26
+ return (
27
+ <div className={styles.printPage}>
28
+ <div
29
+ style={{
30
+ display: 'flex',
31
+ justifyContent: 'space-around',
32
+ flexDirection: 'row',
33
+ alignItems: 'center',
34
+ margin: ' 10px',
35
+ }}>
36
+ <div style={{ display: 'flex', flexDirection: 'column' }}>
37
+ <div
38
+ style={{
39
+ display: 'flex',
40
+ flexDirection: 'row',
41
+ justifyContent: 'space-between',
42
+ }}>
43
+ <div
44
+ style={{
45
+ display: 'flex',
46
+ flexDirection: 'column',
47
+ justifyContent: 'center',
48
+ alignItems: 'center',
49
+ }}>
50
+ <img src={logoImg} alt={'logo'} width={50} height={50} />
51
+ <span style={{ fontSize: '10px', fontWeight: 'bold', margin: '5px' }}>
52
+ {encounterResponse?.visit?.location?.display}
53
+ </span>
54
+ </div>
55
+ <div
56
+ style={{
57
+ display: 'flex',
58
+ flexDirection: 'column',
59
+ justifyContent: 'center',
60
+ alignItems: 'center',
61
+ }}>
62
+ <span style={{ fontSize: '10px', fontWeight: 'bold', margin: '2px' }}>Code :</span>
63
+ <span style={{ fontSize: '10px', fontWeight: 'bold', margin: '2px' }}>District :</span>
64
+ </div>
65
+ </div>
66
+ </div>
67
+
68
+ <div style={{ display: 'flex', flexDirection: 'column' }}>
69
+ <span style={{ margin: '5px', fontSize: '10px' }}>
70
+ <span style={{ fontSize: '10px', fontWeight: 'bold' }}>{patient?.person?.display}</span>,
71
+ {patient?.person?.gender === 'M' ? ' Male' : patient?.person?.gender === 'F' ? ' Female' : ' Unknown'},
72
+ <span>
73
+ {' '}
74
+ {formatDate(parseDate(patient?.person?.birthdate), {
75
+ mode: 'standard',
76
+ time: false,
77
+ })}{' '}
78
+ </span>
79
+ </span>
80
+ <span style={{ margin: '5px', fontSize: '10px' }}>
81
+ HIV Clinic No. :
82
+ {patient?.identifiers?.length
83
+ ? patient?.identifiers?.find((identifier: Identifier) => {
84
+ return identifier?.identifierType.uuid === 'e1731641-30ab-102d-86b0-7a5022ba4115';
85
+ })?.identifier
86
+ : '--'}
87
+ </span>
88
+ <span style={{ margin: '5px', fontSize: '10px' }}>
89
+ Patient Unique Code (UIC). :
90
+ {patient?.identifiers?.length > 0
91
+ ? patient?.identifiers?.find((identifier: Identifier) => {
92
+ return identifier?.identifierType?.uuid === '877169c4-92c6-4cc9-bf45-1ab95faea242';
93
+ })?.identifier
94
+ : '--'}
95
+ </span>
96
+ </div>
97
+
98
+ <div style={{ display: 'flex', flexDirection: 'column' }}>
99
+ <span style={{ margin: '5px', fontSize: '10px' }}>
100
+ Prepared By : {encounterResponse?.auditInfo?.creator?.display}
101
+ </span>
102
+ <span style={{ margin: '5px', fontSize: '10px' }}>
103
+ Date Requested :
104
+ {formatDate(parseDate(encounterResponse?.encounterDatetime), {
105
+ time: false,
106
+ })}
107
+ </span>
108
+ </div>
109
+ </div>
110
+
111
+ <section className={styles.section}>
112
+ <div
113
+ style={{
114
+ display: 'flex',
115
+ justifyContent: 'space-between',
116
+ }}></div>
117
+ </section>
118
+ <section className={styles.section}>
119
+ {Object.keys(results)?.length > 0 && <PrintResultsTable groupedResults={results} />}
120
+ </section>
121
+
122
+ <section className={styles.section}>
123
+ <div
124
+ style={{
125
+ margin: '10px',
126
+ display: 'flex',
127
+ width: '500px',
128
+ flexDirection: 'row',
129
+ }}>
130
+ <span style={{ fontSize: '14px', marginBottom: '10px' }}>
131
+ Results Reviewed / Authorized by :
132
+ <span style={{ marginLeft: '50px' }}>{encounterResponse?.auditInfo?.creator?.display}</span>
133
+ </span>
134
+ </div>
135
+ <div
136
+ style={{
137
+ margin: '10px',
138
+ display: 'flex',
139
+ width: '500px',
140
+ flexDirection: 'row',
141
+ }}>
142
+ <span style={{ fontSize: '14px', marginTop: '10px' }}>
143
+ Sign : .............................................{' '}
144
+ <span style={{ fontSize: '14px', marginLeft: '50px' }}>Date : ............................</span>
145
+ </span>
146
+ </div>
147
+ </section>
148
+ </div>
149
+ );
150
+ };
151
+
152
+ export default PrintResultsSummary;