ui-soxo-bootstrap-core 2.6.40-dev.1 → 2.6.40-dev.12

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 (420) hide show
  1. package/.babelrc +8 -8
  2. package/.github/workflows/npm-publish.yml +5 -4
  3. package/.husky/pre-commit +11 -11
  4. package/.prettierrc.json +10 -10
  5. package/DEVELOPER_GUIDE.md +323 -323
  6. package/PUBLISHING.md +333 -0
  7. package/babel.config.js +2 -2
  8. package/core/components/component-loader/component-loader.js +125 -125
  9. package/core/components/component-wrapper/component-wrapper.js +121 -121
  10. package/core/components/external-window/DEVELOPER_GUIDE.md +705 -705
  11. package/core/components/external-window/external-window.js +236 -236
  12. package/core/components/external-window/external-window.test.js +80 -80
  13. package/core/components/extra-info/extra-info-details.js +155 -155
  14. package/core/components/extra-info/extra-info-details.scss +26 -26
  15. package/core/components/extra-info/extra-info.js +134 -134
  16. package/core/components/index.js +12 -12
  17. package/core/components/landing-api/landing-api.js +707 -707
  18. package/core/components/landing-api/landing-api.scss +41 -41
  19. package/core/components/license-management/license-alert.js +97 -97
  20. package/core/components/menu-template-api/menu-template-api.js +321 -321
  21. package/core/components/root-application-api/root-application-api.js +174 -174
  22. package/core/index.js +13 -13
  23. package/core/lib/Store.js +369 -369
  24. package/core/lib/components/application-bootstrap/application-bootstrap.js +115 -115
  25. package/core/lib/components/approval-form/approval-form.js +280 -280
  26. package/core/lib/components/approval-form/approval-form.scss +183 -183
  27. package/core/lib/components/approval-list/approval-list.js +143 -143
  28. package/core/lib/components/approval-list/approval-list.scss +2 -2
  29. package/core/lib/components/approval-list/components/request-card/request-card.js +42 -42
  30. package/core/lib/components/approval-list/components/request-card/request-card.scss +30 -30
  31. package/core/lib/components/camera/camera.js +230 -230
  32. package/core/lib/components/camera/camera.scss +86 -86
  33. package/core/lib/components/comment-block/comment-block.js +138 -138
  34. package/core/lib/components/comment-block/comment-block.scss +3 -3
  35. package/core/lib/components/confirm-modal/confirm-modal.js +82 -82
  36. package/core/lib/components/confirm-modal/confirm-modal.scss +2 -2
  37. package/core/lib/components/consent/consent.js +67 -67
  38. package/core/lib/components/consent/pdf-signature.js +299 -299
  39. package/core/lib/components/consent/signature-pad.js +90 -90
  40. package/core/lib/components/consent/signature-pad.scss +14 -14
  41. package/core/lib/components/file-upload/file-upload.js +133 -133
  42. package/core/lib/components/finger-print-reader/finger-print-reader.js +295 -295
  43. package/core/lib/components/finger-print-reader/finger-print-reader.scss +47 -47
  44. package/core/lib/components/finger-print-search/finger-print-search.js +200 -200
  45. package/core/lib/components/finger-print-search/finger-print-search.scss +47 -47
  46. package/core/lib/components/global-header/animations.js +18 -18
  47. package/core/lib/components/global-header/global-header.js +286 -286
  48. package/core/lib/components/global-header/global-header.scss +397 -397
  49. package/core/lib/components/header/generic-header.js +76 -76
  50. package/core/lib/components/header/generic-header.scss +99 -99
  51. package/core/lib/components/image-preview/image-preview.js +33 -33
  52. package/core/lib/components/image-wrapper/image-wrapper.js +108 -108
  53. package/core/lib/components/image-wrapper/image-wrapper.scss +12 -12
  54. package/core/lib/components/index.js +206 -206
  55. package/core/lib/components/landing/landing.js +403 -403
  56. package/core/lib/components/language-switcher/language-switcher.js +49 -49
  57. package/core/lib/components/menu-context/menu-context.js +69 -69
  58. package/core/lib/components/menu-template/menu-template.js +249 -249
  59. package/core/lib/components/menu-template/menu-template.scss +9 -9
  60. package/core/lib/components/modal-search/modal-search.js +153 -153
  61. package/core/lib/components/modal-search/modal-search.scss +78 -78
  62. package/core/lib/components/modal-wrapper/modal-manager.js +15 -15
  63. package/core/lib/components/modal-wrapper/modal-wrapper.js +108 -108
  64. package/core/lib/components/modal-wrapper/modal-wrapper.scss +13 -13
  65. package/core/lib/components/notice-board/notice-board.js +132 -132
  66. package/core/lib/components/notice-board/notice-board.scss +65 -65
  67. package/core/lib/components/page-container/page-container.js +55 -55
  68. package/core/lib/components/page-container/page-container.scss +8 -8
  69. package/core/lib/components/page-header/page-header.js +23 -23
  70. package/core/lib/components/page-header/page-header.scss +17 -17
  71. package/core/lib/components/pdf-viewer/pdf-viewer.js +56 -56
  72. package/core/lib/components/portlet-table/components/table-actions/table-actions.js +58 -58
  73. package/core/lib/components/portlet-table/components/table-actions/table-actions.scss +1 -1
  74. package/core/lib/components/portlet-table/components/table-data/table-data.js +106 -106
  75. package/core/lib/components/portlet-table/portlet-table.js +63 -63
  76. package/core/lib/components/portlet-table/portlet-table.scss +90 -90
  77. package/core/lib/components/progress-bar/progress-bar.js +58 -58
  78. package/core/lib/components/progress-bar/progress-bar.scss +15 -15
  79. package/core/lib/components/request-form/request-form.js +110 -110
  80. package/core/lib/components/root-application/root-application.js +70 -70
  81. package/core/lib/components/rupee/rupee.js +14 -14
  82. package/core/lib/components/script-input/script-input.js +169 -169
  83. package/core/lib/components/script-input/script-input.scss +8 -8
  84. package/core/lib/components/sidemenu/animations.js +51 -51
  85. package/core/lib/components/sidemenu/sidemenu.js +713 -713
  86. package/core/lib/components/sidemenu/sidemenu.scss +314 -314
  87. package/core/lib/components/spotlight-search/spotlight-search.component.js +635 -635
  88. package/core/lib/components/spotlight-search/spotlight-search.component.scss +78 -78
  89. package/core/lib/components/table-wrapper/table-wrapper.js +135 -135
  90. package/core/lib/components/table-wrapper/table-wrapper.scss +72 -72
  91. package/core/lib/components/ui_elements/Loader.js +12 -12
  92. package/core/lib/components/ui_elements/Notify.js +12 -12
  93. package/core/lib/components/ui_elements/PlaceHolder.js +33 -33
  94. package/core/lib/components/web-camera/web-camera.js +161 -161
  95. package/core/lib/components/web-camera/web-camera.scss +28 -28
  96. package/core/lib/core.md +9 -9
  97. package/core/lib/elements/Elements.md +2 -2
  98. package/core/lib/elements/basic/LoggedUserRedirect.js +21 -21
  99. package/core/lib/elements/basic/PrivateRoute.js +16 -16
  100. package/core/lib/elements/basic/button/Button.md +43 -43
  101. package/core/lib/elements/basic/button/button.js +170 -170
  102. package/core/lib/elements/basic/card/Card.md +15 -15
  103. package/core/lib/elements/basic/card/card.js +40 -40
  104. package/core/lib/elements/basic/card/card.scss +13 -13
  105. package/core/lib/elements/basic/checkbox/checkbox.js +23 -23
  106. package/core/lib/elements/basic/col/col.js +15 -15
  107. package/core/lib/elements/basic/copy-to-clipboard/Readme.md +40 -40
  108. package/core/lib/elements/basic/copy-to-clipboard/copy-to-clipboard.js +61 -61
  109. package/core/lib/elements/basic/country-phone-input/Readme.md +98 -98
  110. package/core/lib/elements/basic/country-phone-input/country-phone-input.js +81 -81
  111. package/core/lib/elements/basic/country-phone-input/phone-input.scss +75 -75
  112. package/core/lib/elements/basic/datepicker/datepicker.js +33 -33
  113. package/core/lib/elements/basic/dragabble-wrapper/draggable-wrapper.js +203 -203
  114. package/core/lib/elements/basic/empty/empty.js +14 -14
  115. package/core/lib/elements/basic/fingerprint-protrected/fingerprint-protected.js +118 -118
  116. package/core/lib/elements/basic/fingerprint-protrected/fingerprint-protected.scss +10 -10
  117. package/core/lib/elements/basic/form/form.js +70 -70
  118. package/core/lib/elements/basic/form/form.scss +3 -3
  119. package/core/lib/elements/basic/image/image.js +45 -45
  120. package/core/lib/elements/basic/image/image.scss +17 -17
  121. package/core/lib/elements/basic/image/readme.md +26 -26
  122. package/core/lib/elements/basic/image-viewer/image-viewer.js +108 -108
  123. package/core/lib/elements/basic/image-viewer/image-viewer.scss +7 -7
  124. package/core/lib/elements/basic/input/input.js +81 -81
  125. package/core/lib/elements/basic/input/readme.md +77 -77
  126. package/core/lib/elements/basic/json-input/json-input.js +51 -51
  127. package/core/lib/elements/basic/menu-dashboard/menu-dashboard.js +216 -216
  128. package/core/lib/elements/basic/menu-dashboard/menu-dashboard.scss +28 -28
  129. package/core/lib/elements/basic/menu-tree/menu-tree.js +127 -127
  130. package/core/lib/elements/basic/modal/modal.js +64 -64
  131. package/core/lib/elements/basic/modal/readme.md +62 -62
  132. package/core/lib/elements/basic/popconfirm/popconfirm.js +17 -17
  133. package/core/lib/elements/basic/popover/popover.js +12 -12
  134. package/core/lib/elements/basic/radio/radio.js +18 -18
  135. package/core/lib/elements/basic/rangepicker/rangepicker.js +141 -141
  136. package/core/lib/elements/basic/rangepicker/rangepicker.scss +24 -24
  137. package/core/lib/elements/basic/rangepicker/readme.md +81 -81
  138. package/core/lib/elements/basic/reference-select/readme.md +18 -18
  139. package/core/lib/elements/basic/reference-select/reference-select.js +337 -337
  140. package/core/lib/elements/basic/row/row.js +15 -15
  141. package/core/lib/elements/basic/select/select.js +46 -46
  142. package/core/lib/elements/basic/select-box/readme.md +52 -52
  143. package/core/lib/elements/basic/select-box/select-box.js +63 -63
  144. package/core/lib/elements/basic/skeleton/readme.md +35 -35
  145. package/core/lib/elements/basic/skeleton/skeleton.js +35 -35
  146. package/core/lib/elements/basic/skeleton/skeleton.scss +53 -53
  147. package/core/lib/elements/basic/space/space.js +12 -12
  148. package/core/lib/elements/basic/switch/readme.md +29 -29
  149. package/core/lib/elements/basic/switch/switch.js +67 -67
  150. package/core/lib/elements/basic/tab/tab.js +14 -14
  151. package/core/lib/elements/basic/table/readme.md +8 -8
  152. package/core/lib/elements/basic/table/table.js +95 -95
  153. package/core/lib/elements/basic/tag/tag.js +63 -63
  154. package/core/lib/elements/basic/tag/tag.scss +2 -2
  155. package/core/lib/elements/basic/timeline/timeline.js +13 -13
  156. package/core/lib/elements/basic/title/readme.md +20 -20
  157. package/core/lib/elements/basic/title/title.js +37 -37
  158. package/core/lib/elements/basic/user-search/user-search.js +192 -192
  159. package/core/lib/elements/complex/barcode/barcode.js +27 -27
  160. package/core/lib/elements/complex/bargraph/bar-graph.js +262 -262
  161. package/core/lib/elements/complex/basic-table/basic-table.js +110 -110
  162. package/core/lib/elements/complex/basic-table/basic-table.scss +4 -4
  163. package/core/lib/elements/complex/date-display/date-display.js +37 -37
  164. package/core/lib/elements/complex/error-boundary/error-boundary.js +29 -29
  165. package/core/lib/elements/complex/google-location-input/map-container-library-load.js +92 -92
  166. package/core/lib/elements/complex/google-map/google-map.js +230 -230
  167. package/core/lib/elements/complex/google-map/google-map.scss +13 -13
  168. package/core/lib/elements/complex/line-graph/line-graph.js +108 -108
  169. package/core/lib/elements/complex/location-search-input/location-search-input.js +100 -100
  170. package/core/lib/elements/complex/pie-chart/pie-chart.js +202 -202
  171. package/core/lib/elements/complex/qr-code/qr-code.js +27 -27
  172. package/core/lib/elements/complex/qrscanner/qrscanner.js +57 -57
  173. package/core/lib/elements/complex/search-debounce/search-debounce.js +37 -37
  174. package/core/lib/elements/complex/statistic-card/dashboard-statistic-card.js +75 -75
  175. package/core/lib/elements/complex/statistic-card/statistic-card.js +28 -28
  176. package/core/lib/elements/index.js +226 -226
  177. package/core/lib/hooks/device-detect.js +25 -25
  178. package/core/lib/hooks/index.js +9 -9
  179. package/core/lib/hooks/use-location.js +33 -33
  180. package/core/lib/hooks/use-otp-timer.js +80 -80
  181. package/core/lib/hooks/use-window-size.js +34 -34
  182. package/core/lib/i18n.js +69 -69
  183. package/core/lib/index.js +106 -106
  184. package/core/lib/introduction.md +73 -73
  185. package/core/lib/js-styleguide.md +4112 -4112
  186. package/core/lib/models/actions/actions.js +127 -127
  187. package/core/lib/models/actions/components/action-detail/action-detail.js +190 -190
  188. package/core/lib/models/actions/components/custom-actions/custom-actions.js +185 -185
  189. package/core/lib/models/attachments/attachments.js +231 -231
  190. package/core/lib/models/base-loader.js +99 -99
  191. package/core/lib/models/base.js +716 -716
  192. package/core/lib/models/branches/branches.js +125 -125
  193. package/core/lib/models/checklists/checklists.js +114 -114
  194. package/core/lib/models/columns/columns.js +169 -169
  195. package/core/lib/models/columns/components/columns-add/columns-add.js +171 -171
  196. package/core/lib/models/comments/comments.js +213 -213
  197. package/core/lib/models/departments/departments.js +107 -107
  198. package/core/lib/models/financial-years/financial_years.js +127 -127
  199. package/core/lib/models/forms/components/form-creator/form-creator.js +665 -665
  200. package/core/lib/models/forms/components/form-creator/form-creator.scss +39 -39
  201. package/core/lib/models/forms/components/form-detail/form-detail.js +224 -224
  202. package/core/lib/models/forms/forms.js +121 -121
  203. package/core/lib/models/index.js +203 -203
  204. package/core/lib/models/invoice-numbers/invoice_numbers.js +204 -204
  205. package/core/lib/models/lookup-types/components/lookup-detail/lookup-detail.js +145 -145
  206. package/core/lib/models/lookup-types/lookup-types.js +113 -113
  207. package/core/lib/models/lookup-values/components/lookup-values-add/lookup-values-add.js +126 -126
  208. package/core/lib/models/lookup-values/lookup-values.js +107 -107
  209. package/core/lib/models/menu-roles/menu-roles.js +127 -127
  210. package/core/lib/models/menus/components/menu-add/menu-add.js +228 -228
  211. package/core/lib/models/menus/components/menu-detail/menu-detail.js +170 -170
  212. package/core/lib/models/menus/components/menu-list/menu-list.js +550 -550
  213. package/core/lib/models/menus/components/menu-list/menu-list.scss +5 -5
  214. package/core/lib/models/menus/components/menu-roles-add/menu-roles-add.js +183 -183
  215. package/core/lib/models/menus/menus.js +499 -499
  216. package/core/lib/models/models/components/model-detail/model-detail.js +137 -137
  217. package/core/lib/models/models/components/models.js +128 -128
  218. package/core/lib/models/modules/modules.js +204 -204
  219. package/core/lib/models/outbox/outbox.js +73 -73
  220. package/core/lib/models/pages/pages.js +107 -107
  221. package/core/lib/models/permissions/permissions.js +71 -71
  222. package/core/lib/models/process/components/process-add/process-add.js +181 -181
  223. package/core/lib/models/process/components/process-dashboard/process-dashboard.js +1068 -1068
  224. package/core/lib/models/process/components/process-dashboard/process-dashboard.scss +66 -66
  225. package/core/lib/models/process/components/process-detail/process-detail.js +140 -140
  226. package/core/lib/models/process/components/process-timeline/process-timeline.js +139 -139
  227. package/core/lib/models/process/components/task-detail/task-detail.js +240 -240
  228. package/core/lib/models/process/components/task-detail/task-detail.scss +27 -27
  229. package/core/lib/models/process/components/task-form/task-form.js +528 -528
  230. package/core/lib/models/process/components/task-form/task-form.scss +7 -7
  231. package/core/lib/models/process/components/task-list/task-list.js +221 -221
  232. package/core/lib/models/process/components/task-list/task-list.scss +14 -14
  233. package/core/lib/models/process/components/task-overview/task-overview.js +299 -299
  234. package/core/lib/models/process/components/task-overview-legacy/task-overview-legacy.js +192 -192
  235. package/core/lib/models/process/components/task-routes/task-routes.js +45 -45
  236. package/core/lib/models/process/components/task-status/task-status.js +175 -175
  237. package/core/lib/models/process/components/task-status/task-status.scss +11 -11
  238. package/core/lib/models/process/process.js +780 -780
  239. package/core/lib/models/process-transactions/process-transactions.js +123 -123
  240. package/core/lib/models/roles/roles.js +106 -106
  241. package/core/lib/models/scripts/scripts.js +111 -111
  242. package/core/lib/models/step-transactions/step-transcations.js +147 -147
  243. package/core/lib/models/steps/components/step-add/step-add.js +261 -261
  244. package/core/lib/models/steps/components/step-detail/step-detail.js +157 -157
  245. package/core/lib/models/steps/steps.js +356 -356
  246. package/core/lib/models/user-preferences/user-preferences.js +83 -83
  247. package/core/lib/models/users/components/user-add/user-add.js +226 -226
  248. package/core/lib/models/users/users.js +119 -119
  249. package/core/lib/modules/business/launch-page/launch-page.js +29 -29
  250. package/core/lib/modules/business/launch-page/launch-page.scss +5 -5
  251. package/core/lib/modules/business/slots/slots.js +231 -231
  252. package/core/lib/modules/business/slots/slots.scss +108 -108
  253. package/core/lib/modules/forms/components/field-customizer/field-customizer.js +138 -138
  254. package/core/lib/modules/forms/components/field-selector/field-selector.js +157 -157
  255. package/core/lib/modules/forms/components/field-selector/field-selector.scss +25 -25
  256. package/core/lib/modules/forms/components/form-display/form-display.js +203 -203
  257. package/core/lib/modules/forms/components/form-display/form-display.scss +9 -9
  258. package/core/lib/modules/forms/components/tab-customizer/tab-customizer.js +124 -124
  259. package/core/lib/modules/generic/generic-add/generic-add.js +213 -213
  260. package/core/lib/modules/generic/generic-detail/generic-detail.js +199 -199
  261. package/core/lib/modules/generic/generic-edit/generic-edit.js +120 -120
  262. package/core/lib/modules/generic/generic-list/ExportReactCSV.js +414 -414
  263. package/core/lib/modules/generic/generic-list/generic-list.js +705 -705
  264. package/core/lib/modules/generic/generic-list/generic-list.scss +68 -68
  265. package/core/lib/modules/generic/generic-upload/generic-upload.js +483 -483
  266. package/core/lib/modules/generic/table-settings/table-settings.js +226 -226
  267. package/core/lib/modules/generic/table-settings/table-settings.scss +37 -37
  268. package/core/lib/modules/index.js +52 -52
  269. package/core/lib/modules/modules-routes/module-routes.js +35 -35
  270. package/core/lib/pages/change-password/change-password.js +204 -204
  271. package/core/lib/pages/change-password/change-password.scss +73 -73
  272. package/core/lib/pages/homepage/homepage.js +53 -53
  273. package/core/lib/pages/index.js +19 -19
  274. package/core/lib/pages/login/commnication-mode-selection.js +46 -46
  275. package/core/lib/pages/login/communication-mode-selection.scss +60 -60
  276. package/core/lib/pages/login/login.js +872 -872
  277. package/core/lib/pages/login/login.scss +353 -353
  278. package/core/lib/pages/login/reset-password.js +124 -124
  279. package/core/lib/pages/login/reset-password.scss +31 -31
  280. package/core/lib/pages/manage-users/manage-users.js +429 -429
  281. package/core/lib/pages/manage-users/manage-users.scss +25 -25
  282. package/core/lib/pages/profile/profile.js +247 -247
  283. package/core/lib/pages/profile/profile.scss +107 -107
  284. package/core/lib/pages/profile/theme-config.js +18 -18
  285. package/core/lib/pages/profile/themes-backup.json +310 -310
  286. package/core/lib/pages/profile/themes.json +254 -254
  287. package/core/lib/pages/register/register.js +176 -176
  288. package/core/lib/pages/register/register.scss +128 -128
  289. package/core/lib/react-styleguide.md +756 -756
  290. package/core/lib/utils/api/api.utils.js +207 -207
  291. package/core/lib/utils/api/readme.md +426 -426
  292. package/core/lib/utils/async.js +35 -35
  293. package/core/lib/utils/common/common.utils.js +237 -237
  294. package/core/lib/utils/common/readme.md +30 -30
  295. package/core/lib/utils/date/date.utils.js +295 -295
  296. package/core/lib/utils/date/readme.md +2 -2
  297. package/core/lib/utils/firebase.support.utils.js +98 -98
  298. package/core/lib/utils/firebase.utils.js +808 -808
  299. package/core/lib/utils/font-awesome.utils.js +168 -168
  300. package/core/lib/utils/form/form.utils.js +255 -255
  301. package/core/lib/utils/generic/generic.utils.js +70 -70
  302. package/core/lib/utils/http/auth.helper.js +95 -95
  303. package/core/lib/utils/http/http.utils.js +186 -186
  304. package/core/lib/utils/http/readme.md +14 -14
  305. package/core/lib/utils/index.js +43 -43
  306. package/core/lib/utils/location/location.utils.js +137 -137
  307. package/core/lib/utils/location/readme.md +18 -18
  308. package/core/lib/utils/modal.utils.js +15 -15
  309. package/core/lib/utils/notification.utils.js +34 -34
  310. package/core/lib/utils/pwa/pwa.utils.js +88 -88
  311. package/core/lib/utils/script.utils.js +235 -235
  312. package/core/lib/utils/setting.utils.js +68 -68
  313. package/core/lib/utils/upload.utils.js +29 -29
  314. package/core/models/Preference/Preferences.js +46 -46
  315. package/core/models/base/base.js +403 -403
  316. package/core/models/base-clone-loader.js +107 -107
  317. package/core/models/base-clone.js +187 -187
  318. package/core/models/base-loader.js +97 -97
  319. package/core/models/core-scripts/core-scripts.js +179 -179
  320. package/core/models/dashboard/dashboard.js +201 -201
  321. package/core/models/detail-loader.js +88 -88
  322. package/core/models/doctor/components/doctor-add/doctor-add.js +432 -432
  323. package/core/models/doctor/components/doctor-add/doctor-add.scss +32 -32
  324. package/core/models/groups.js +82 -82
  325. package/core/models/index.js +100 -100
  326. package/core/models/lookup-types/components/lookup-detail/lookup-detail.js +129 -129
  327. package/core/models/lookup-types/lookup-types.js +96 -96
  328. package/core/models/lookup-values/components/lookup-values-modal/lookup-values-modal.js +95 -95
  329. package/core/models/lookup-values/lookup-values.js +92 -92
  330. package/core/models/menu-roles/components/menu-roles-add/menu-roles-add.js +153 -153
  331. package/core/models/menu-roles/menu-roles.js +158 -158
  332. package/core/models/menus/components/menu-add/menu-add.js +288 -288
  333. package/core/models/menus/components/menu-add/menu-add.scss +31 -31
  334. package/core/models/menus/components/menu-detail/menu-detail.js +263 -263
  335. package/core/models/menus/components/menu-list/menu-list.js +392 -392
  336. package/core/models/menus/components/menu-lists/menu-lists.js +635 -584
  337. package/core/models/menus/components/menu-lists/menu-lists.scss +46 -46
  338. package/core/models/menus/menus.js +338 -338
  339. package/core/models/model-columns.js +121 -121
  340. package/core/models/models/components/model-detail/model-add.js +120 -120
  341. package/core/models/models/components/model-detail/model-detail.js +133 -133
  342. package/core/models/models/models.js +154 -154
  343. package/core/models/pages/components/page-add/page-add.js +163 -163
  344. package/core/models/pages/components/page-add/page-add.scss +30 -30
  345. package/core/models/pages/components/page-details/page-details.js +209 -209
  346. package/core/models/pages/components/page-list/page-list.js +248 -248
  347. package/core/models/pages/pages.js +142 -142
  348. package/core/models/pages.js +142 -142
  349. package/core/models/roles/components/role-add/menu-label.js +14 -14
  350. package/core/models/roles/components/role-add/menu-tree.js +127 -127
  351. package/core/models/roles/components/role-add/role-add.js +222 -222
  352. package/core/models/roles/components/role-add/role-add.scss +4 -4
  353. package/core/models/roles/components/role-list/role-list.js +406 -406
  354. package/core/models/roles/roles.js +196 -196
  355. package/core/models/staff/components/staff-add/staff-add.js +455 -455
  356. package/core/models/user-roles/components/user-roles-add/user-roles-add.js +149 -149
  357. package/core/models/user-roles/user-roles.js +113 -113
  358. package/core/models/users/components/assign-role/assign-role.js +428 -428
  359. package/core/models/users/components/assign-role/assign-role.scss +281 -281
  360. package/core/models/users/components/assign-role/avatar-props.js +45 -45
  361. package/core/models/users/components/user-add/user-add.js +847 -847
  362. package/core/models/users/components/user-add/user-edit.js +110 -110
  363. package/core/models/users/components/user-detail/user-detail.js +236 -236
  364. package/core/models/users/components/user-list/user-list.js +397 -397
  365. package/core/models/users/users.js +379 -379
  366. package/core/modules/Informations/change-info/change-info.js +618 -618
  367. package/core/modules/Informations/change-info/change-info.scss +134 -134
  368. package/core/modules/dashboard/components/dashboard-card/animations.js +64 -64
  369. package/core/modules/dashboard/components/dashboard-card/dashboard-card.js +197 -197
  370. package/core/modules/dashboard/components/dashboard-card/menu-dashboard-card.js +430 -430
  371. package/core/modules/dashboard/components/dashboard-card/menu-dashboard-card.scss +59 -59
  372. package/core/modules/dashboard/components/pop-query-dashboard/pop-query-dashboard.js +66 -66
  373. package/core/modules/generic/components/generic-add/generic-add.js +121 -121
  374. package/core/modules/generic/components/generic-add/generic-add.scss +13 -13
  375. package/core/modules/generic/components/generic-add-modal/generic-add-modal.js +125 -125
  376. package/core/modules/generic/components/generic-add-modal/generic-add-modal.scss +13 -13
  377. package/core/modules/generic/components/generic-detail/generic-detail.js +184 -184
  378. package/core/modules/generic/components/generic-detail/generic-detail.scss +25 -25
  379. package/core/modules/generic/components/generic-edit/generic-edit.js +123 -123
  380. package/core/modules/generic/components/generic-list/generic-list.js +335 -335
  381. package/core/modules/generic/components/generic-list/generic-list.scss +35 -35
  382. package/core/modules/index.js +42 -42
  383. package/core/modules/module-routes/module-routes.js +37 -37
  384. package/core/modules/reporting/components/index.js +6 -6
  385. package/core/modules/reporting/components/reporting-dashboard/README.md +316 -316
  386. package/core/modules/reporting/components/reporting-dashboard/adavance-search/advance-search.js +271 -271
  387. package/core/modules/reporting/components/reporting-dashboard/adavance-search/advance-search.scss +76 -76
  388. package/core/modules/reporting/components/reporting-dashboard/display-columns/build-display-columns.js +90 -90
  389. package/core/modules/reporting/components/reporting-dashboard/display-columns/build-display-columns.test.js +74 -74
  390. package/core/modules/reporting/components/reporting-dashboard/display-columns/display-cell-renderer.js +449 -449
  391. package/core/modules/reporting/components/reporting-dashboard/display-columns/display-cell-renderer.test.js +199 -199
  392. package/core/modules/reporting/components/reporting-dashboard/reporting-dashboard.js +1116 -1116
  393. package/core/modules/reporting/components/reporting-dashboard/reporting-dashboard.scss +215 -215
  394. package/core/modules/reporting/components/reporting-dashboard/reporting-table.js +519 -519
  395. package/core/modules/steps/action-buttons.js +92 -92
  396. package/core/modules/steps/action-buttons.scss +62 -62
  397. package/core/modules/steps/chat-assistant.js +141 -141
  398. package/core/modules/steps/narration.js +192 -192
  399. package/core/modules/steps/openai-realtime.js +275 -275
  400. package/core/modules/steps/progress-storage.js +140 -140
  401. package/core/modules/steps/readme.md +167 -167
  402. package/core/modules/steps/steps.js +1567 -1567
  403. package/core/modules/steps/steps.scss +907 -907
  404. package/core/modules/steps/timeline.js +56 -56
  405. package/core/modules/steps/voice-navigation.js +709 -709
  406. package/core/pages/homepage-api/homepage-api.js +106 -106
  407. package/core/pages/homepage-api/homepage-api.scss +233 -233
  408. package/core/pages/homepage-api/menu-dashboard.js +169 -169
  409. package/core/pages/homepage-api/menu-dashboard.scss +11 -11
  410. package/core/translation.json +53 -53
  411. package/core/translations.json +19 -19
  412. package/core/utils/script.utils.js +129 -129
  413. package/core/utils/settings.utils.js +25 -25
  414. package/eslint.config.mjs +79 -79
  415. package/index.js +35 -35
  416. package/jest.config.js +7 -7
  417. package/jest.setup.js +1 -1
  418. package/package.json +124 -124
  419. package/tsconfig.json +26 -26
  420. package/webpack.config.js +173 -173
@@ -1,519 +1,519 @@
1
- import React, { useState, useEffect } from 'react';
2
- import { Table, Skeleton, Input, Modal, message, Pagination } from 'antd';
3
- import { QrcodeOutlined } from '@ant-design/icons';
4
- import { ExportReactCSV, getExportData, Card, TableComponent, QrScanner } from './../../../../lib/';
5
- import moment from 'moment-timezone';
6
- import { CoreScripts } from './../../../../models/';
7
- import Button from '../../../../lib/elements/basic/button/button';
8
- import buildDisplayColumns from './display-columns/build-display-columns';
9
- import { getRedirectLink } from './display-columns/display-cell-renderer';
10
- import * as ReportingDashboardComp from '../index';
11
-
12
- const { Search } = Input;
13
- const genericComponents = require('./../../../../lib');
14
-
15
- /**
16
- * @typedef {Object} ReportingTablePagination
17
- * @property {number} [current] Current page number.
18
- * @property {number} [pageSize] Number of rows per page.
19
- * @property {number} [total] Total number of records available.
20
- */
21
-
22
- /**
23
- * @typedef {Object} ReportingTableRequestPayload
24
- * @property {Object} [body] Request body sent when loading reporting data.
25
- */
26
-
27
- /**
28
- * @typedef {Object} ReportingTableProps
29
- * @property {Array<Object>} [patients] Preloaded table rows supplied by a parent component.
30
- * @property {Array<Object>} [columns] Preconfigured display column definitions.
31
- * @property {boolean} [loading] External loading state used when parent owns data fetching.
32
- * @property {Object.<string, React.ComponentType<any>>} [CustomComponents] Custom renderers/actions available to display columns.
33
- * @property {Function} [refresh] Refresh callback passed down to nested action components.
34
- * @property {boolean} [isFixedIndex] Enables fixed index rendering in generated columns.
35
- * @property {string} [barcodeFilterKey] Record field compared against scanned QR/barcode values.
36
- * @property {boolean} [showScanner] Controls whether the QR scanner button is shown.
37
- * @property {Object} [config] Reporting configuration returned by the core script.
38
- * @property {ReportingTablePagination} [pagination] External pagination state.
39
- * @property {(pager: ReportingTablePagination) => void} [handlePagination] Parent-owned pagination handler.
40
- * @property {string} [attributes] JSON string containing extra button/config attributes.
41
- * @property {Function} [fetchReportData] Parent refresh callback used in controlled mode.
42
- * @property {string|number} [reportId] Report id used to load configuration and rows.
43
- * @property {string|number} [requestId] Alternative report request id used for independent loading.
44
- * @property {ReportingTableRequestPayload} [requestPayload] Optional request payload override for listing calls.
45
- * @property {(pager: ReportingTablePagination) => void} [onPaginationChange] Callback fired after independent pagination updates.
46
- * @property {string} [mode] Report mode used when fetching by mode/submode.
47
- * @property {string} [submode] Report submode used when fetching by mode/submode.
48
- * @property {Object} [replacements] Dynamic replacements used by some report APIs and action links.
49
- * @property {boolean} [isNuradesk] Switches to Nuradesk-specific report loading behavior.
50
- * @property {string} [dbPtr] Optional branch pointer override for report API calls.
51
- */
52
-
53
- /**
54
- * ReportingTable renders report rows with search, summary, export, QR scan, and
55
- * pagination support.
56
- *
57
- * The component supports two data modes:
58
- * - controlled mode: parent supplies rows/columns/loading/pagination
59
- * - independent mode: the component fetches configuration and listing data
60
- * using `reportId`, `requestId`, or mode/submode inputs
61
- *
62
- * @param {ReportingTableProps} props
63
- * @returns {React.ReactNode}
64
- */
65
- export default function ReportingTable({
66
- patients: propsPatients,
67
- columns: propsColumns,
68
- loading: propsLoading,
69
- CustomComponents,
70
- refresh,
71
- isFixedIndex,
72
- barcodeFilterKey,
73
- showScanner,
74
- config: propsConfig,
75
- pagination: propsPagination,
76
- handlePagination: propsHandlePagination,
77
- attributes,
78
- fetchReportData,
79
- reportId,
80
- requestId,
81
- requestPayload,
82
- // dbPtr: propsDbPtr,
83
- onPaginationChange,
84
- mode,
85
- submode,
86
- replacements,
87
- isNuradesk,
88
- dbPtr
89
- }) {
90
- const [internalPatients, setInternalPatients] = useState([]);
91
- const [internalColumns, setInternalColumns] = useState([]);
92
- const [internalLoading, setInternalLoading] = useState(false);
93
- const [internalConfig, setInternalConfig] = useState({});
94
- const [internalPagination, setInternalPagination] = useState({ current: 1, pageSize: 20, total: 0 });
95
-
96
- // Independent mode is enabled when enough identifiers are present for the
97
- // table to load its own schema and data instead of relying on parent props.
98
- const shouldFetchData = !!(requestId || reportId || replacements.submode || replacements.mode);
99
- const requestPayloadKey = JSON.stringify(requestPayload || {});
100
-
101
- const propValues = (attributes && JSON.parse(attributes)) || {};
102
- const { buttonAttributes = [] } = propValues;
103
-
104
- const [query, setQuery] = useState('');
105
- const [exportData, setExportData] = useState({});
106
- const [isScannerVisible, setScannerVisible] = useState(false);
107
- const [visible, setVisible] = useState(false);
108
- const [ActiveComponent, setActiveComponent] = useState(null);
109
- const [single, setSingle] = useState({});
110
-
111
- // The table can operate in either controlled or self-fetching mode. These
112
- // selectors keep the render path mode-agnostic after initial setup.
113
- const patients = shouldFetchData ? internalPatients : propsPatients;
114
- const columns = propsColumns?.length ? propsColumns : internalColumns;
115
- const loading = shouldFetchData ? internalLoading : propsLoading;
116
- const config = Object.keys(propsConfig || {}).length ? propsConfig : internalConfig;
117
- const pagination = shouldFetchData ? internalPagination : propsPagination;
118
- const otherDetails = config?.other_details1 ? JSON.parse(config.other_details1) : {};
119
- const cols = buildDisplayColumns({
120
- columns,
121
- patients,
122
- isFixedIndex,
123
- CustomComponents: { ...CustomComponents, ...genericComponents, ...ReportingDashboardComp },
124
- refresh,
125
- otherDetails,
126
- });
127
-
128
- /**
129
- * Updates the local text filter used for in-memory search across row values.
130
- *
131
- * @param {React.ChangeEvent<HTMLInputElement>} event
132
- */
133
- function onSearch(event) {
134
- setQuery(event.target.value);
135
- }
136
-
137
- /**
138
- * Loads reporting configuration and paginated data when the component is
139
- * operating in independent mode.
140
- *
141
- * Flow:
142
- * 1. Resolve the active report key and db pointer.
143
- * 2. Fetch report configuration to build display columns/caption metadata.
144
- * 3. Fetch listing data using the current page and page size.
145
- * 4. Sync local pagination and notify the parent if needed.
146
- *
147
- * @param {ReportingTablePagination} [pager]
148
- * @returns {Promise<void>}
149
- */
150
- const loadIndependentData = async (pager) => {
151
- setInternalLoading(true);
152
- const dbPointer = dbPtr || localStorage.db_ptr;
153
-
154
- const currentPager = pager || internalPagination;
155
- let reportKey = requestId || reportId || replacements?.mode;
156
-
157
- try {
158
- if (!reportKey) return;
159
-
160
- // Load the report definition first so display columns and captions stay in
161
- // sync with the server-side report being requested.
162
- let formBody = {
163
- reportId: reportId,
164
- };
165
- if (replacements?.mode && replacements?.submode) {
166
- formBody = {
167
- mode: replacements.mode,
168
- submode: replacements.submode,
169
- // replacements: { ...replacements },
170
- };
171
- }
172
-
173
- let data = await CoreScripts.getCorescript({ ...formBody },dbPointer);
174
- if (data?.result?.display_columns) {
175
- setInternalColumns(JSON.parse(data.result.display_columns));
176
- }
177
- setInternalConfig(data);
178
- // if (isNuradesk) {
179
- // setInternalPatients(data.result || []);
180
- // setInternalColumns(Object.keys(data.result[0]).map((key) => ({ title: key, field: key })));
181
- // }
182
- // }
183
-
184
- // Then request the actual row data for the active page.
185
- const baseBody = requestPayload?.body || {};
186
- let body = {
187
- body: {
188
- ...baseBody,
189
- page: currentPager.current,
190
- limit: currentPager.pageSize,
191
- ...(requestPayload?.body ? {} : { mode, submode }),
192
- },
193
- };
194
- // if (!isNuradesk) {
195
- if (isNuradesk) {
196
- reportKey = data.result.id;
197
- body = {
198
- body: {
199
- ...replacements,
200
- page: currentPager.current,
201
- limit: currentPager.pageSize,
202
- },
203
- };
204
- }
205
- const result = await CoreScripts.getReportingLisitng(reportKey, body, dbPtr);
206
- const apiData = Array.isArray(result) ? result : Array.isArray(result?.result) ? result.result : [];
207
- const resultDetails = apiData[0] || [];
208
-
209
- setInternalPatients(resultDetails || []);
210
- // Fall back to inferred columns when the report definition does not
211
- // provide an explicit `display_columns` config.
212
- if (!data.result.display_columns && resultDetails.length > 0) {
213
- setInternalColumns(Object.keys(resultDetails[0]).map((key) => ({ title: key, field: key })));
214
- }
215
- // }
216
-
217
- const nextPagination = {
218
- current: currentPager.current,
219
- pageSize: currentPager.pageSize,
220
- total: resultDetails?.[0]?.TotalCount ?? internalPagination.total,
221
- };
222
-
223
- setInternalPagination((prev) => ({
224
- ...prev,
225
- ...nextPagination,
226
- }));
227
-
228
- if (onPaginationChange) {
229
- onPaginationChange(nextPagination);
230
- }
231
- } catch (error) {
232
- console.error('Error loading independent table data', error);
233
- } finally {
234
- setInternalLoading(false);
235
- }
236
- };
237
-
238
- useEffect(() => {
239
- if (propsPagination?.current || propsPagination?.pageSize || propsPagination?.total === 0) {
240
- setInternalPagination((prev) => ({
241
- ...prev,
242
- current: propsPagination?.current || prev.current,
243
- pageSize: propsPagination?.pageSize || prev.pageSize,
244
- total: typeof propsPagination?.total === 'number' ? propsPagination.total : prev.total,
245
- }));
246
- }
247
- }, [propsPagination?.current, propsPagination?.pageSize, propsPagination?.total]);
248
-
249
- useEffect(() => {
250
- if (shouldFetchData) {
251
- const nextPager = {
252
- current: propsPagination?.current || 1,
253
- pageSize: propsPagination?.pageSize || internalPagination.pageSize,
254
- };
255
- loadIndependentData(nextPager);
256
- }
257
- }, [requestId, reportId, mode, submode, requestPayloadKey]);
258
-
259
- /**
260
- * Routes pagination changes to either the internal loader or the parent
261
- * handler, depending on the current data mode.
262
- *
263
- * @param {ReportingTablePagination} pager
264
- */
265
- const handlePaginationInternal = (pager) => {
266
- if (shouldFetchData) {
267
- loadIndependentData(pager);
268
- } else if (propsHandlePagination) {
269
- propsHandlePagination(pager);
270
- }
271
- };
272
-
273
- useEffect(() => {
274
- if (patients) {
275
- // Export uses the visible display column structure and appends the same
276
- // summary row that appears in the table when summary columns are enabled.
277
- const exportCols = cols.map((col) => {
278
- if (col.title && typeof col.title === 'object' && col.title.props) {
279
- return { ...col, title: col.title.props.title };
280
- }
281
- return col;
282
- });
283
- const summaryCols = columns.filter((col) => col.enable_summary);
284
- let dataToExport = [...patients];
285
-
286
- if (summaryCols.length > 0) {
287
- const summaryValues = calculateSummaryValues(summaryCols, patients);
288
- const summaryRow = { isSummaryRow: true };
289
-
290
- cols.forEach((col) => {
291
- const colKey = col.field || col.key || col.dataIndex;
292
- if (colKey && !summaryRow[colKey]) {
293
- summaryRow[colKey] = '';
294
- }
295
-
296
- if (summaryValues[col.field] !== undefined) {
297
- summaryRow[col.field] = summaryValues[col.field];
298
- } else {
299
- const captionConfig = columns.find((c) => col.field && c.caption_field === col.field);
300
- if (captionConfig) {
301
- summaryRow[col.field] = captionConfig.summary_caption || '';
302
- }
303
- }
304
- });
305
- dataToExport.push(summaryRow);
306
- }
307
- let exportDatas = getExportData(dataToExport, exportCols);
308
-
309
- if (exportDatas.exportDataColumns.length && exportDatas.exportDataHeaders.length) {
310
- setExportData({ exportDatas });
311
- }
312
- }
313
- }, [patients, columns]);
314
-
315
- let filtered = patients;
316
- if (patients && query) {
317
- filtered = patients.filter((record) => {
318
- let keys = Object.keys(record);
319
- let flag = false;
320
- keys.forEach((key) => {
321
- let ele = record[key];
322
- if (ele && typeof ele === 'string' && ele.toLowerCase().indexOf(query.toLowerCase()) !== -1) {
323
- flag = true;
324
- }
325
- });
326
- return flag;
327
- });
328
- }
329
-
330
- /**
331
- * Handles successful QR scans by looking up a matching record and navigating
332
- * through the first configured action column.
333
- *
334
- * @param {string} code
335
- */
336
- const handleScanSuccess = (code) => {
337
- const matched = filtered.filter((patient) => patient[barcodeFilterKey] === code);
338
- if (matched.length) {
339
- const patient = matched[0];
340
- message.success(`Match found for ${code}, redirecting...`);
341
- const actionColumn = columns.find((col) => col.field === 'action') || columns.find((col) => col.type === 'action');
342
- if (actionColumn) {
343
- const redirectLink = getRedirectLink(actionColumn, patient,CustomComponents);
344
- window.location.href = redirectLink;
345
- }
346
- } else {
347
- Modal.warning({
348
- title: 'No matching records.',
349
- content: `No match for scanned code: ${code}`,
350
- });
351
- }
352
- };
353
-
354
- /**
355
- * Opens a reporting dashboard action component declared in `buttonAttributes`.
356
- *
357
- * @param {Object} button
358
- */
359
- const handleOpenEdit = (button) => {
360
- const componentName = button.component;
361
- const ComponentToRender = ReportingDashboardComp[componentName];
362
- if (!ComponentToRender) {
363
- console.error(`Component ${componentName} not found!`);
364
- return;
365
- }
366
- setSingle({});
367
- setActiveComponent(() => ComponentToRender);
368
- setVisible(true);
369
- };
370
-
371
- /**
372
- * Computes summary values for the current page based on per-column summary
373
- * configuration.
374
- *
375
- * Supported functions:
376
- * - `sum`
377
- * - `count`
378
- * - `avg`
379
- * - `min`
380
- * - `max`
381
- *
382
- * @param {Array<Object>} summaryCols
383
- * @param {Array<Object>} pageData
384
- * @returns {Object.<string, number>}
385
- */
386
- function calculateSummaryValues(summaryCols, pageData) {
387
- const summaryValues = {};
388
- summaryCols.forEach((col) => {
389
- const field = col.field;
390
- if (col.function === 'sum') {
391
- summaryValues[field] = pageData.reduce((total, row) => total + Number(row[field] || 0), 0);
392
- }
393
- if (col.function === 'count') {
394
- summaryValues[field] = pageData.length;
395
- }
396
- if (col.function === 'avg') {
397
- const total = pageData.reduce((sum, row) => sum + Number(row[field] || 0), 0);
398
- summaryValues[field] = pageData.length ? total / pageData.length : 0;
399
- }
400
- if (col.function === 'min') {
401
- const values = pageData.map((row) => Number(row[field] || 0));
402
- summaryValues[field] = values.length ? Math.min(...values) : 0;
403
- }
404
- if (col.function === 'max') {
405
- const values = pageData.map((row) => Number(row[field] || 0));
406
- summaryValues[field] = values.length ? Math.max(...values) : 0;
407
- }
408
- });
409
- return summaryValues;
410
- }
411
-
412
- return (
413
- <>
414
- <div
415
- className="table-header"
416
- style={{ display: 'flex', justifyContent: 'flex-end', alignItems: 'center', gap: 8, paddingTop: 10, paddingBottom: 10 }}
417
- >
418
- <div className="table-right" style={{ display: 'flex', alignItems: 'center', justifyContent: 'flex-end', gap: 8, flexWrap: 'wrap' }}>
419
- <div className="table-search" style={{ minWidth: 240, width: 280, maxWidth: '100%', flex: '0 0 auto' }}>
420
- <Search placeholder="Enter Search Value" allowClear onChange={onSearch} />
421
- </div>
422
- {showScanner && (
423
- <Button size="small" type="primary" icon={<QrcodeOutlined />} onClick={() => setScannerVisible(true)}>
424
- Scan QR
425
- </Button>
426
- )}
427
- {Array.isArray(buttonAttributes) &&
428
- buttonAttributes.map((btn, index) => (
429
- <Button key={index} size="small" type="primary" style={{ marginLeft: 8 }} onClick={() => handleOpenEdit(btn)}>
430
- {btn.title}
431
- </Button>
432
- ))}
433
- <Modal open={visible} onCancel={() => setVisible(false)} footer={null} destroyOnClose width={950} style={{ top: 10 }}>
434
- {ActiveComponent && (
435
- <ActiveComponent
436
- formContent={single}
437
- callback={() => {
438
- setVisible(false);
439
- refresh && refresh();
440
- shouldFetchData ? loadIndependentData() : fetchReportData && fetchReportData();
441
- }}
442
- />
443
- )}
444
- </Modal>
445
- <Modal open={isScannerVisible} title="Scan QR Code" footer={null} onCancel={() => setScannerVisible(false)} destroyOnClose>
446
- <QrScanner onScanSuccess={handleScanSuccess} onClose={() => setScannerVisible(false)} />
447
- </Modal>
448
- <div>
449
- {exportData.exportDatas && !isNuradesk && (
450
- <ExportReactCSV
451
- title={config.caption}
452
- fileName={`${(config.caption || 'Report').trim().replace(/\s+/g, '_')}_${moment().format('YYYY-MM-DD-HH-mm-ss-SSS')}.xlsx`}
453
- headers={exportData.exportDatas.exportDataHeaders}
454
- csvData={exportData.exportDatas.exportDataColumns}
455
- />
456
- )}
457
- </div>
458
- </div>
459
- </div>
460
- <div>
461
- <Card>
462
- {loading ? (
463
- <Skeleton active paragraph={{ rows: 6 }} />
464
- ) : (
465
- <TableComponent
466
- size="small"
467
- // Change x to '100%' or true to make it responsive to the container width
468
- // Your vertical logic (adjust '10' based on how many rows fit on your screen)
469
- scroll={{ x: 'max-content', y: (filtered?.length || 0) > 11 ? 400 : undefined }}
470
- rowKey={(record) => record.OpNo}
471
- dataSource={filtered}
472
- columns={cols}
473
- sticky
474
- pagination={false}
475
- summary={(pageData) => {
476
- const summaryCols = columns.filter((col) => col.enable_summary);
477
- if (!summaryCols.length) return null;
478
- const summaryValues = calculateSummaryValues(summaryCols, pageData);
479
- return (
480
- <Table.Summary.Row className="report-summary-row">
481
- {cols.map((col, index) => {
482
- if (summaryValues[col.field] !== undefined) {
483
- return (
484
- <Table.Summary.Cell key={index}>
485
- <strong style={{ fontWeight: 900 }}>{summaryValues[col.field]}</strong>
486
- </Table.Summary.Cell>
487
- );
488
- }
489
- const captionConfig = columns.find((c) => col.field && c.caption_field === col.field);
490
- if (captionConfig) {
491
- return (
492
- <Table.Summary.Cell key={index}>
493
- <strong style={{ fontWeight: 900 }}>{captionConfig.summary_caption || ''}</strong>
494
- </Table.Summary.Cell>
495
- );
496
- }
497
- return <Table.Summary.Cell key={index} />;
498
- })}
499
- </Table.Summary.Row>
500
- );
501
- }}
502
- />
503
- )}
504
- <div style={{ display: 'flex', justifyContent: 'flex-end', marginTop: 8 }}>
505
- <Pagination
506
- showSizeChanger
507
- current={pagination?.current}
508
- pageSize={pagination?.pageSize}
509
- total={pagination?.total}
510
- pageSizeOptions={[20, 30, 50, 100]}
511
- onChange={(page, pageSize) => handlePaginationInternal({ current: page, pageSize })}
512
- />
513
- </div>
514
- <p className="size-hint">{patients ? patients.length : 0} records.</p>
515
- </Card>
516
- </div>
517
- </>
518
- );
519
- }
1
+ import React, { useState, useEffect } from 'react';
2
+ import { Table, Skeleton, Input, Modal, message, Pagination } from 'antd';
3
+ import { QrcodeOutlined } from '@ant-design/icons';
4
+ import { ExportReactCSV, getExportData, Card, TableComponent, QrScanner } from './../../../../lib/';
5
+ import moment from 'moment-timezone';
6
+ import { CoreScripts } from './../../../../models/';
7
+ import Button from '../../../../lib/elements/basic/button/button';
8
+ import buildDisplayColumns from './display-columns/build-display-columns';
9
+ import { getRedirectLink } from './display-columns/display-cell-renderer';
10
+ import * as ReportingDashboardComp from '../index';
11
+
12
+ const { Search } = Input;
13
+ const genericComponents = require('./../../../../lib');
14
+
15
+ /**
16
+ * @typedef {Object} ReportingTablePagination
17
+ * @property {number} [current] Current page number.
18
+ * @property {number} [pageSize] Number of rows per page.
19
+ * @property {number} [total] Total number of records available.
20
+ */
21
+
22
+ /**
23
+ * @typedef {Object} ReportingTableRequestPayload
24
+ * @property {Object} [body] Request body sent when loading reporting data.
25
+ */
26
+
27
+ /**
28
+ * @typedef {Object} ReportingTableProps
29
+ * @property {Array<Object>} [patients] Preloaded table rows supplied by a parent component.
30
+ * @property {Array<Object>} [columns] Preconfigured display column definitions.
31
+ * @property {boolean} [loading] External loading state used when parent owns data fetching.
32
+ * @property {Object.<string, React.ComponentType<any>>} [CustomComponents] Custom renderers/actions available to display columns.
33
+ * @property {Function} [refresh] Refresh callback passed down to nested action components.
34
+ * @property {boolean} [isFixedIndex] Enables fixed index rendering in generated columns.
35
+ * @property {string} [barcodeFilterKey] Record field compared against scanned QR/barcode values.
36
+ * @property {boolean} [showScanner] Controls whether the QR scanner button is shown.
37
+ * @property {Object} [config] Reporting configuration returned by the core script.
38
+ * @property {ReportingTablePagination} [pagination] External pagination state.
39
+ * @property {(pager: ReportingTablePagination) => void} [handlePagination] Parent-owned pagination handler.
40
+ * @property {string} [attributes] JSON string containing extra button/config attributes.
41
+ * @property {Function} [fetchReportData] Parent refresh callback used in controlled mode.
42
+ * @property {string|number} [reportId] Report id used to load configuration and rows.
43
+ * @property {string|number} [requestId] Alternative report request id used for independent loading.
44
+ * @property {ReportingTableRequestPayload} [requestPayload] Optional request payload override for listing calls.
45
+ * @property {(pager: ReportingTablePagination) => void} [onPaginationChange] Callback fired after independent pagination updates.
46
+ * @property {string} [mode] Report mode used when fetching by mode/submode.
47
+ * @property {string} [submode] Report submode used when fetching by mode/submode.
48
+ * @property {Object} [replacements] Dynamic replacements used by some report APIs and action links.
49
+ * @property {boolean} [isNuradesk] Switches to Nuradesk-specific report loading behavior.
50
+ * @property {string} [dbPtr] Optional branch pointer override for report API calls.
51
+ */
52
+
53
+ /**
54
+ * ReportingTable renders report rows with search, summary, export, QR scan, and
55
+ * pagination support.
56
+ *
57
+ * The component supports two data modes:
58
+ * - controlled mode: parent supplies rows/columns/loading/pagination
59
+ * - independent mode: the component fetches configuration and listing data
60
+ * using `reportId`, `requestId`, or mode/submode inputs
61
+ *
62
+ * @param {ReportingTableProps} props
63
+ * @returns {React.ReactNode}
64
+ */
65
+ export default function ReportingTable({
66
+ patients: propsPatients,
67
+ columns: propsColumns,
68
+ loading: propsLoading,
69
+ CustomComponents,
70
+ refresh,
71
+ isFixedIndex,
72
+ barcodeFilterKey,
73
+ showScanner,
74
+ config: propsConfig,
75
+ pagination: propsPagination,
76
+ handlePagination: propsHandlePagination,
77
+ attributes,
78
+ fetchReportData,
79
+ reportId,
80
+ requestId,
81
+ requestPayload,
82
+ // dbPtr: propsDbPtr,
83
+ onPaginationChange,
84
+ mode,
85
+ submode,
86
+ replacements,
87
+ isNuradesk,
88
+ dbPtr
89
+ }) {
90
+ const [internalPatients, setInternalPatients] = useState([]);
91
+ const [internalColumns, setInternalColumns] = useState([]);
92
+ const [internalLoading, setInternalLoading] = useState(false);
93
+ const [internalConfig, setInternalConfig] = useState({});
94
+ const [internalPagination, setInternalPagination] = useState({ current: 1, pageSize: 20, total: 0 });
95
+
96
+ // Independent mode is enabled when enough identifiers are present for the
97
+ // table to load its own schema and data instead of relying on parent props.
98
+ const shouldFetchData = !!(requestId || reportId || replacements.submode || replacements.mode);
99
+ const requestPayloadKey = JSON.stringify(requestPayload || {});
100
+
101
+ const propValues = (attributes && JSON.parse(attributes)) || {};
102
+ const { buttonAttributes = [] } = propValues;
103
+
104
+ const [query, setQuery] = useState('');
105
+ const [exportData, setExportData] = useState({});
106
+ const [isScannerVisible, setScannerVisible] = useState(false);
107
+ const [visible, setVisible] = useState(false);
108
+ const [ActiveComponent, setActiveComponent] = useState(null);
109
+ const [single, setSingle] = useState({});
110
+
111
+ // The table can operate in either controlled or self-fetching mode. These
112
+ // selectors keep the render path mode-agnostic after initial setup.
113
+ const patients = shouldFetchData ? internalPatients : propsPatients;
114
+ const columns = propsColumns?.length ? propsColumns : internalColumns;
115
+ const loading = shouldFetchData ? internalLoading : propsLoading;
116
+ const config = Object.keys(propsConfig || {}).length ? propsConfig : internalConfig;
117
+ const pagination = shouldFetchData ? internalPagination : propsPagination;
118
+ const otherDetails = config?.other_details1 ? JSON.parse(config.other_details1) : {};
119
+ const cols = buildDisplayColumns({
120
+ columns,
121
+ patients,
122
+ isFixedIndex,
123
+ CustomComponents: { ...CustomComponents, ...genericComponents, ...ReportingDashboardComp },
124
+ refresh,
125
+ otherDetails,
126
+ });
127
+
128
+ /**
129
+ * Updates the local text filter used for in-memory search across row values.
130
+ *
131
+ * @param {React.ChangeEvent<HTMLInputElement>} event
132
+ */
133
+ function onSearch(event) {
134
+ setQuery(event.target.value);
135
+ }
136
+
137
+ /**
138
+ * Loads reporting configuration and paginated data when the component is
139
+ * operating in independent mode.
140
+ *
141
+ * Flow:
142
+ * 1. Resolve the active report key and db pointer.
143
+ * 2. Fetch report configuration to build display columns/caption metadata.
144
+ * 3. Fetch listing data using the current page and page size.
145
+ * 4. Sync local pagination and notify the parent if needed.
146
+ *
147
+ * @param {ReportingTablePagination} [pager]
148
+ * @returns {Promise<void>}
149
+ */
150
+ const loadIndependentData = async (pager) => {
151
+ setInternalLoading(true);
152
+ const dbPointer = dbPtr || localStorage.db_ptr;
153
+
154
+ const currentPager = pager || internalPagination;
155
+ let reportKey = requestId || reportId || replacements?.mode;
156
+
157
+ try {
158
+ if (!reportKey) return;
159
+
160
+ // Load the report definition first so display columns and captions stay in
161
+ // sync with the server-side report being requested.
162
+ let formBody = {
163
+ reportId: reportId,
164
+ };
165
+ if (replacements?.mode && replacements?.submode) {
166
+ formBody = {
167
+ mode: replacements.mode,
168
+ submode: replacements.submode,
169
+ // replacements: { ...replacements },
170
+ };
171
+ }
172
+
173
+ let data = await CoreScripts.getCorescript({ ...formBody },dbPointer);
174
+ if (data?.result?.display_columns) {
175
+ setInternalColumns(JSON.parse(data.result.display_columns));
176
+ }
177
+ setInternalConfig(data);
178
+ // if (isNuradesk) {
179
+ // setInternalPatients(data.result || []);
180
+ // setInternalColumns(Object.keys(data.result[0]).map((key) => ({ title: key, field: key })));
181
+ // }
182
+ // }
183
+
184
+ // Then request the actual row data for the active page.
185
+ const baseBody = requestPayload?.body || {};
186
+ let body = {
187
+ body: {
188
+ ...baseBody,
189
+ page: currentPager.current,
190
+ limit: currentPager.pageSize,
191
+ ...(requestPayload?.body ? {} : { mode, submode }),
192
+ },
193
+ };
194
+ // if (!isNuradesk) {
195
+ if (isNuradesk) {
196
+ reportKey = data.result.id;
197
+ body = {
198
+ body: {
199
+ ...replacements,
200
+ page: currentPager.current,
201
+ limit: currentPager.pageSize,
202
+ },
203
+ };
204
+ }
205
+ const result = await CoreScripts.getReportingLisitng(reportKey, body, dbPtr);
206
+ const apiData = Array.isArray(result) ? result : Array.isArray(result?.result) ? result.result : [];
207
+ const resultDetails = apiData[0] || [];
208
+
209
+ setInternalPatients(resultDetails || []);
210
+ // Fall back to inferred columns when the report definition does not
211
+ // provide an explicit `display_columns` config.
212
+ if (!data.result.display_columns && resultDetails.length > 0) {
213
+ setInternalColumns(Object.keys(resultDetails[0]).map((key) => ({ title: key, field: key })));
214
+ }
215
+ // }
216
+
217
+ const nextPagination = {
218
+ current: currentPager.current,
219
+ pageSize: currentPager.pageSize,
220
+ total: resultDetails?.[0]?.TotalCount ?? internalPagination.total,
221
+ };
222
+
223
+ setInternalPagination((prev) => ({
224
+ ...prev,
225
+ ...nextPagination,
226
+ }));
227
+
228
+ if (onPaginationChange) {
229
+ onPaginationChange(nextPagination);
230
+ }
231
+ } catch (error) {
232
+ console.error('Error loading independent table data', error);
233
+ } finally {
234
+ setInternalLoading(false);
235
+ }
236
+ };
237
+
238
+ useEffect(() => {
239
+ if (propsPagination?.current || propsPagination?.pageSize || propsPagination?.total === 0) {
240
+ setInternalPagination((prev) => ({
241
+ ...prev,
242
+ current: propsPagination?.current || prev.current,
243
+ pageSize: propsPagination?.pageSize || prev.pageSize,
244
+ total: typeof propsPagination?.total === 'number' ? propsPagination.total : prev.total,
245
+ }));
246
+ }
247
+ }, [propsPagination?.current, propsPagination?.pageSize, propsPagination?.total]);
248
+
249
+ useEffect(() => {
250
+ if (shouldFetchData) {
251
+ const nextPager = {
252
+ current: propsPagination?.current || 1,
253
+ pageSize: propsPagination?.pageSize || internalPagination.pageSize,
254
+ };
255
+ loadIndependentData(nextPager);
256
+ }
257
+ }, [requestId, reportId, mode, submode, requestPayloadKey]);
258
+
259
+ /**
260
+ * Routes pagination changes to either the internal loader or the parent
261
+ * handler, depending on the current data mode.
262
+ *
263
+ * @param {ReportingTablePagination} pager
264
+ */
265
+ const handlePaginationInternal = (pager) => {
266
+ if (shouldFetchData) {
267
+ loadIndependentData(pager);
268
+ } else if (propsHandlePagination) {
269
+ propsHandlePagination(pager);
270
+ }
271
+ };
272
+
273
+ useEffect(() => {
274
+ if (patients) {
275
+ // Export uses the visible display column structure and appends the same
276
+ // summary row that appears in the table when summary columns are enabled.
277
+ const exportCols = cols.map((col) => {
278
+ if (col.title && typeof col.title === 'object' && col.title.props) {
279
+ return { ...col, title: col.title.props.title };
280
+ }
281
+ return col;
282
+ });
283
+ const summaryCols = columns.filter((col) => col.enable_summary);
284
+ let dataToExport = [...patients];
285
+
286
+ if (summaryCols.length > 0) {
287
+ const summaryValues = calculateSummaryValues(summaryCols, patients);
288
+ const summaryRow = { isSummaryRow: true };
289
+
290
+ cols.forEach((col) => {
291
+ const colKey = col.field || col.key || col.dataIndex;
292
+ if (colKey && !summaryRow[colKey]) {
293
+ summaryRow[colKey] = '';
294
+ }
295
+
296
+ if (summaryValues[col.field] !== undefined) {
297
+ summaryRow[col.field] = summaryValues[col.field];
298
+ } else {
299
+ const captionConfig = columns.find((c) => col.field && c.caption_field === col.field);
300
+ if (captionConfig) {
301
+ summaryRow[col.field] = captionConfig.summary_caption || '';
302
+ }
303
+ }
304
+ });
305
+ dataToExport.push(summaryRow);
306
+ }
307
+ let exportDatas = getExportData(dataToExport, exportCols);
308
+
309
+ if (exportDatas.exportDataColumns.length && exportDatas.exportDataHeaders.length) {
310
+ setExportData({ exportDatas });
311
+ }
312
+ }
313
+ }, [patients, columns]);
314
+
315
+ let filtered = patients;
316
+ if (patients && query) {
317
+ filtered = patients.filter((record) => {
318
+ let keys = Object.keys(record);
319
+ let flag = false;
320
+ keys.forEach((key) => {
321
+ let ele = record[key];
322
+ if (ele && typeof ele === 'string' && ele.toLowerCase().indexOf(query.toLowerCase()) !== -1) {
323
+ flag = true;
324
+ }
325
+ });
326
+ return flag;
327
+ });
328
+ }
329
+
330
+ /**
331
+ * Handles successful QR scans by looking up a matching record and navigating
332
+ * through the first configured action column.
333
+ *
334
+ * @param {string} code
335
+ */
336
+ const handleScanSuccess = (code) => {
337
+ const matched = filtered.filter((patient) => patient[barcodeFilterKey] === code);
338
+ if (matched.length) {
339
+ const patient = matched[0];
340
+ message.success(`Match found for ${code}, redirecting...`);
341
+ const actionColumn = columns.find((col) => col.field === 'action') || columns.find((col) => col.type === 'action');
342
+ if (actionColumn) {
343
+ const redirectLink = getRedirectLink(actionColumn, patient,CustomComponents);
344
+ window.location.href = redirectLink;
345
+ }
346
+ } else {
347
+ Modal.warning({
348
+ title: 'No matching records.',
349
+ content: `No match for scanned code: ${code}`,
350
+ });
351
+ }
352
+ };
353
+
354
+ /**
355
+ * Opens a reporting dashboard action component declared in `buttonAttributes`.
356
+ *
357
+ * @param {Object} button
358
+ */
359
+ const handleOpenEdit = (button) => {
360
+ const componentName = button.component;
361
+ const ComponentToRender = ReportingDashboardComp[componentName];
362
+ if (!ComponentToRender) {
363
+ console.error(`Component ${componentName} not found!`);
364
+ return;
365
+ }
366
+ setSingle({});
367
+ setActiveComponent(() => ComponentToRender);
368
+ setVisible(true);
369
+ };
370
+
371
+ /**
372
+ * Computes summary values for the current page based on per-column summary
373
+ * configuration.
374
+ *
375
+ * Supported functions:
376
+ * - `sum`
377
+ * - `count`
378
+ * - `avg`
379
+ * - `min`
380
+ * - `max`
381
+ *
382
+ * @param {Array<Object>} summaryCols
383
+ * @param {Array<Object>} pageData
384
+ * @returns {Object.<string, number>}
385
+ */
386
+ function calculateSummaryValues(summaryCols, pageData) {
387
+ const summaryValues = {};
388
+ summaryCols.forEach((col) => {
389
+ const field = col.field;
390
+ if (col.function === 'sum') {
391
+ summaryValues[field] = pageData.reduce((total, row) => total + Number(row[field] || 0), 0);
392
+ }
393
+ if (col.function === 'count') {
394
+ summaryValues[field] = pageData.length;
395
+ }
396
+ if (col.function === 'avg') {
397
+ const total = pageData.reduce((sum, row) => sum + Number(row[field] || 0), 0);
398
+ summaryValues[field] = pageData.length ? total / pageData.length : 0;
399
+ }
400
+ if (col.function === 'min') {
401
+ const values = pageData.map((row) => Number(row[field] || 0));
402
+ summaryValues[field] = values.length ? Math.min(...values) : 0;
403
+ }
404
+ if (col.function === 'max') {
405
+ const values = pageData.map((row) => Number(row[field] || 0));
406
+ summaryValues[field] = values.length ? Math.max(...values) : 0;
407
+ }
408
+ });
409
+ return summaryValues;
410
+ }
411
+
412
+ return (
413
+ <>
414
+ <div
415
+ className="table-header"
416
+ style={{ display: 'flex', justifyContent: 'flex-end', alignItems: 'center', gap: 8, paddingTop: 10, paddingBottom: 10 }}
417
+ >
418
+ <div className="table-right" style={{ display: 'flex', alignItems: 'center', justifyContent: 'flex-end', gap: 8, flexWrap: 'wrap' }}>
419
+ <div className="table-search" style={{ minWidth: 240, width: 280, maxWidth: '100%', flex: '0 0 auto' }}>
420
+ <Search placeholder="Enter Search Value" allowClear onChange={onSearch} />
421
+ </div>
422
+ {showScanner && (
423
+ <Button size="small" type="primary" icon={<QrcodeOutlined />} onClick={() => setScannerVisible(true)}>
424
+ Scan QR
425
+ </Button>
426
+ )}
427
+ {Array.isArray(buttonAttributes) &&
428
+ buttonAttributes.map((btn, index) => (
429
+ <Button key={index} size="small" type="primary" style={{ marginLeft: 8 }} onClick={() => handleOpenEdit(btn)}>
430
+ {btn.title}
431
+ </Button>
432
+ ))}
433
+ <Modal open={visible} onCancel={() => setVisible(false)} footer={null} destroyOnClose width={950} style={{ top: 10 }}>
434
+ {ActiveComponent && (
435
+ <ActiveComponent
436
+ formContent={single}
437
+ callback={() => {
438
+ setVisible(false);
439
+ refresh && refresh();
440
+ shouldFetchData ? loadIndependentData() : fetchReportData && fetchReportData();
441
+ }}
442
+ />
443
+ )}
444
+ </Modal>
445
+ <Modal open={isScannerVisible} title="Scan QR Code" footer={null} onCancel={() => setScannerVisible(false)} destroyOnClose>
446
+ <QrScanner onScanSuccess={handleScanSuccess} onClose={() => setScannerVisible(false)} />
447
+ </Modal>
448
+ <div>
449
+ {exportData.exportDatas && !isNuradesk && (
450
+ <ExportReactCSV
451
+ title={config.caption}
452
+ fileName={`${(config.caption || 'Report').trim().replace(/\s+/g, '_')}_${moment().format('YYYY-MM-DD-HH-mm-ss-SSS')}.xlsx`}
453
+ headers={exportData.exportDatas.exportDataHeaders}
454
+ csvData={exportData.exportDatas.exportDataColumns}
455
+ />
456
+ )}
457
+ </div>
458
+ </div>
459
+ </div>
460
+ <div>
461
+ <Card>
462
+ {loading ? (
463
+ <Skeleton active paragraph={{ rows: 6 }} />
464
+ ) : (
465
+ <TableComponent
466
+ size="small"
467
+ // Change x to '100%' or true to make it responsive to the container width
468
+ // Your vertical logic (adjust '10' based on how many rows fit on your screen)
469
+ scroll={{ x: 'max-content', y: (filtered?.length || 0) > 11 ? 400 : undefined }}
470
+ rowKey={(record) => record.OpNo}
471
+ dataSource={filtered}
472
+ columns={cols}
473
+ sticky
474
+ pagination={false}
475
+ summary={(pageData) => {
476
+ const summaryCols = columns.filter((col) => col.enable_summary);
477
+ if (!summaryCols.length) return null;
478
+ const summaryValues = calculateSummaryValues(summaryCols, pageData);
479
+ return (
480
+ <Table.Summary.Row className="report-summary-row">
481
+ {cols.map((col, index) => {
482
+ if (summaryValues[col.field] !== undefined) {
483
+ return (
484
+ <Table.Summary.Cell key={index}>
485
+ <strong style={{ fontWeight: 900 }}>{summaryValues[col.field]}</strong>
486
+ </Table.Summary.Cell>
487
+ );
488
+ }
489
+ const captionConfig = columns.find((c) => col.field && c.caption_field === col.field);
490
+ if (captionConfig) {
491
+ return (
492
+ <Table.Summary.Cell key={index}>
493
+ <strong style={{ fontWeight: 900 }}>{captionConfig.summary_caption || ''}</strong>
494
+ </Table.Summary.Cell>
495
+ );
496
+ }
497
+ return <Table.Summary.Cell key={index} />;
498
+ })}
499
+ </Table.Summary.Row>
500
+ );
501
+ }}
502
+ />
503
+ )}
504
+ <div style={{ display: 'flex', justifyContent: 'flex-end', marginTop: 8 }}>
505
+ <Pagination
506
+ showSizeChanger
507
+ current={pagination?.current}
508
+ pageSize={pagination?.pageSize}
509
+ total={pagination?.total}
510
+ pageSizeOptions={[20, 30, 50, 100]}
511
+ onChange={(page, pageSize) => handlePaginationInternal({ current: page, pageSize })}
512
+ />
513
+ </div>
514
+ <p className="size-hint">{patients ? patients.length : 0} records.</p>
515
+ </Card>
516
+ </div>
517
+ </>
518
+ );
519
+ }