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,584 +1,635 @@
1
- import React, { useState, useCallback, useEffect } from 'react';
2
- import { Space, Popconfirm, Input, Drawer, Skeleton, Collapse, message, Tag, Empty } from 'antd';
3
- import { ReloadOutlined, DeleteOutlined, EditOutlined, PlusCircleFilled, CopyOutlined } from '@ant-design/icons';
4
- import { Button, Card, Switch, DraggableWrapper } from '../../../../lib';
5
- // for draggable menu list import { DndProvider } from "react-dnd";
6
- import { DndProvider } from 'react-dnd';
7
- import { Link, useParams, useLocation } from 'react-router-dom';
8
- import { HTML5Backend } from 'react-dnd-html5-backend';
9
-
10
- import './menu-lists.scss';
11
-
12
- const { Search } = Input;
13
- const { Panel } = Collapse;
14
-
15
- const MenuLists = ({ model, match, relativeAdd = false, additional_queries = [], disableAddModal = model.disableAddModal }) => {
16
- const actions = [];
17
-
18
- match = useParams();
19
- const id = match.id;
20
- // const location = useLocation();
21
- const step = parseInt(new URLSearchParams(location.search).get('step')) || 1;
22
-
23
- const [records, setRecords] = useState([]);
24
- const [originalRecords, setOriginalRecords] = useState([]);
25
- const [loading, setLoading] = useState(false);
26
- const [drawerVisible, setDrawerVisible] = useState(false);
27
- const [single, setSingle] = useState({});
28
- const [drawerTitle, setDrawerTitle] = useState('');
29
- const [selectedRecord, setSelectedRecord] = useState(null);
30
- const [query, setQuery] = useState('');
31
- const [dragMode, setDragMode] = useState(false);
32
- const [orderChanged, setOrderChanged] = useState(false);
33
-
34
- const [nextId, setNextId] = useState(10000);
35
-
36
- useEffect(() => {
37
- loadMenus();
38
- }, [id]);
39
-
40
- const loadMenus = () => {
41
- setLoading(true);
42
-
43
- model
44
- .getCoreMenuLists
45
- // {
46
- // queries: [...additional_queries, { field: 'step', value: step }, { field: 'header_id', value: null }],
47
- // }
48
- ()
49
- .then((res) => {
50
- const sorted = (res.result || []).sort((a, b) => a.order - b.order);
51
- setRecords(sorted);
52
- setOriginalRecords(sorted);
53
- setLoading(false);
54
- })
55
- .catch(() => setLoading(false));
56
- };
57
-
58
- /**
59
- *
60
- */
61
- const getRecords = () => {
62
- let firstStepQueries = [
63
- {
64
- field: 'step',
65
- value: step,
66
- },
67
- ];
68
-
69
- let config = {
70
- queries: [...additional_queries, ...firstStepQueries],
71
- };
72
-
73
- if (!additional_queries.length) {
74
- config.queries.push({
75
- field: 'header_id',
76
- value: null,
77
- });
78
- }
79
-
80
- setLoading(true);
81
-
82
- // Get the records
83
- return model
84
- .get(config)
85
-
86
- .then((res) => {
87
- // setRecords(res.result);
88
- var result = res.result;
89
-
90
- if (Array.isArray(result)) {
91
- setRecords(result.sort((a, b) => a.order - b.order));
92
- // } else {
93
- // setRecords([])
94
- }
95
-
96
- setLoading(false);
97
- })
98
- .catch((error) => {
99
- setLoading(false);
100
-
101
- console.log(error);
102
- });
103
- };
104
-
105
- const toggleDragMode = (checked) => {
106
- setDragMode(checked);
107
- if (checked) {
108
- message.info('Drag mode enabled - Drag items anywhere, including the "Drop here" zones!');
109
- }
110
- };
111
-
112
- // Function to find and remove an item from anywhere in the tree
113
- const findAndRemoveItem = (items, itemId) => {
114
- let foundItem = null;
115
-
116
- const removeRecursive = (arr) => {
117
- for (let i = 0; i < arr.length; i++) {
118
- if (arr[i].id === itemId) {
119
- foundItem = { ...arr[i] };
120
- arr.splice(i, 1);
121
- return true;
122
- }
123
- if (arr[i].sub_menus && arr[i].sub_menus.length > 0) {
124
- if (removeRecursive(arr[i].sub_menus)) {
125
- return true;
126
- }
127
- }
128
- }
129
- return false;
130
- };
131
-
132
- const newItems = JSON.parse(JSON.stringify(items));
133
- removeRecursive(newItems);
134
- return { newItems, foundItem };
135
- };
136
-
137
- // Function to insert item at specific location
138
- const insertItemAt = (items, item, targetLevel, targetParentId, targetIndex) => {
139
- const newItems = JSON.parse(JSON.stringify(items));
140
-
141
- // Update item properties for new location
142
- const updatedItem = {
143
- ...item,
144
- step: targetLevel,
145
- header_id: targetParentId,
146
- };
147
-
148
- if (targetLevel === 1) {
149
- // Insert at main level
150
- newItems.splice(targetIndex, 0, updatedItem);
151
- } else {
152
- // Insert into nested level
153
- const insertRecursive = (arr) => {
154
- for (let i = 0; i < arr.length; i++) {
155
- if (arr[i].id === targetParentId) {
156
- if (!arr[i].sub_menus) arr[i].sub_menus = [];
157
- arr[i].sub_menus.splice(targetIndex, 0, updatedItem);
158
- return true;
159
- }
160
- if (arr[i].sub_menus && arr[i].sub_menus.length > 0) {
161
- if (insertRecursive(arr[i].sub_menus)) {
162
- return true;
163
- }
164
- }
165
- }
166
- return false;
167
- };
168
- insertRecursive(newItems);
169
- }
170
-
171
- return newItems;
172
- };
173
-
174
- const handleCrossLevelMove = useCallback(
175
- (draggedItem, dropTarget) => {
176
- const { targetLevel, targetParentId, targetIndex } = dropTarget;
177
-
178
- // Prevent moving item into itself or its children
179
- if (draggedItem.id === targetParentId) {
180
- message.warning('Cannot move item into itself');
181
- return;
182
- }
183
-
184
- // Remove item from original location
185
- const { newItems: itemsAfterRemove, foundItem } = findAndRemoveItem(records, draggedItem.id);
186
-
187
- if (!foundItem) {
188
- message.error('Could not find item to move');
189
- return;
190
- }
191
-
192
- // Insert item at new location
193
- const finalItems = insertItemAt(itemsAfterRemove, foundItem, targetLevel, targetParentId, targetIndex);
194
-
195
- setRecords(finalItems);
196
- setOrderChanged(true);
197
- message.success(`Moved "${foundItem.name}" to level ${targetLevel}`);
198
- },
199
- [records]
200
- );
201
-
202
- const deleteRecord = (rec) => {
203
- model.delete(rec).then(loadMenus);
204
- };
205
-
206
- const filtered = records.filter((r) => r.caption?.toUpperCase().includes(query.toUpperCase()));
207
-
208
- const visibleItems = filtered;
209
-
210
- const onSearch = (event) => {
211
- setQuery(event.target.value);
212
- };
213
-
214
- // ------------------------
215
- // Recursive function to build payload
216
- // ------------------------
217
- const buildUpdatedOrderPayload = (menus, originalMenus) => {
218
- const result = [];
219
-
220
- menus.forEach((menu, index) => {
221
- const original = findMenuById(originalMenus, menu.id);
222
-
223
- const newOrder = index + 1;
224
- const oldOrder = original ? original.order : null;
225
-
226
- const orderChanged = oldOrder !== newOrder;
227
- const parentChanged = original?.header_id !== menu.header_id;
228
-
229
- // check sub menus
230
- let updatedSubMenus = [];
231
- if (menu.sub_menus?.length) {
232
- updatedSubMenus = buildUpdatedOrderPayload(menu.sub_menus, originalMenus);
233
- }
234
-
235
- // include only if changed OR has changed children
236
- if (orderChanged || parentChanged || updatedSubMenus.length) {
237
- result.push({
238
- id: menu.id,
239
- order: newOrder,
240
- header_id: menu.header_id,
241
- name: menu.name,
242
- ...(updatedSubMenus.length && { sub_menus: updatedSubMenus }),
243
- });
244
- }
245
- });
246
-
247
- return result;
248
- };
249
-
250
- // ------------------------
251
- // Find menu by id (recursive)
252
- // ------------------------
253
- const findMenuById = (menus, id) => {
254
- for (const menu of menus || []) {
255
- if (menu.id === id) return menu;
256
-
257
- if (menu.sub_menus?.length) {
258
- const found = findMenuById(menu.sub_menus, id);
259
- if (found) return found;
260
- }
261
- }
262
- return null;
263
- };
264
-
265
- // ------------------------
266
- // Save order function
267
- // ------------------------
268
- const saveOrder = () => {
269
- const updatedMenus = buildUpdatedOrderPayload(records, originalRecords);
270
-
271
- if (!updatedMenus.length) {
272
- message.info('No order changes to save');
273
- return;
274
- }
275
-
276
- const payload = { menus: updatedMenus };
277
-
278
- model
279
- .saveOrder({ formBody: payload })
280
- .then(() => {
281
- setOrderChanged(false);
282
- loadMenus(); // refresh from backend
283
- })
284
- .catch(console.error);
285
- };
286
-
287
- const movePanel = useCallback(
288
- (from, to) => {
289
- if (!dragMode) return;
290
- if (from === to) return;
291
-
292
- setRecords((prevRecords) => {
293
- const updated = [...prevRecords];
294
- const [moved] = updated.splice(from, 1);
295
- updated.splice(to, 0, moved);
296
- return updated;
297
- });
298
-
299
- setOrderChanged(true);
300
- },
301
- [dragMode]
302
- );
303
-
304
- return (
305
- <Card className="generic-list">
306
- <div style={{ marginBottom: 16 }}>
307
- <div style={{ display: 'flex', justifyContent: 'space-between', alignItems: 'center', marginBottom: 16 }}>
308
- <Search placeholder="Enter Search Value" allowClear style={{ width: 300, marginBottom: '0px' }} onChange={onSearch} />
309
-
310
- <Space size="small">
311
- <Button onClick={getRecords} size={'small'} type="default">
312
- <ReloadOutlined />
313
- </Button>
314
-
315
- <Switch checked={dragMode} onChange={toggleDragMode} checkedChildren="Order On" unCheckedChildren="Order Off" />
316
-
317
- {dragMode && orderChanged && (
318
- <Button onClick={saveOrder} type="primary" size="small">
319
- Save Order
320
- </Button>
321
- )}
322
-
323
- <Button
324
- type="primary"
325
- size="small"
326
- onClick={() => {
327
- setSelectedRecord({ step: 1, header_id: null });
328
- setDrawerTitle('Create New Menu');
329
- setDrawerVisible(true);
330
- }}
331
- >
332
- Create New Menu
333
- </Button>
334
- </Space>
335
- </div>
336
-
337
- {dragMode && (
338
- <div
339
- style={{
340
- padding: '8px 12px',
341
- backgroundColor: '#e6f7ff',
342
- borderRadius: 4,
343
- border: '1px solid #91d5ff',
344
- fontSize: 13,
345
- }}
346
- >
347
- <strong>💡 Tips:</strong>
348
- <ul style={{ margin: '4px 0', paddingLeft: 20 }}>
349
- <li>Drag between items to reorder at the same level</li>
350
- <li>Drop on the green " Drop here" zone to make it a child of that item</li>
351
- <li>Works at all levels - unlimited nesting supported!</li>
352
- </ul>
353
- </div>
354
- )}
355
- </div>
356
- {loading ? (
357
- <Skeleton active />
358
- ) : (
359
- <>
360
- <>
361
- {/* {!view ? ( */}
362
- <Card>
363
- <DndProvider backend={HTML5Backend}>
364
- <Collapse accordion>
365
- {visibleItems && visibleItems.length > 0 ? (
366
- visibleItems.map((item, index) => (
367
- <Panel
368
- key={item.id}
369
- header={
370
- <DraggableWrapper
371
- id={item.id}
372
- index={index}
373
- movePanel={movePanel}
374
- item={item}
375
- dragEnabled={dragMode}
376
- level={1}
377
- parentId={null}
378
- onCrossLevelMove={handleCrossLevelMove}
379
- canAcceptChildren={true}
380
- />
381
- }
382
- showArrow={item.sub_menus && item.sub_menus.length > 0}
383
- extra={panelActions(item, model, setSelectedRecord, setDrawerTitle, setDrawerVisible, deleteRecord)}
384
- >
385
- {item.sub_menus && item.sub_menus.length > 0 && (
386
- <NestedMenu
387
- parentId={item.id}
388
- step={item.step + 1}
389
- items={item.sub_menus || []}
390
- model={model}
391
- dragMode={dragMode}
392
- setSelectedRecord={setSelectedRecord}
393
- setDrawerTitle={setDrawerTitle}
394
- setDrawerVisible={setDrawerVisible}
395
- deleteRecord={deleteRecord}
396
- level={2}
397
- onCrossLevelMove={handleCrossLevelMove}
398
- onChange={(subMenus) => {
399
- const updated = records.map((r) => (r.id === item.id ? { ...r, sub_menus: subMenus } : r));
400
- setRecords(updated);
401
- setOrderChanged(true);
402
- }}
403
- />
404
- )}
405
- </Panel>
406
- ))
407
- ) : (
408
- <div style={{ textAlign: 'center', padding: '40px 0' }}>
409
- <Empty description="No Menu Items" />
410
- </div>
411
- )}
412
- </Collapse>
413
- </DndProvider>
414
- </Card>
415
- {/* ) : (
416
- <CardList model={model} data={filtered ? filtered : records} />
417
- )} */}
418
- </>
419
- </>
420
- )}
421
-
422
- <Drawer title={drawerTitle} open={drawerVisible} width="80%" destroyOnClose onClose={() => setDrawerVisible(false)}>
423
- {model.ModalAddComponent && (
424
- <model.ModalAddComponent
425
- formContent={selectedRecord}
426
- match={match}
427
- model={model}
428
- additional_queries={additional_queries}
429
- callback={() => {
430
- setDrawerVisible(false);
431
- loadMenus();
432
- }}
433
- />
434
- )}
435
- </Drawer>
436
- </Card>
437
- );
438
- };
439
- /* -----------------------------------------------------------------------
440
- PANEL ACTIONS
441
- ------------------------------------------------------------------------ */
442
- function panelActions(item, model, setSelectedRecord, setDrawerTitle, setDrawerVisible, deleteRecord) {
443
- return (
444
- <Space onClick={(e) => e.stopPropagation()}>
445
- <Tag color={item.is_visible === true ? 'green' : 'blue'}>{item.is_visible === true ? 'VISIBLE' : 'HIDDEN'}</Tag>{' '}
446
- <Button
447
- size="small"
448
- type="dashed"
449
- onClick={() => {
450
- setSelectedRecord({
451
- // parent menu id
452
- header_id: item.id,
453
- step: item.step + 1,
454
- // new record
455
- id: null,
456
- });
457
- setDrawerTitle(`Add Submenu to "${item.name}"`);
458
- setDrawerVisible(true);
459
- }}
460
- >
461
- <PlusCircleFilled />
462
- Add Sub Menu
463
- </Button>
464
- {model.ModalAddComponent && (
465
- <Button
466
- size="small"
467
- type="default"
468
- onClick={() => {
469
- setSelectedRecord(item);
470
- setDrawerTitle('Edit Menu');
471
- setDrawerVisible(true);
472
- }}
473
- >
474
- <EditOutlined />
475
- </Button>
476
- )}
477
- <Button
478
- size="small"
479
- type="default"
480
- onClick={() => {
481
- setSelectedRecord({ ...item, id: null, copy: true });
482
- setDrawerTitle('Copy Menu');
483
- setDrawerVisible(true);
484
- }}
485
- >
486
- <CopyOutlined />
487
- </Button>
488
- <Popconfirm title="Are you sure?" onConfirm={() => deleteRecord(item)}>
489
- <Button danger size="small" type="default">
490
- <DeleteOutlined />
491
- </Button>
492
- </Popconfirm>
493
- </Space>
494
- );
495
- }
496
-
497
- function NestedMenu({
498
- parentId,
499
- step,
500
- items,
501
- model,
502
- dragMode,
503
- setSelectedRecord,
504
- setDrawerTitle,
505
- setDrawerVisible,
506
- deleteRecord,
507
- level,
508
- onCrossLevelMove,
509
- onChange,
510
- }) {
511
- // do not render Collapse
512
- if (!items || items.length === 0) return null;
513
-
514
- const [localItems, setLocalItems] = useState(items);
515
-
516
- useEffect(() => {
517
- setLocalItems(items);
518
- }, [items]);
519
-
520
- const moveSubMenu = useCallback(
521
- (from, to) => {
522
- if (!dragMode || from === to) return;
523
-
524
- const updated = [...localItems];
525
- const [moved] = updated.splice(from, 1);
526
- updated.splice(to, 0, moved);
527
-
528
- setLocalItems(updated);
529
- onChange?.(updated);
530
- },
531
- [dragMode, localItems, onChange]
532
- );
533
- if (!localItems || localItems.length === 0) {
534
- // <-- don’t render anything if no submenus
535
- return null;
536
- }
537
-
538
- return (
539
- <Collapse accordion>
540
- {localItems.map((child, index) => (
541
- <Panel
542
- key={child.id}
543
- header={
544
- <DraggableWrapper
545
- id={child.id}
546
- index={index}
547
- movePanel={moveSubMenu}
548
- item={child}
549
- dragEnabled={dragMode}
550
- level={level}
551
- parentId={parentId}
552
- onCrossLevelMove={onCrossLevelMove}
553
- canAcceptChildren={true}
554
- />
555
- }
556
- // only show arrow if sub_menus exist
557
- showArrow={child.sub_menus && child.sub_menus.length > 0}
558
- extra={panelActions(child, model, setSelectedRecord, setDrawerTitle, setDrawerVisible, deleteRecord)}
559
- >
560
- <NestedMenu
561
- model={model}
562
- parentId={child.id}
563
- step={step + 1}
564
- items={child.sub_menus || []}
565
- dragMode={dragMode}
566
- setSelectedRecord={setSelectedRecord}
567
- setDrawerTitle={setDrawerTitle}
568
- setDrawerVisible={setDrawerVisible}
569
- deleteRecord={deleteRecord}
570
- level={level + 1}
571
- onCrossLevelMove={onCrossLevelMove}
572
- onChange={(submenus) => {
573
- const updated = localItems.map((item) => (item.id === child.id ? { ...item, sub_menus: submenus } : item));
574
- setLocalItems(updated);
575
- onChange?.(updated);
576
- }}
577
- />
578
- </Panel>
579
- ))}
580
- </Collapse>
581
- );
582
- }
583
-
584
- export default MenuLists;
1
+ import React, { useState, useCallback, useEffect } from 'react';
2
+ import { Space, Popconfirm, Input, Drawer, Skeleton, Collapse, message, Tag, Empty } from 'antd';
3
+ import { ReloadOutlined, DeleteOutlined, EditOutlined, PlusCircleFilled, CopyOutlined } from '@ant-design/icons';
4
+ import { Button, Card, Switch, DraggableWrapper } from '../../../../lib';
5
+ // for draggable menu list import { DndProvider } from "react-dnd";
6
+ import { DndProvider } from 'react-dnd';
7
+ import { Link, useParams, useLocation } from 'react-router-dom';
8
+ import { HTML5Backend } from 'react-dnd-html5-backend';
9
+
10
+ import './menu-lists.scss';
11
+
12
+ const { Search } = Input;
13
+ const { Panel } = Collapse;
14
+
15
+ const MenuLists = ({ model, match, relativeAdd = false, additional_queries = [], disableAddModal = model.disableAddModal }) => {
16
+ const actions = [];
17
+
18
+ match = useParams();
19
+ const id = match.id;
20
+ // const location = useLocation();
21
+ const step = parseInt(new URLSearchParams(location.search).get('step')) || 1;
22
+
23
+ const [records, setRecords] = useState([]);
24
+ const [originalRecords, setOriginalRecords] = useState([]);
25
+ const [loading, setLoading] = useState(false);
26
+ const [drawerVisible, setDrawerVisible] = useState(false);
27
+ const [single, setSingle] = useState({});
28
+ const [drawerTitle, setDrawerTitle] = useState('');
29
+ const [selectedRecord, setSelectedRecord] = useState(null);
30
+ const [query, setQuery] = useState('');
31
+ const [dragMode, setDragMode] = useState(false);
32
+ const [orderChanged, setOrderChanged] = useState(false);
33
+ // keys of manually-expanded top-level panels (used when NOT searching)
34
+ const [manualOpenKeys, setManualOpenKeys] = useState([]);
35
+
36
+ const [nextId, setNextId] = useState(10000);
37
+
38
+ useEffect(() => {
39
+ loadMenus();
40
+ }, [id]);
41
+
42
+ const loadMenus = () => {
43
+ setLoading(true);
44
+
45
+ model
46
+ .getCoreMenuLists
47
+ // {
48
+ // queries: [...additional_queries, { field: 'step', value: step }, { field: 'header_id', value: null }],
49
+ // }
50
+ ()
51
+ .then((res) => {
52
+ const sorted = (res.result || []).sort((a, b) => a.order - b.order);
53
+ setRecords(sorted);
54
+ setOriginalRecords(sorted);
55
+ setLoading(false);
56
+ })
57
+ .catch(() => setLoading(false));
58
+ };
59
+
60
+ /**
61
+ *
62
+ */
63
+ const getRecords = () => {
64
+ let firstStepQueries = [
65
+ {
66
+ field: 'step',
67
+ value: step,
68
+ },
69
+ ];
70
+
71
+ let config = {
72
+ queries: [...additional_queries, ...firstStepQueries],
73
+ };
74
+
75
+ if (!additional_queries.length) {
76
+ config.queries.push({
77
+ field: 'header_id',
78
+ value: null,
79
+ });
80
+ }
81
+
82
+ setLoading(true);
83
+
84
+ // Get the records
85
+ return model
86
+ .get(config)
87
+
88
+ .then((res) => {
89
+ // setRecords(res.result);
90
+ var result = res.result;
91
+
92
+ if (Array.isArray(result)) {
93
+ setRecords(result.sort((a, b) => a.order - b.order));
94
+ // } else {
95
+ // setRecords([])
96
+ }
97
+
98
+ setLoading(false);
99
+ })
100
+ .catch((error) => {
101
+ setLoading(false);
102
+
103
+ console.log(error);
104
+ });
105
+ };
106
+
107
+ const toggleDragMode = (checked) => {
108
+ setDragMode(checked);
109
+ if (checked) {
110
+ message.info('Drag mode enabled - Drag items anywhere, including the "Drop here" zones!');
111
+ }
112
+ };
113
+
114
+ // Function to find and remove an item from anywhere in the tree
115
+ const findAndRemoveItem = (items, itemId) => {
116
+ let foundItem = null;
117
+
118
+ const removeRecursive = (arr) => {
119
+ for (let i = 0; i < arr.length; i++) {
120
+ if (arr[i].id === itemId) {
121
+ foundItem = { ...arr[i] };
122
+ arr.splice(i, 1);
123
+ return true;
124
+ }
125
+ if (arr[i].sub_menus && arr[i].sub_menus.length > 0) {
126
+ if (removeRecursive(arr[i].sub_menus)) {
127
+ return true;
128
+ }
129
+ }
130
+ }
131
+ return false;
132
+ };
133
+
134
+ const newItems = JSON.parse(JSON.stringify(items));
135
+ removeRecursive(newItems);
136
+ return { newItems, foundItem };
137
+ };
138
+
139
+ // Function to insert item at specific location
140
+ const insertItemAt = (items, item, targetLevel, targetParentId, targetIndex) => {
141
+ const newItems = JSON.parse(JSON.stringify(items));
142
+
143
+ // Update item properties for new location
144
+ const updatedItem = {
145
+ ...item,
146
+ step: targetLevel,
147
+ header_id: targetParentId,
148
+ };
149
+
150
+ if (targetLevel === 1) {
151
+ // Insert at main level
152
+ newItems.splice(targetIndex, 0, updatedItem);
153
+ } else {
154
+ // Insert into nested level
155
+ const insertRecursive = (arr) => {
156
+ for (let i = 0; i < arr.length; i++) {
157
+ if (arr[i].id === targetParentId) {
158
+ if (!arr[i].sub_menus) arr[i].sub_menus = [];
159
+ arr[i].sub_menus.splice(targetIndex, 0, updatedItem);
160
+ return true;
161
+ }
162
+ if (arr[i].sub_menus && arr[i].sub_menus.length > 0) {
163
+ if (insertRecursive(arr[i].sub_menus)) {
164
+ return true;
165
+ }
166
+ }
167
+ }
168
+ return false;
169
+ };
170
+ insertRecursive(newItems);
171
+ }
172
+
173
+ return newItems;
174
+ };
175
+
176
+ const handleCrossLevelMove = useCallback(
177
+ (draggedItem, dropTarget) => {
178
+ const { targetLevel, targetParentId, targetIndex } = dropTarget;
179
+
180
+ // Prevent moving item into itself or its children
181
+ if (draggedItem.id === targetParentId) {
182
+ message.warning('Cannot move item into itself');
183
+ return;
184
+ }
185
+
186
+ // Remove item from original location
187
+ const { newItems: itemsAfterRemove, foundItem } = findAndRemoveItem(records, draggedItem.id);
188
+
189
+ if (!foundItem) {
190
+ message.error('Could not find item to move');
191
+ return;
192
+ }
193
+
194
+ // Insert item at new location
195
+ const finalItems = insertItemAt(itemsAfterRemove, foundItem, targetLevel, targetParentId, targetIndex);
196
+
197
+ setRecords(finalItems);
198
+ setOrderChanged(true);
199
+ message.success(`Moved "${foundItem.name}" to level ${targetLevel}`);
200
+ },
201
+ [records]
202
+ );
203
+
204
+ const deleteRecord = (rec) => {
205
+ model.delete(rec).then(loadMenus);
206
+ };
207
+
208
+ // Recursively filter the menu tree by caption. A menu is kept if its own
209
+ // caption matches OR any descendant matches. When the menu itself matches we
210
+ // keep its full subtree; when only a descendant matches we keep just the
211
+ // pruned path so the matching item stays reachable.
212
+ const filterMenus = (menus, q) => {
213
+ if (!q) return menus || [];
214
+ const lower = q.toLowerCase();
215
+
216
+ return (menus || []).reduce((acc, menu) => {
217
+ const selfMatch = (menu.caption || '').toLowerCase().includes(lower);
218
+ if (selfMatch) {
219
+ acc.push({ ...menu });
220
+ return acc;
221
+ }
222
+ const subFiltered = filterMenus(menu.sub_menus || [], q);
223
+ if (subFiltered.length) {
224
+ acc.push({ ...menu, sub_menus: subFiltered });
225
+ }
226
+ return acc;
227
+ }, []);
228
+ };
229
+
230
+ // Ids of items (at one level) that have sub_menus — used to auto-expand
231
+ // every branch leading to a search match.
232
+ const expandableKeys = (items) => (items || []).filter((m) => m.sub_menus && m.sub_menus.length > 0).map((m) => String(m.id));
233
+
234
+ const searchActive = !!query;
235
+ const visibleItems = filterMenus(records, query);
236
+
237
+ // While searching, derive the open panels synchronously from the visible tree
238
+ // (every branch on the path to a match), so the matched sub-menu is revealed.
239
+ // When not searching, use the user's manually-expanded panels.
240
+ const activeKeys = searchActive ? expandableKeys(visibleItems) : manualOpenKeys;
241
+
242
+ const onSearch = (event) => {
243
+ setQuery(event.target.value);
244
+ };
245
+
246
+ // ------------------------
247
+ // Recursive function to build payload
248
+ // ------------------------
249
+ const buildUpdatedOrderPayload = (menus, originalMenus) => {
250
+ const result = [];
251
+
252
+ menus.forEach((menu, index) => {
253
+ const original = findMenuById(originalMenus, menu.id);
254
+
255
+ const newOrder = index + 1;
256
+ const oldOrder = original ? original.order : null;
257
+
258
+ const orderChanged = oldOrder !== newOrder;
259
+ const parentChanged = original?.header_id !== menu.header_id;
260
+
261
+ // check sub menus
262
+ let updatedSubMenus = [];
263
+ if (menu.sub_menus?.length) {
264
+ updatedSubMenus = buildUpdatedOrderPayload(menu.sub_menus, originalMenus);
265
+ }
266
+
267
+ // include only if changed OR has changed children
268
+ if (orderChanged || parentChanged || updatedSubMenus.length) {
269
+ result.push({
270
+ id: menu.id,
271
+ order: newOrder,
272
+ header_id: menu.header_id,
273
+ name: menu.name,
274
+ ...(updatedSubMenus.length && { sub_menus: updatedSubMenus }),
275
+ });
276
+ }
277
+ });
278
+
279
+ return result;
280
+ };
281
+
282
+ // ------------------------
283
+ // Find menu by id (recursive)
284
+ // ------------------------
285
+ const findMenuById = (menus, id) => {
286
+ for (const menu of menus || []) {
287
+ if (menu.id === id) return menu;
288
+
289
+ if (menu.sub_menus?.length) {
290
+ const found = findMenuById(menu.sub_menus, id);
291
+ if (found) return found;
292
+ }
293
+ }
294
+ return null;
295
+ };
296
+
297
+ // ------------------------
298
+ // Save order function
299
+ // ------------------------
300
+ const saveOrder = () => {
301
+ const updatedMenus = buildUpdatedOrderPayload(records, originalRecords);
302
+
303
+ if (!updatedMenus.length) {
304
+ message.info('No order changes to save');
305
+ return;
306
+ }
307
+
308
+ const payload = { menus: updatedMenus };
309
+
310
+ model
311
+ .saveOrder({ formBody: payload })
312
+ .then(() => {
313
+ setOrderChanged(false);
314
+ loadMenus(); // refresh from backend
315
+ })
316
+ .catch(console.error);
317
+ };
318
+
319
+ const movePanel = useCallback(
320
+ (from, to) => {
321
+ if (!dragMode) return;
322
+ if (from === to) return;
323
+
324
+ setRecords((prevRecords) => {
325
+ const updated = [...prevRecords];
326
+ const [moved] = updated.splice(from, 1);
327
+ updated.splice(to, 0, moved);
328
+ return updated;
329
+ });
330
+
331
+ setOrderChanged(true);
332
+ },
333
+ [dragMode]
334
+ );
335
+
336
+ return (
337
+ <Card className="generic-list">
338
+ <div style={{ marginBottom: 16 }}>
339
+ <div style={{ display: 'flex', justifyContent: 'space-between', alignItems: 'center', marginBottom: 16 }}>
340
+ <Search placeholder="Search menus & sub-menus…" allowClear style={{ width: 300, marginBottom: '0px' }} onChange={onSearch} />
341
+
342
+ <Space size="small">
343
+ <Button onClick={getRecords} size={'small'} type="default">
344
+ <ReloadOutlined />
345
+ </Button>
346
+
347
+ <Switch checked={dragMode} onChange={toggleDragMode} checkedChildren="Order On" unCheckedChildren="Order Off" />
348
+
349
+ {dragMode && orderChanged && (
350
+ <Button onClick={saveOrder} type="primary" size="small">
351
+ Save Order
352
+ </Button>
353
+ )}
354
+
355
+ <Button
356
+ type="primary"
357
+ size="small"
358
+ onClick={() => {
359
+ setSelectedRecord({ step: 1, header_id: null });
360
+ setDrawerTitle('Create New Menu');
361
+ setDrawerVisible(true);
362
+ }}
363
+ >
364
+ Create New Menu
365
+ </Button>
366
+ </Space>
367
+ </div>
368
+
369
+ {dragMode && (
370
+ <div
371
+ style={{
372
+ padding: '8px 12px',
373
+ backgroundColor: '#e6f7ff',
374
+ borderRadius: 4,
375
+ border: '1px solid #91d5ff',
376
+ fontSize: 13,
377
+ }}
378
+ >
379
+ <strong>💡 Tips:</strong>
380
+ <ul style={{ margin: '4px 0', paddingLeft: 20 }}>
381
+ <li>Drag between items to reorder at the same level</li>
382
+ <li>Drop on the green "↳ Drop here" zone to make it a child of that item</li>
383
+ <li>Works at all levels - unlimited nesting supported!</li>
384
+ </ul>
385
+ </div>
386
+ )}
387
+ </div>
388
+ {loading ? (
389
+ <Skeleton active />
390
+ ) : (
391
+ <>
392
+ <>
393
+ {/* {!view ? ( */}
394
+ <Card>
395
+ <DndProvider backend={HTML5Backend}>
396
+ <Collapse
397
+ accordion={!searchActive}
398
+ activeKey={activeKeys}
399
+ onChange={(keys) => setManualOpenKeys(Array.isArray(keys) ? keys : keys ? [keys] : [])}
400
+ >
401
+ {visibleItems && visibleItems.length > 0 ? (
402
+ visibleItems.map((item, index) => (
403
+ <Panel
404
+ key={item.id}
405
+ header={
406
+ <DraggableWrapper
407
+ id={item.id}
408
+ index={index}
409
+ movePanel={movePanel}
410
+ item={item}
411
+ dragEnabled={dragMode}
412
+ level={1}
413
+ parentId={null}
414
+ onCrossLevelMove={handleCrossLevelMove}
415
+ canAcceptChildren={true}
416
+ />
417
+ }
418
+ showArrow={item.sub_menus && item.sub_menus.length > 0}
419
+ extra={panelActions(item, model, setSelectedRecord, setDrawerTitle, setDrawerVisible, deleteRecord)}
420
+ >
421
+ {item.sub_menus && item.sub_menus.length > 0 && (
422
+ <NestedMenu
423
+ parentId={item.id}
424
+ step={item.step + 1}
425
+ items={item.sub_menus || []}
426
+ model={model}
427
+ dragMode={dragMode}
428
+ searchActive={searchActive}
429
+ query={query}
430
+ setSelectedRecord={setSelectedRecord}
431
+ setDrawerTitle={setDrawerTitle}
432
+ setDrawerVisible={setDrawerVisible}
433
+ deleteRecord={deleteRecord}
434
+ level={2}
435
+ onCrossLevelMove={handleCrossLevelMove}
436
+ onChange={(subMenus) => {
437
+ const updated = records.map((r) => (r.id === item.id ? { ...r, sub_menus: subMenus } : r));
438
+ setRecords(updated);
439
+ setOrderChanged(true);
440
+ }}
441
+ />
442
+ )}
443
+ </Panel>
444
+ ))
445
+ ) : (
446
+ <div style={{ textAlign: 'center', padding: '40px 0' }}>
447
+ <Empty description="No Menu Items" />
448
+ </div>
449
+ )}
450
+ </Collapse>
451
+ </DndProvider>
452
+ </Card>
453
+ {/* ) : (
454
+ <CardList model={model} data={filtered ? filtered : records} />
455
+ )} */}
456
+ </>
457
+ </>
458
+ )}
459
+
460
+ <Drawer title={drawerTitle} open={drawerVisible} width="80%" destroyOnClose onClose={() => setDrawerVisible(false)}>
461
+ {model.ModalAddComponent && (
462
+ <model.ModalAddComponent
463
+ formContent={selectedRecord}
464
+ match={match}
465
+ model={model}
466
+ additional_queries={additional_queries}
467
+ callback={() => {
468
+ setDrawerVisible(false);
469
+ loadMenus();
470
+ }}
471
+ />
472
+ )}
473
+ </Drawer>
474
+ </Card>
475
+ );
476
+ };
477
+ /* -----------------------------------------------------------------------
478
+ PANEL ACTIONS
479
+ ------------------------------------------------------------------------ */
480
+ function panelActions(item, model, setSelectedRecord, setDrawerTitle, setDrawerVisible, deleteRecord) {
481
+ return (
482
+ <Space onClick={(e) => e.stopPropagation()}>
483
+ <Tag color={item.is_visible === true ? 'green' : 'blue'}>{item.is_visible === true ? 'VISIBLE' : 'HIDDEN'}</Tag>{' '}
484
+ <Button
485
+ size="small"
486
+ type="dashed"
487
+ onClick={() => {
488
+ setSelectedRecord({
489
+ // parent menu id
490
+ header_id: item.id,
491
+ step: item.step + 1,
492
+ // new record
493
+ id: null,
494
+ });
495
+ setDrawerTitle(`Add Submenu to "${item.name}"`);
496
+ setDrawerVisible(true);
497
+ }}
498
+ >
499
+ <PlusCircleFilled />
500
+ Add Sub Menu
501
+ </Button>
502
+ {model.ModalAddComponent && (
503
+ <Button
504
+ size="small"
505
+ type="default"
506
+ onClick={() => {
507
+ setSelectedRecord(item);
508
+ setDrawerTitle('Edit Menu');
509
+ setDrawerVisible(true);
510
+ }}
511
+ >
512
+ <EditOutlined />
513
+ </Button>
514
+ )}
515
+ <Button
516
+ size="small"
517
+ type="default"
518
+ onClick={() => {
519
+ setSelectedRecord({ ...item, id: null, copy: true });
520
+ setDrawerTitle('Copy Menu');
521
+ setDrawerVisible(true);
522
+ }}
523
+ >
524
+ <CopyOutlined />
525
+ </Button>
526
+ <Popconfirm title="Are you sure?" onConfirm={() => deleteRecord(item)}>
527
+ <Button danger size="small" type="default">
528
+ <DeleteOutlined />
529
+ </Button>
530
+ </Popconfirm>
531
+ </Space>
532
+ );
533
+ }
534
+
535
+ function NestedMenu({
536
+ parentId,
537
+ step,
538
+ items,
539
+ model,
540
+ dragMode,
541
+ searchActive,
542
+ query,
543
+ setSelectedRecord,
544
+ setDrawerTitle,
545
+ setDrawerVisible,
546
+ deleteRecord,
547
+ level,
548
+ onCrossLevelMove,
549
+ onChange,
550
+ }) {
551
+ const [localItems, setLocalItems] = useState(items);
552
+ // keys of manually-expanded sub-panels (used when NOT searching)
553
+ const [manualOpenKeys, setManualOpenKeys] = useState([]);
554
+
555
+ useEffect(() => {
556
+ setLocalItems(items);
557
+ }, [items]);
558
+
559
+ const moveSubMenu = useCallback(
560
+ (from, to) => {
561
+ if (!dragMode || from === to) return;
562
+
563
+ const updated = [...localItems];
564
+ const [moved] = updated.splice(from, 1);
565
+ updated.splice(to, 0, moved);
566
+
567
+ setLocalItems(updated);
568
+ onChange?.(updated);
569
+ },
570
+ [dragMode, localItems, onChange]
571
+ );
572
+ if (!localItems || localItems.length === 0) {
573
+ // <-- don’t render anything if no submenus
574
+ return null;
575
+ }
576
+
577
+ // While searching, derive the open sub-panels from the items so the path to a
578
+ // match stays expanded; otherwise use the user's manually-expanded panels.
579
+ const activeKeys = searchActive
580
+ ? localItems.filter((c) => c.sub_menus && c.sub_menus.length > 0).map((c) => String(c.id))
581
+ : manualOpenKeys;
582
+
583
+ return (
584
+ <Collapse
585
+ accordion={!searchActive}
586
+ activeKey={activeKeys}
587
+ onChange={(keys) => setManualOpenKeys(Array.isArray(keys) ? keys : keys ? [keys] : [])}
588
+ >
589
+ {localItems.map((child, index) => (
590
+ <Panel
591
+ key={child.id}
592
+ header={
593
+ <DraggableWrapper
594
+ id={child.id}
595
+ index={index}
596
+ movePanel={moveSubMenu}
597
+ item={child}
598
+ dragEnabled={dragMode}
599
+ level={level}
600
+ parentId={parentId}
601
+ onCrossLevelMove={onCrossLevelMove}
602
+ canAcceptChildren={true}
603
+ />
604
+ }
605
+ // only show arrow if sub_menus exist
606
+ showArrow={child.sub_menus && child.sub_menus.length > 0}
607
+ extra={panelActions(child, model, setSelectedRecord, setDrawerTitle, setDrawerVisible, deleteRecord)}
608
+ >
609
+ <NestedMenu
610
+ model={model}
611
+ parentId={child.id}
612
+ step={step + 1}
613
+ items={child.sub_menus || []}
614
+ dragMode={dragMode}
615
+ searchActive={searchActive}
616
+ query={query}
617
+ setSelectedRecord={setSelectedRecord}
618
+ setDrawerTitle={setDrawerTitle}
619
+ setDrawerVisible={setDrawerVisible}
620
+ deleteRecord={deleteRecord}
621
+ level={level + 1}
622
+ onCrossLevelMove={onCrossLevelMove}
623
+ onChange={(submenus) => {
624
+ const updated = localItems.map((item) => (item.id === child.id ? { ...item, sub_menus: submenus } : item));
625
+ setLocalItems(updated);
626
+ onChange?.(updated);
627
+ }}
628
+ />
629
+ </Panel>
630
+ ))}
631
+ </Collapse>
632
+ );
633
+ }
634
+
635
+ export default MenuLists;