@campxdev/shared 1.9.1 → 1.9.2

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 (240) hide show
  1. package/antd.customize.less +73 -73
  2. package/exports.ts +19 -19
  3. package/package.json +86 -86
  4. package/src/assets/fonts/avenir/index.ts +13 -13
  5. package/src/assets/fonts/poppins/index.ts +18 -18
  6. package/src/assets/images/index.ts +17 -17
  7. package/src/assets/images/unauth.svg +92 -92
  8. package/src/components/ActionButton.tsx +20 -20
  9. package/src/components/ApplicationProfile/ApplicationProfile.tsx +378 -344
  10. package/src/components/ApplicationProfile/DepartmentFilter.tsx +77 -0
  11. package/src/components/ApplicationProfile/UserProfileRelation.tsx +130 -130
  12. package/src/components/ApplicationProfile/index.tsx +1 -1
  13. package/src/components/ApplicationProfile/services.ts +71 -68
  14. package/src/components/Attachment.tsx +26 -26
  15. package/src/components/AutocompleteSearch/AutocompleteSearch.tsx +57 -57
  16. package/src/components/AutocompleteSearch/index.tsx +1 -1
  17. package/src/components/Breadcrumbs.tsx +72 -72
  18. package/src/components/Card.tsx +98 -98
  19. package/src/components/CardsGrid.tsx +28 -28
  20. package/src/components/ChangePassword.tsx +123 -123
  21. package/src/components/Chips.tsx +81 -81
  22. package/src/components/Detail.tsx +15 -15
  23. package/src/components/DetailsGrid.tsx +52 -52
  24. package/src/components/DividerHeading.tsx +41 -41
  25. package/src/components/DrawerWrapper/DialogWrapper.tsx +67 -67
  26. package/src/components/DrawerWrapper/DrawerTemplate.tsx +53 -53
  27. package/src/components/DrawerWrapper/DrawerWrapper.tsx +53 -53
  28. package/src/components/DropDownButton/AnchorElement.tsx +44 -44
  29. package/src/components/DropDownButton/DropDownButton.tsx +184 -184
  30. package/src/components/DropDownButton/DropdownMenuItem.tsx +122 -122
  31. package/src/components/DropDownButton/index.tsx +1 -1
  32. package/src/components/DropDownButton/styles.tsx +59 -59
  33. package/src/components/ErrorBoundary/ErrorBoundary.tsx +33 -33
  34. package/src/components/ErrorBoundary/ErrorFallback.tsx +240 -240
  35. package/src/components/ErrorBoundary/GlobalNetworkLoadingIndicator.tsx +13 -13
  36. package/src/components/ErrorBoundary/index.tsx +1 -1
  37. package/src/components/ErrorBox.tsx +42 -42
  38. package/src/components/ExcelToJsonInput/ExcelJsonUpload.tsx +50 -50
  39. package/src/components/ExcelToJsonInput/ExcelToJsonInput.tsx +57 -57
  40. package/src/components/ExcelToJsonInput/index.tsx +1 -1
  41. package/src/components/FilterComponents/FilterButton.tsx +56 -56
  42. package/src/components/FilterComponents/SearchBar.tsx +45 -45
  43. package/src/components/FloatingContainer.tsx +33 -33
  44. package/src/components/Form/Form.tsx +147 -147
  45. package/src/components/Form/RenderForm.tsx +189 -189
  46. package/src/components/FullScreenLoader.tsx +19 -19
  47. package/src/components/HookForm/AutoCompleteSearch.tsx +113 -113
  48. package/src/components/HookForm/DatePicker.tsx +40 -40
  49. package/src/components/HookForm/DateTimePicker.tsx +47 -47
  50. package/src/components/HookForm/FormLabel.tsx +14 -14
  51. package/src/components/HookForm/MultiCheckbox.tsx +69 -69
  52. package/src/components/HookForm/MultiSelect.tsx +61 -61
  53. package/src/components/HookForm/RadioGroup.tsx +47 -47
  54. package/src/components/HookForm/SingleCheckbox.tsx +34 -34
  55. package/src/components/HookForm/SingleSelect.tsx +46 -46
  56. package/src/components/HookForm/TextField.tsx +40 -40
  57. package/src/components/HookForm/TimePicker.tsx +40 -40
  58. package/src/components/HookForm/index.ts +23 -23
  59. package/src/components/IconButtons/IconButtons.tsx +137 -137
  60. package/src/components/IconButtons/Icons.tsx +268 -268
  61. package/src/components/IconButtons/assets/edit.svg +4 -4
  62. package/src/components/IconButtons/assets/eye.svg +6 -6
  63. package/src/components/IconButtons/assets/trash.svg +7 -7
  64. package/src/components/IconButtons/index.tsx +8 -8
  65. package/src/components/IconLabel.tsx +37 -37
  66. package/src/components/Image/Image.tsx +45 -45
  67. package/src/components/Image/index.tsx +1 -1
  68. package/src/components/ImageUpload.tsx +98 -98
  69. package/src/components/Input/AsyncSearchSelect/AsyncSearchSelect.tsx +216 -216
  70. package/src/components/Input/AsyncSearchSelect/index.tsx +1 -1
  71. package/src/components/Input/AsyncSearchSelect/styles.tsx +105 -105
  72. package/src/components/Input/AutoCompleteSearch.tsx +121 -121
  73. package/src/components/Input/DatePicker.tsx +87 -87
  74. package/src/components/Input/DateRangePicker.tsx +105 -105
  75. package/src/components/Input/DateTimePicker.tsx +84 -84
  76. package/src/components/Input/FormLabel.tsx +21 -21
  77. package/src/components/Input/MultiCheckbox.tsx +64 -64
  78. package/src/components/Input/MultiSelect.tsx +161 -161
  79. package/src/components/Input/RadioGroup.tsx +104 -104
  80. package/src/components/Input/SingleCheckbox.tsx +78 -78
  81. package/src/components/Input/SingleSelect.tsx +104 -104
  82. package/src/components/Input/TextField.tsx +46 -46
  83. package/src/components/Input/TimePicker.tsx +77 -77
  84. package/src/components/Input/index.ts +26 -26
  85. package/src/components/Input/types.ts +3 -3
  86. package/src/components/Institutions/InsititutionsDialog.tsx +86 -86
  87. package/src/components/Institutions/InsititutionsNotAssignedDialog .tsx +12 -12
  88. package/src/components/Institutions/InstitutionsDropdown.tsx +33 -33
  89. package/src/components/Institutions/index.tsx +1 -1
  90. package/src/components/Institutions/services.ts +12 -12
  91. package/src/components/JsonPreview.tsx +7 -7
  92. package/src/components/LabelValue.tsx +21 -21
  93. package/src/components/Layout/ChangePassword.tsx +49 -49
  94. package/src/components/Layout/Header/AppHeader.tsx +109 -109
  95. package/src/components/Layout/Header/AppsMenu.tsx +132 -132
  96. package/src/components/Layout/Header/HeaderActions/CogWheelMenu.tsx +33 -33
  97. package/src/components/Layout/Header/HeaderActions/FreshChatButton.tsx +61 -61
  98. package/src/components/Layout/Header/HeaderActions/FreshDeskHelpButton.tsx +53 -53
  99. package/src/components/Layout/Header/HeaderActions/HeaderActions.tsx +20 -20
  100. package/src/components/Layout/Header/HeaderActions/UserBox.tsx +56 -56
  101. package/src/components/Layout/Header/Notification.tsx +13 -13
  102. package/src/components/Layout/Header/applications.ts +102 -102
  103. package/src/components/Layout/Header/assets/campx_square_small.svg +8 -8
  104. package/src/components/Layout/Header/assets/commuteX.png +0 -0
  105. package/src/components/Layout/Header/assets/commutex.svg +14 -14
  106. package/src/components/Layout/Header/assets/commutexSmall.svg +11 -11
  107. package/src/components/Layout/Header/assets/enroll.svg +14 -14
  108. package/src/components/Layout/Header/assets/enrollx.svg +14 -14
  109. package/src/components/Layout/Header/assets/exams_small.svg +11 -11
  110. package/src/components/Layout/Header/assets/examsx.svg +14 -14
  111. package/src/components/Layout/Header/assets/hostel_small.svg +13 -13
  112. package/src/components/Layout/Header/assets/hostelx.svg +13 -13
  113. package/src/components/Layout/Header/assets/index.ts +37 -37
  114. package/src/components/Layout/Header/assets/libraryx.svg +12 -12
  115. package/src/components/Layout/Header/assets/pay_small.svg +15 -15
  116. package/src/components/Layout/Header/assets/payx.svg +19 -19
  117. package/src/components/Layout/Header/assets/people_small.svg +9 -9
  118. package/src/components/Layout/Header/assets/peoplex.svg +12 -12
  119. package/src/components/Layout/Header/assets/squarex.svg +12 -12
  120. package/src/components/Layout/Header/icons.tsx +57 -57
  121. package/src/components/Layout/Header/index.tsx +1 -1
  122. package/src/components/Layout/Header/styles.tsx +133 -133
  123. package/src/components/Layout/Helmet.tsx +108 -108
  124. package/src/components/Layout/LayoutWrapper.tsx +28 -28
  125. package/src/components/Layout/SideMenuHeader.tsx +29 -29
  126. package/src/components/Layout/SideNav.tsx +168 -168
  127. package/src/components/Layout/Tickets/HelpWidget/HelpWidget.tsx +273 -273
  128. package/src/components/Layout/Tickets/HelpWidget/styles.tsx +94 -94
  129. package/src/components/Layout/Tickets/MyTickets.tsx +72 -72
  130. package/src/components/Layout/Tickets/TicketDetails.tsx +65 -65
  131. package/src/components/Layout/Tickets/TimeLine.tsx +64 -64
  132. package/src/components/Layout/Tickets/index.tsx +1 -1
  133. package/src/components/Layout/Tickets/services.ts +11 -11
  134. package/src/components/Layout/Tickets/styles.tsx +136 -136
  135. package/src/components/LinearProgress.tsx +19 -19
  136. package/src/components/ListItemButton.tsx +95 -95
  137. package/src/components/LoginForm.tsx +274 -274
  138. package/src/components/MediaRow/MediaRow.tsx +67 -67
  139. package/src/components/MediaRow/index.tsx +1 -1
  140. package/src/components/ModalButtons/DialogButton.tsx +130 -130
  141. package/src/components/ModalButtons/DrawerButton.tsx +115 -115
  142. package/src/components/ModalButtons/PopoverButton.tsx +99 -99
  143. package/src/components/ModalButtons/index.tsx +4 -4
  144. package/src/components/NoDataIllustration.tsx +35 -35
  145. package/src/components/PageContent.tsx +12 -12
  146. package/src/components/PageHeader.tsx +55 -55
  147. package/src/components/PageNotFound.tsx +26 -26
  148. package/src/components/PopupConfirm/ConfirmContextProvider.tsx +40 -40
  149. package/src/components/PopupConfirm/PopupConfirm.tsx +67 -67
  150. package/src/components/PopupConfirm/index.tsx +1 -1
  151. package/src/components/PopupConfirm/useConfirm.ts +47 -47
  152. package/src/components/ResetPassword.tsx +107 -107
  153. package/src/components/Row.tsx +24 -24
  154. package/src/components/Spinner.tsx +18 -18
  155. package/src/components/StepsHeader/StepsHeader.tsx +115 -115
  156. package/src/components/StepsHeader/index.tsx +1 -1
  157. package/src/components/StyledTableContainer.tsx +33 -33
  158. package/src/components/SwitchButton.tsx +41 -41
  159. package/src/components/Table.tsx +42 -42
  160. package/src/components/Tables/BasicTable/Table.tsx +198 -198
  161. package/src/components/Tables/BasicTable/TableFooter.tsx +86 -86
  162. package/src/components/Tables/BasicTable/index.tsx +1 -1
  163. package/src/components/Tables/ReactTable/BatchActionsHeader.tsx +58 -58
  164. package/src/components/Tables/ReactTable/ReactTable.tsx +295 -295
  165. package/src/components/Tables/ReactTable/RenderTableBody.tsx +49 -49
  166. package/src/components/Tables/ReactTable/index.tsx +1 -1
  167. package/src/components/Tables/ReactTable/react-table-config.d.ts +128 -128
  168. package/src/components/Tables/common/NoRecordsFound.tsx +31 -31
  169. package/src/components/Tables/common/TableStats.tsx +22 -22
  170. package/src/components/Tables/common/icons.tsx +50 -50
  171. package/src/components/Tables/common/no-data-illu.svg +23 -23
  172. package/src/components/Tables/common/styles.tsx +170 -170
  173. package/src/components/Tables/common/types.ts +57 -57
  174. package/src/components/Tabs/NavigationTabs.tsx +80 -80
  175. package/src/components/Tabs/Tabs.tsx +53 -53
  176. package/src/components/Tabs/TabsContainer.tsx +53 -53
  177. package/src/components/Tabs/index.tsx +1 -1
  178. package/src/components/Tabs/styles.tsx +55 -55
  179. package/src/components/ToastContainer/ToastContainer.tsx +57 -57
  180. package/src/components/ToastContainer/index.tsx +1 -1
  181. package/src/components/UploadButton/UploadButton.tsx +126 -126
  182. package/src/components/UploadButton/index.tsx +1 -1
  183. package/src/components/UploadButton/types.ts +19 -19
  184. package/src/components/UploadDocument.tsx +108 -108
  185. package/src/components/UploadFileDialog/UploadFileDialog.tsx +238 -238
  186. package/src/components/UploadFileDialog/index.tsx +1 -1
  187. package/src/components/index.ts +110 -110
  188. package/src/config/axios.ts +80 -80
  189. package/src/config/axiosEvaluator.ts +53 -53
  190. package/src/config/axiosXTenant.ts +57 -57
  191. package/src/constants/UIConstants.ts +124 -124
  192. package/src/constants/formValidations.ts +6 -6
  193. package/src/constants/index.ts +5 -5
  194. package/src/constants/isDevelopment.ts +4 -4
  195. package/src/constants/permissions.ts +67 -67
  196. package/src/constants/validateMessages.ts +12 -12
  197. package/src/contexts/LoginFormProvider.tsx +37 -37
  198. package/src/contexts/Providers.tsx +74 -74
  199. package/src/contexts/PublicProviders.tsx +30 -30
  200. package/src/contexts/QueryClientProvider.tsx +22 -22
  201. package/src/contexts/RootModal.tsx +76 -76
  202. package/src/hooks/index.ts +5 -5
  203. package/src/hooks/useAuth.ts +248 -248
  204. package/src/hooks/useExternalScript.ts +38 -38
  205. package/src/hooks/useFetch.ts +53 -53
  206. package/src/hooks/useFilters.ts +83 -83
  207. package/src/hooks/useRouter.ts +31 -31
  208. package/src/layouts/Components/DashBoardMenu.tsx +232 -232
  209. package/src/layouts/Components/icons/index.tsx +403 -403
  210. package/src/layouts/Components/styles.tsx +74 -74
  211. package/src/layouts/ComponentsLayout.tsx +3 -3
  212. package/src/permissions/PageWithPermission.tsx +18 -18
  213. package/src/permissions/PermissionDeniedPage.tsx +16 -16
  214. package/src/permissions/ValidateAccess.tsx +18 -18
  215. package/src/permissions/index.ts +2 -2
  216. package/src/react-app-env.d.ts +1 -1
  217. package/src/shared-state/AssetsStore.ts +15 -15
  218. package/src/shared-state/InstitutionsStore.ts +8 -8
  219. package/src/shared-state/PermissionsStore.ts +1163 -1159
  220. package/src/shared-state/UserStore.ts +13 -13
  221. package/src/shared-state/index.ts +4 -4
  222. package/src/theme/App.less +3 -3
  223. package/src/theme/MuiThemeProvider.tsx +18 -18
  224. package/src/theme/customCssBaseline.ts +135 -135
  225. package/src/theme/index.css +28 -28
  226. package/src/theme/muiTheme.ts +597 -597
  227. package/src/theme/theme.d.ts +77 -77
  228. package/src/utils/adminAxios.ts +15 -15
  229. package/src/utils/alphabet.ts +23 -23
  230. package/src/utils/arrayPadEnd.ts +3 -3
  231. package/src/utils/formatCurrency.ts +9 -9
  232. package/src/utils/getUrlParams.ts +5 -5
  233. package/src/utils/index.ts +7 -7
  234. package/src/utils/logout.ts +25 -25
  235. package/src/utils/ordinalSuffixOf.ts +14 -14
  236. package/src/utils/romanize.ts +40 -40
  237. package/src/utils/withRouteWrapper.tsx +25 -25
  238. package/src/utils/withSuspense.tsx +6 -6
  239. package/styled-components.tsx +60 -60
  240. package/tsconfig.json +21 -21
@@ -1,344 +1,378 @@
1
- import {
2
- Avatar,
3
- Box,
4
- Button,
5
- CircularProgress,
6
- InputAdornment,
7
- Typography,
8
- styled,
9
- } from '@mui/material'
10
- import { useEffect, useState } from 'react'
11
- import { useMutation, useQuery } from 'react-query'
12
- import { toast } from 'react-toastify'
13
- import { useImmer } from 'use-immer'
14
- import { PageContent } from '../PageContent'
15
- import PageHeader from '../PageHeader'
16
- import useConfirm from '../PopupConfirm/useConfirm'
17
- import Spinner from '../Spinner'
18
- import UserProfileRelation from './UserProfileRelation'
19
- import {
20
- defaultFilterObj,
21
- fetchApplicationUsers,
22
- fetchProfiles,
23
- removeUserApplicationProfile,
24
- updateUserApplicationProfile,
25
- } from './services'
26
-
27
- import { axiosErrorToast } from '../../config/axios'
28
- import { ValidateAccess } from '../../permissions'
29
- import { Permission, PermissionsStore } from '../../shared-state'
30
- import ActionButton from '../ActionButton'
31
- import SearchBar from '../FilterComponents/SearchBar'
32
- import { SingleSelect } from '../Input'
33
- import { DialogButton } from '../ModalButtons'
34
- import Table from '../Tables/BasicTable/Table'
35
-
36
- interface ApplicationProfileProps {
37
- application: 'exams' | 'square' | 'payments' | 'enroll_x' | 'hostels'
38
- title: string
39
- permissions?: {
40
- add: string
41
- edit: string
42
- view: string
43
- delete: string
44
- }
45
- }
46
- function ApplicationProfile({
47
- application,
48
- title,
49
- permissions,
50
- }: ApplicationProfileProps) {
51
- const { isConfirmed } = useConfirm()
52
- const [filters, setFilters] = useImmer(defaultFilterObj)
53
- const { data, isLoading, refetch } = useQuery(
54
- ['application-users', filters],
55
- () => fetchApplicationUsers({ application, ...filters }),
56
- )
57
-
58
- const { data: profiles, isLoading: profilesLoading } = useQuery(
59
- 'profiles',
60
- () => fetchProfiles({ application }),
61
- )
62
-
63
- const { mutate, isLoading: removingUserProfile } = useMutation(
64
- removeUserApplicationProfile,
65
- {
66
- onSuccess: (res) => {
67
- refetch()
68
- toast.success('User profile removed from application')
69
- },
70
- },
71
- )
72
-
73
- const handleRemove = async (data) => {
74
- const confirmed = await isConfirmed(
75
- 'Are you sure you want to remove the user profile?',
76
- )
77
- if (!confirmed) return
78
- mutate({ userId: data.id, application: application })
79
- }
80
- const columns = [
81
- {
82
- title: 'User',
83
- key: 'user',
84
- dataIndex: 'user',
85
- render: (_, row) => <UserComponent userData={row} />,
86
- },
87
- {
88
- title: 'Profile',
89
- key: 'profile',
90
- dataIndex: 'profile',
91
- render: (_, row) => (
92
- <RenderProfileDropDown
93
- profiles={profiles?.profiles}
94
- data={row}
95
- application={application}
96
- refetchFn={refetch}
97
- permissions={permissions}
98
- />
99
- ),
100
- },
101
- {
102
- title: 'Actions',
103
- key: 'actions',
104
- dataIndex: 'actions',
105
- render: (_, row) => (
106
- <ValidateAccess
107
- accessKey={
108
- permissions
109
- ? Permission[permissions.delete]
110
- : Permission.CAN_DASHBOARD_VIEW
111
- }
112
- >
113
- <Button
114
- variant="text"
115
- onClick={() => handleRemove(row)}
116
- sx={{ padding: '0px', margin: '0px' }}
117
- >
118
- Remove
119
- </Button>
120
- </ValidateAccess>
121
- ),
122
- },
123
- ]
124
- const handleLimitChange = (value: number) => {
125
- setFilters((s) => {
126
- s.limit = value
127
- s.offset = 0
128
- })
129
- }
130
-
131
- const handlePagination = (value: number) => {
132
- setFilters((s) => {
133
- s.offset = value * s.limit - s.limit
134
- })
135
- }
136
-
137
- if (profilesLoading) return <Spinner />
138
- return (
139
- <>
140
- <PageHeader
141
- title={title}
142
- actions={[
143
- <ValidateAccess
144
- key={1}
145
- accessKey={
146
- permissions
147
- ? Permission[permissions.add]
148
- : Permission.CAN_DASHBOARD_VIEW
149
- }
150
- >
151
- <DialogButton
152
- title={'Add User Profile Relation'}
153
- anchor={({ open }) => (
154
- <ActionButton onClick={open}>
155
- Add User Profile Relation
156
- </ActionButton>
157
- )}
158
- content={({ close }) => (
159
- <UserProfileRelation
160
- close={close}
161
- application={application}
162
- profiles={profiles?.profiles}
163
- />
164
- )}
165
- />
166
- </ValidateAccess>,
167
- ]}
168
- />
169
- <PageContent sx={{ marginTop: '25px' }}>
170
- <StyledTableContainer>
171
- <SearchBar
172
- onSearch={(value) => {
173
- setFilters((s) => {
174
- s.search = value
175
- })
176
- }}
177
- textFieldProps={{
178
- placeholder: 'Search by Name',
179
- title: 'Search by Name',
180
- sx: { width: '300px', marginBottom: '25px' },
181
- size: 'small',
182
- }}
183
- />
184
- <Table
185
- columns={columns}
186
- dataSource={data?.result ?? []}
187
- loading={isLoading || removingUserProfile}
188
- pagination={{
189
- limit: filters.limit,
190
- onChangeLimit: handleLimitChange,
191
- onChange: handlePagination,
192
- totalCount: data?.count,
193
- page: filters.offset / filters.limit,
194
- }}
195
- />
196
- </StyledTableContainer>
197
- </PageContent>
198
- </>
199
- )
200
- }
201
- export default ApplicationProfile
202
-
203
- export const RenderProfileDropDown = ({
204
- profiles,
205
- data,
206
- refetchFn,
207
- application,
208
- permissions,
209
- }) => {
210
- const permissionState = PermissionsStore.useState((s) => s)
211
- const CanEdit = permissions
212
- ? permissionState.permissions[Permission[permissions.edit]]
213
- : permissionState.permissions['can_dashboard_view']
214
-
215
- const [state, setState] = useState({
216
- userId: null,
217
- profileId: null,
218
- })
219
-
220
- useEffect(() => {
221
- setState((pre) => ({
222
- userId: data.id,
223
- profileId: data?.profiles?.find((p) => p.application === application)?.id,
224
- }))
225
- }, [data])
226
-
227
- const { mutate, isLoading } = useMutation(updateUserApplicationProfile, {
228
- onSuccess: (res) => {
229
- refetchFn()
230
- toast.success('User application profile updated successfully')
231
- },
232
- onError: (err) => {
233
- // eslint-disable-next-line no-console
234
- console.log(err)
235
- axiosErrorToast(err)
236
- },
237
- })
238
- const handleChange = (e) => {
239
- setState({
240
- userId: data.id,
241
- profileId: e.target.value,
242
- })
243
- mutate({
244
- userId: state.userId,
245
- profileIds: [e.target.value],
246
- application: application,
247
- })
248
- }
249
-
250
- if (!data) return null
251
- return (
252
- <>
253
- <StyledDropDownContainer>
254
- <SingleSelect
255
- label={'Profile'}
256
- name={'profile'}
257
- value={state?.profileId}
258
- onChange={(e) => {
259
- handleChange(e)
260
- }}
261
- disabled={!CanEdit}
262
- endAdornment={
263
- isLoading && (
264
- <InputAdornment position="end">
265
- <CircularProgress
266
- size={20}
267
- color={'secondary'}
268
- sx={{ marginRight: '10px' }}
269
- />
270
- </InputAdornment>
271
- )
272
- }
273
- options={
274
- profiles?.map((profile) => ({
275
- label: profile.name,
276
- value: profile.id,
277
- })) ?? []
278
- }
279
- />
280
- </StyledDropDownContainer>
281
- </>
282
- )
283
- }
284
-
285
- export const UserComponent = ({ userData }) => {
286
- return (
287
- <>
288
- <Box
289
- sx={{
290
- width: '100%',
291
- display: 'flex',
292
- gap: '10px',
293
- alignItems: 'center',
294
- justifyContent: 'flex-start',
295
- }}
296
- >
297
- <Avatar alt={userData?.fullName} />
298
- <Typography variant="subtitle1">{userData?.fullName}</Typography>
299
- </Box>
300
- </>
301
- )
302
- }
303
- export const StyledTableContainer = styled(Box)(({ theme }) => ({
304
- width: '70%',
305
- margin: 'auto',
306
- '& .MuiTableHead-root': {
307
- border: '1px solid white',
308
- borderBottom: theme.borders.grayLight,
309
- borderWidth: '2px',
310
-
311
- backgroundColor: 'white',
312
- '& th': {
313
- textAlign: 'left',
314
- padding: '5px',
315
- },
316
- '& .MuiBox-root': {
317
- color: '#121212b3',
318
- fontSize: '164x',
319
- },
320
- },
321
- '& tbody': {
322
- border: '1px solid white',
323
- borderBottom: theme.borders.grayLight,
324
- borderWidth: '2px',
325
- '& tr': {
326
- border: '1px solid white',
327
- },
328
- },
329
- '& td': {
330
- textAlign: 'center',
331
- border: '1px solid white',
332
- padding: '15px 0px',
333
- },
334
- }))
335
-
336
- export const StyledDropDownContainer = styled(Box)(({ theme }) => ({
337
- width: '200px',
338
- '& .MuiTypography-root': {
339
- display: 'none',
340
- },
341
- '& .MuiInputBase-root': {
342
- backgroundColor: '#f5f6f8',
343
- },
344
- }))
1
+ import {
2
+ Avatar,
3
+ Box,
4
+ Button,
5
+ CircularProgress,
6
+ InputAdornment,
7
+ Typography,
8
+ styled,
9
+ } from '@mui/material'
10
+ import { useEffect, useState } from 'react'
11
+ import { useMutation, useQuery } from 'react-query'
12
+ import { toast } from 'react-toastify'
13
+ import { useImmer } from 'use-immer'
14
+ import { PageContent } from '../PageContent'
15
+ import PageHeader from '../PageHeader'
16
+ import useConfirm from '../PopupConfirm/useConfirm'
17
+ import Spinner from '../Spinner'
18
+ import UserProfileRelation from './UserProfileRelation'
19
+ import {
20
+ defaultFilterObj,
21
+ fetchApplicationUsers,
22
+ fetchProfiles,
23
+ removeUserApplicationProfile,
24
+ updateUserApplicationProfile,
25
+ } from './services'
26
+
27
+ import { axiosErrorToast } from '../../config/axios'
28
+ import { ValidateAccess } from '../../permissions'
29
+ import { Permission, PermissionsStore } from '../../shared-state'
30
+ import ActionButton from '../ActionButton'
31
+ import FilterButton from '../FilterComponents/FilterButton'
32
+ import SearchBar from '../FilterComponents/SearchBar'
33
+ import { SingleSelect } from '../Input'
34
+ import { DialogButton, DrawerButton } from '../ModalButtons'
35
+ import Table from '../Tables/BasicTable/Table'
36
+ import DeprartmentFilter from './DepartmentFilter'
37
+
38
+ interface ApplicationProfileProps {
39
+ application: 'exams' | 'square' | 'payments' | 'enroll_x' | 'hostels'
40
+ title: string
41
+ permissions?: {
42
+ add: string
43
+ edit: string
44
+ view: string
45
+ delete: string
46
+ }
47
+ }
48
+ function ApplicationProfile({
49
+ application,
50
+ title,
51
+ permissions,
52
+ }: ApplicationProfileProps) {
53
+ const { isConfirmed } = useConfirm()
54
+ const [state, setState] = useImmer(defaultFilterObj)
55
+
56
+ const { data, isLoading, refetch } = useQuery(
57
+ [
58
+ 'application-users',
59
+ ...Object.keys(state.filters).map((key) => state.filters[key]),
60
+ ],
61
+ () => fetchApplicationUsers({ application, ...state.filters }),
62
+ )
63
+
64
+ const { data: profiles, isLoading: profilesLoading } = useQuery(
65
+ 'profiles',
66
+ () => fetchProfiles({ application }),
67
+ )
68
+
69
+ const { mutate, isLoading: removingUserProfile } = useMutation(
70
+ removeUserApplicationProfile,
71
+ {
72
+ onSuccess: () => {
73
+ refetch()
74
+ toast.success('User profile removed from application')
75
+ },
76
+ },
77
+ )
78
+
79
+ const handleRemove = async (data) => {
80
+ const confirmed = await isConfirmed(
81
+ 'Are you sure you want to remove the user profile?',
82
+ )
83
+ if (!confirmed) return
84
+ mutate({ userId: data.id, application: application })
85
+ }
86
+
87
+ const columns = [
88
+ {
89
+ title: 'User',
90
+ key: 'user',
91
+ dataIndex: 'user',
92
+ render: (_, row) => <UserComponent userData={row} />,
93
+ },
94
+ {
95
+ title: 'Profile',
96
+ key: 'profile',
97
+ dataIndex: 'profile',
98
+ render: (_, row) => (
99
+ <RenderProfileDropDown
100
+ profiles={profiles?.profiles}
101
+ data={row}
102
+ application={application}
103
+ refetchFn={refetch}
104
+ permissions={permissions}
105
+ />
106
+ ),
107
+ },
108
+ {
109
+ title: 'Actions',
110
+ key: 'actions',
111
+ dataIndex: 'actions',
112
+ render: (_, row) => (
113
+ <ValidateAccess
114
+ accessKey={
115
+ permissions
116
+ ? Permission[permissions.delete]
117
+ : Permission.CAN_DASHBOARD_VIEW
118
+ }
119
+ >
120
+ <Button
121
+ variant="text"
122
+ onClick={() => handleRemove(row)}
123
+ sx={{ padding: '0px', margin: '0px' }}
124
+ >
125
+ Remove
126
+ </Button>
127
+ </ValidateAccess>
128
+ ),
129
+ },
130
+ ]
131
+ const handleLimitChange = (value: number) => {
132
+ setState((s) => {
133
+ s.filters.limit = value
134
+ s.filters.offset = 0
135
+ })
136
+ }
137
+
138
+ const handlePagination = (value: number) => {
139
+ setState((s) => {
140
+ s.filters.offset = value * s.filters.limit - s.filters.limit
141
+ })
142
+ }
143
+
144
+ if (profilesLoading) return <Spinner />
145
+ return (
146
+ <>
147
+ <PageHeader
148
+ title={title}
149
+ actions={[
150
+ <ValidateAccess
151
+ key={1}
152
+ accessKey={
153
+ permissions
154
+ ? Permission[permissions.add]
155
+ : Permission.CAN_DASHBOARD_VIEW
156
+ }
157
+ >
158
+ <DialogButton
159
+ title={'Add User Profile Relation'}
160
+ anchor={({ open }) => (
161
+ <ActionButton onClick={open}>
162
+ Add User Profile Relation
163
+ </ActionButton>
164
+ )}
165
+ content={({ close }) => (
166
+ <UserProfileRelation
167
+ close={close}
168
+ application={application}
169
+ profiles={profiles?.profiles}
170
+ />
171
+ )}
172
+ />
173
+ </ValidateAccess>,
174
+ ]}
175
+ />
176
+ <PageContent sx={{ marginTop: '25px' }}>
177
+ <StyledTableContainer>
178
+ <Box
179
+ sx={{
180
+ display: 'flex',
181
+ gap: '10px',
182
+ alignItems: 'center',
183
+ marginBottom: '10px',
184
+ }}
185
+ >
186
+ <Box>
187
+ <SearchBar
188
+ onSearch={(value) =>
189
+ setState((s) => {
190
+ s.filters.search = value
191
+ })
192
+ }
193
+ textFieldProps={{
194
+ placeholder: 'Search by Name',
195
+ title: 'Search by Name',
196
+ sx: { width: '300px' },
197
+ size: 'medium',
198
+ }}
199
+ />
200
+ </Box>
201
+ <DrawerButton
202
+ anchor={({ open }) => (
203
+ <FilterButton
204
+ onClick={open}
205
+ filtersApplied={state.filtersApplied}
206
+ />
207
+ )}
208
+ content={({ close }) => (
209
+ <DeprartmentFilter
210
+ close={close}
211
+ setAppliedFilters={setState}
212
+ appliedFilters={state}
213
+ />
214
+ )}
215
+ title="People Filters"
216
+ />
217
+ </Box>
218
+ <Table
219
+ columns={columns}
220
+ dataSource={data?.result ?? []}
221
+ loading={isLoading || removingUserProfile}
222
+ pagination={{
223
+ limit: state.filters.limit,
224
+ onChangeLimit: handleLimitChange,
225
+ onChange: handlePagination,
226
+ totalCount: data?.count,
227
+ page: state.filters.offset / state.filters.limit,
228
+ }}
229
+ />
230
+ </StyledTableContainer>
231
+ </PageContent>
232
+ </>
233
+ )
234
+ }
235
+ export default ApplicationProfile
236
+
237
+ export const RenderProfileDropDown = ({
238
+ profiles,
239
+ data,
240
+ refetchFn,
241
+ application,
242
+ permissions,
243
+ }) => {
244
+ const permissionState = PermissionsStore.useState((s) => s)
245
+ const CanEdit = permissions
246
+ ? permissionState.permissions[Permission[permissions.edit]]
247
+ : permissionState.permissions['can_dashboard_view']
248
+
249
+ const [state, setState] = useState({
250
+ userId: null,
251
+ profileId: null,
252
+ })
253
+
254
+ useEffect(() => {
255
+ setState(() => ({
256
+ userId: data.id,
257
+ profileId: data?.profiles?.find((p) => p.application === application)?.id,
258
+ }))
259
+ }, [data])
260
+
261
+ const { mutate, isLoading } = useMutation(updateUserApplicationProfile, {
262
+ onSuccess: () => {
263
+ refetchFn()
264
+ toast.success('User application profile updated successfully')
265
+ },
266
+ onError: (err) => {
267
+ // eslint-disable-next-line no-console
268
+ console.log(err)
269
+ axiosErrorToast(err)
270
+ },
271
+ })
272
+ const handleChange = (e) => {
273
+ setState({
274
+ userId: data.id,
275
+ profileId: e.target.value,
276
+ })
277
+ mutate({
278
+ userId: state.userId,
279
+ profileIds: [e.target.value],
280
+ application: application,
281
+ })
282
+ }
283
+
284
+ if (!data) return null
285
+ return (
286
+ <>
287
+ <StyledDropDownContainer>
288
+ <SingleSelect
289
+ label={'Profile'}
290
+ name={'profile'}
291
+ value={state?.profileId}
292
+ onChange={(e) => {
293
+ handleChange(e)
294
+ }}
295
+ disabled={!CanEdit}
296
+ endAdornment={
297
+ isLoading && (
298
+ <InputAdornment position="end">
299
+ <CircularProgress
300
+ size={20}
301
+ color={'secondary'}
302
+ sx={{ marginRight: '10px' }}
303
+ />
304
+ </InputAdornment>
305
+ )
306
+ }
307
+ options={
308
+ profiles?.map((profile) => ({
309
+ label: profile.name,
310
+ value: profile.id,
311
+ })) ?? []
312
+ }
313
+ />
314
+ </StyledDropDownContainer>
315
+ </>
316
+ )
317
+ }
318
+
319
+ export const UserComponent = ({ userData }) => {
320
+ return (
321
+ <>
322
+ <Box
323
+ sx={{
324
+ width: '100%',
325
+ display: 'flex',
326
+ gap: '10px',
327
+ alignItems: 'center',
328
+ justifyContent: 'flex-start',
329
+ }}
330
+ >
331
+ <Avatar alt={userData?.fullName} />
332
+ <Typography variant="subtitle1">{userData?.fullName}</Typography>
333
+ </Box>
334
+ </>
335
+ )
336
+ }
337
+ export const StyledTableContainer = styled(Box)(({ theme }) => ({
338
+ width: '70%',
339
+ margin: 'auto',
340
+ '& .MuiTableHead-root': {
341
+ border: '1px solid white',
342
+ borderBottom: theme.borders.grayLight,
343
+ borderWidth: '2px',
344
+
345
+ backgroundColor: 'white',
346
+ '& th': {
347
+ textAlign: 'left',
348
+ padding: '5px',
349
+ },
350
+ '& .MuiBox-root': {
351
+ color: '#121212b3',
352
+ fontSize: '164x',
353
+ },
354
+ },
355
+ '& tbody': {
356
+ border: '1px solid white',
357
+ borderBottom: theme.borders.grayLight,
358
+ borderWidth: '2px',
359
+ '& tr': {
360
+ border: '1px solid white',
361
+ },
362
+ },
363
+ '& td': {
364
+ textAlign: 'center',
365
+ border: '1px solid white',
366
+ padding: '15px 0px',
367
+ },
368
+ }))
369
+
370
+ export const StyledDropDownContainer = styled(Box)(() => ({
371
+ width: '200px',
372
+ '& .MuiTypography-root': {
373
+ display: 'none',
374
+ },
375
+ '& .MuiInputBase-root': {
376
+ backgroundColor: '#f5f6f8',
377
+ },
378
+ }))