@strapi/plugin-users-permissions 0.0.0-next.f426350b859ddae6592e9bfa99e6be94ae22e117 → 0.0.0-next.f4ec69568d980c6fee91ce2ee0f41c138347aa81

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 (218) hide show
  1. package/.eslintignore +1 -2
  2. package/.eslintrc +17 -0
  3. package/LICENSE +18 -3
  4. package/admin/src/components/BoundRoute/{index.js → index.jsx} +2 -2
  5. package/admin/src/components/FormModal/Input/{index.js → index.jsx} +32 -31
  6. package/admin/src/components/FormModal/index.jsx +115 -0
  7. package/admin/src/components/Permissions/PermissionRow/{CheckboxWrapper.js → CheckboxWrapper.jsx} +4 -3
  8. package/admin/src/components/Permissions/PermissionRow/{SubCategory.js → SubCategory.jsx} +13 -22
  9. package/admin/src/components/Permissions/index.jsx +47 -0
  10. package/admin/src/components/Permissions/reducer.js +1 -1
  11. package/admin/src/components/Policies/{index.js → index.jsx} +7 -5
  12. package/admin/src/components/UsersPermissions/{index.js → index.jsx} +15 -7
  13. package/admin/src/components/UsersPermissions/reducer.js +1 -1
  14. package/admin/src/index.js +19 -47
  15. package/admin/src/pages/AdvancedSettings/index.jsx +214 -0
  16. package/admin/src/pages/AdvancedSettings/utils/layout.js +20 -35
  17. package/admin/src/pages/AdvancedSettings/utils/schema.js +5 -2
  18. package/admin/src/pages/EmailTemplates/components/EmailForm.jsx +156 -0
  19. package/admin/src/pages/EmailTemplates/components/{EmailTable.js → EmailTable.jsx} +20 -17
  20. package/admin/src/pages/EmailTemplates/index.jsx +148 -0
  21. package/admin/src/pages/EmailTemplates/utils/schema.js +18 -6
  22. package/admin/src/pages/Providers/index.jsx +262 -0
  23. package/admin/src/pages/Providers/utils/forms.js +23 -11
  24. package/admin/src/pages/Roles/constants.js +3 -3
  25. package/admin/src/{hooks → pages/Roles/hooks}/usePlugins.js +19 -12
  26. package/admin/src/pages/Roles/index.jsx +24 -0
  27. package/admin/src/pages/Roles/pages/CreatePage.jsx +194 -0
  28. package/admin/src/pages/Roles/pages/EditPage.jsx +215 -0
  29. package/admin/src/pages/Roles/pages/ListPage/components/TableBody.jsx +119 -0
  30. package/admin/src/pages/Roles/{ListPage/index.js → pages/ListPage/index.jsx} +93 -65
  31. package/admin/src/translations/en.json +1 -1
  32. package/admin/src/translations/zh-Hans.json +80 -80
  33. package/admin/src/utils/prefixPluginTranslations.js +13 -0
  34. package/dist/_chunks/ar-BguGUqwK.js +44 -0
  35. package/dist/_chunks/ar-BguGUqwK.js.map +1 -0
  36. package/dist/_chunks/ar-CK8BRRXB.mjs +44 -0
  37. package/dist/_chunks/ar-CK8BRRXB.mjs.map +1 -0
  38. package/dist/_chunks/cs-BVigMk0l.mjs +50 -0
  39. package/dist/_chunks/cs-BVigMk0l.mjs.map +1 -0
  40. package/dist/_chunks/cs-BW8-K_GY.js +50 -0
  41. package/dist/_chunks/cs-BW8-K_GY.js.map +1 -0
  42. package/dist/_chunks/de-BKUdRFI4.mjs +62 -0
  43. package/dist/_chunks/de-BKUdRFI4.mjs.map +1 -0
  44. package/dist/_chunks/de-owXpVluI.js +62 -0
  45. package/dist/_chunks/de-owXpVluI.js.map +1 -0
  46. package/dist/_chunks/dk-BQiTK50l.mjs +86 -0
  47. package/dist/_chunks/dk-BQiTK50l.mjs.map +1 -0
  48. package/dist/_chunks/dk-LXAnbuBk.js +86 -0
  49. package/dist/_chunks/dk-LXAnbuBk.js.map +1 -0
  50. package/dist/_chunks/en-DOHtPf-2.mjs +86 -0
  51. package/dist/_chunks/en-DOHtPf-2.mjs.map +1 -0
  52. package/dist/_chunks/en-MHo5mcsU.js +86 -0
  53. package/dist/_chunks/en-MHo5mcsU.js.map +1 -0
  54. package/dist/_chunks/es-BwLCLXAQ.js +86 -0
  55. package/dist/_chunks/es-BwLCLXAQ.js.map +1 -0
  56. package/dist/_chunks/es-DNgOVMjD.mjs +86 -0
  57. package/dist/_chunks/es-DNgOVMjD.mjs.map +1 -0
  58. package/dist/_chunks/fr-DkgRugiU.mjs +50 -0
  59. package/dist/_chunks/fr-DkgRugiU.mjs.map +1 -0
  60. package/dist/_chunks/fr-DkhpSjjm.js +50 -0
  61. package/dist/_chunks/fr-DkhpSjjm.js.map +1 -0
  62. package/dist/_chunks/id-BTemOeTZ.js +62 -0
  63. package/dist/_chunks/id-BTemOeTZ.js.map +1 -0
  64. package/dist/_chunks/id-BdEsvnaF.mjs +62 -0
  65. package/dist/_chunks/id-BdEsvnaF.mjs.map +1 -0
  66. package/dist/_chunks/index-2awRBazk.js +281 -0
  67. package/dist/_chunks/index-2awRBazk.js.map +1 -0
  68. package/dist/_chunks/index-BAHBK68t.js +1172 -0
  69. package/dist/_chunks/index-BAHBK68t.js.map +1 -0
  70. package/dist/_chunks/index-BHbzbu1p.mjs +246 -0
  71. package/dist/_chunks/index-BHbzbu1p.mjs.map +1 -0
  72. package/dist/_chunks/index-Be4qNiZI.js +640 -0
  73. package/dist/_chunks/index-Be4qNiZI.js.map +1 -0
  74. package/dist/_chunks/index-C88EQQJQ-C4oUQUND.js +12026 -0
  75. package/dist/_chunks/index-C88EQQJQ-C4oUQUND.js.map +1 -0
  76. package/dist/_chunks/index-C88EQQJQ-DAZ1lfuF.mjs +12002 -0
  77. package/dist/_chunks/index-C88EQQJQ-DAZ1lfuF.mjs.map +1 -0
  78. package/dist/_chunks/index-CbKOY95_.mjs +344 -0
  79. package/dist/_chunks/index-CbKOY95_.mjs.map +1 -0
  80. package/dist/_chunks/index-CeweK3q9.mjs +617 -0
  81. package/dist/_chunks/index-CeweK3q9.mjs.map +1 -0
  82. package/dist/_chunks/index-QcDREbPt.mjs +262 -0
  83. package/dist/_chunks/index-QcDREbPt.mjs.map +1 -0
  84. package/dist/_chunks/index-YTFP-hNZ.js +245 -0
  85. package/dist/_chunks/index-YTFP-hNZ.js.map +1 -0
  86. package/dist/_chunks/index-YZkqoYZN.js +366 -0
  87. package/dist/_chunks/index-YZkqoYZN.js.map +1 -0
  88. package/dist/_chunks/index-mKh-etKG.mjs +1142 -0
  89. package/dist/_chunks/index-mKh-etKG.mjs.map +1 -0
  90. package/dist/_chunks/it-B-rv0E24.mjs +62 -0
  91. package/dist/_chunks/it-B-rv0E24.mjs.map +1 -0
  92. package/dist/_chunks/it-D1rH6V6_.js +62 -0
  93. package/dist/_chunks/it-D1rH6V6_.js.map +1 -0
  94. package/dist/_chunks/ja-C8K-VBPD.mjs +48 -0
  95. package/dist/_chunks/ja-C8K-VBPD.mjs.map +1 -0
  96. package/dist/_chunks/ja-DqShgTMf.js +48 -0
  97. package/dist/_chunks/ja-DqShgTMf.js.map +1 -0
  98. package/dist/_chunks/ko-B9DGEPWH.js +86 -0
  99. package/dist/_chunks/ko-B9DGEPWH.js.map +1 -0
  100. package/dist/_chunks/ko-Busb0wIY.mjs +86 -0
  101. package/dist/_chunks/ko-Busb0wIY.mjs.map +1 -0
  102. package/dist/_chunks/ms-ByvsQjRt.mjs +49 -0
  103. package/dist/_chunks/ms-ByvsQjRt.mjs.map +1 -0
  104. package/dist/_chunks/ms-CPBU3LWf.js +49 -0
  105. package/dist/_chunks/ms-CPBU3LWf.js.map +1 -0
  106. package/dist/_chunks/nl-5qO8Rpcy.mjs +48 -0
  107. package/dist/_chunks/nl-5qO8Rpcy.mjs.map +1 -0
  108. package/dist/_chunks/nl-CwNB6YoO.js +48 -0
  109. package/dist/_chunks/nl-CwNB6YoO.js.map +1 -0
  110. package/dist/_chunks/pl-BdIzifBE.mjs +86 -0
  111. package/dist/_chunks/pl-BdIzifBE.mjs.map +1 -0
  112. package/dist/_chunks/pl-Do9UD69f.js +86 -0
  113. package/dist/_chunks/pl-Do9UD69f.js.map +1 -0
  114. package/dist/_chunks/pt-BIO24ioG.mjs +48 -0
  115. package/dist/_chunks/pt-BIO24ioG.mjs.map +1 -0
  116. package/dist/_chunks/pt-BR-D7dZhxuP.js +44 -0
  117. package/dist/_chunks/pt-BR-D7dZhxuP.js.map +1 -0
  118. package/dist/_chunks/pt-BR-f0p23AQZ.mjs +44 -0
  119. package/dist/_chunks/pt-BR-f0p23AQZ.mjs.map +1 -0
  120. package/dist/_chunks/pt-fdvyOnUp.js +48 -0
  121. package/dist/_chunks/pt-fdvyOnUp.js.map +1 -0
  122. package/dist/_chunks/ru-C94rjPGA.js +86 -0
  123. package/dist/_chunks/ru-C94rjPGA.js.map +1 -0
  124. package/dist/_chunks/ru-VWy-IB7K.mjs +86 -0
  125. package/dist/_chunks/ru-VWy-IB7K.mjs.map +1 -0
  126. package/dist/_chunks/sk-BABEhykl.js +50 -0
  127. package/dist/_chunks/sk-BABEhykl.js.map +1 -0
  128. package/dist/_chunks/sk-B_LIcepm.mjs +50 -0
  129. package/dist/_chunks/sk-B_LIcepm.mjs.map +1 -0
  130. package/dist/_chunks/sv-ABLKOokl.mjs +86 -0
  131. package/dist/_chunks/sv-ABLKOokl.mjs.map +1 -0
  132. package/dist/_chunks/sv-Be43LhA9.js +86 -0
  133. package/dist/_chunks/sv-Be43LhA9.js.map +1 -0
  134. package/dist/_chunks/th-DKyP7ueR.mjs +60 -0
  135. package/dist/_chunks/th-DKyP7ueR.mjs.map +1 -0
  136. package/dist/_chunks/th-DgVhVLhL.js +60 -0
  137. package/dist/_chunks/th-DgVhVLhL.js.map +1 -0
  138. package/dist/_chunks/tr-B_idhkEs.js +85 -0
  139. package/dist/_chunks/tr-B_idhkEs.js.map +1 -0
  140. package/dist/_chunks/tr-qa1Q5UjC.mjs +85 -0
  141. package/dist/_chunks/tr-qa1Q5UjC.mjs.map +1 -0
  142. package/dist/_chunks/uk-BmRqbeQc.mjs +49 -0
  143. package/dist/_chunks/uk-BmRqbeQc.mjs.map +1 -0
  144. package/dist/_chunks/uk-LHOivnhP.js +49 -0
  145. package/dist/_chunks/uk-LHOivnhP.js.map +1 -0
  146. package/dist/_chunks/vi-CdVRdKDw.js +50 -0
  147. package/dist/_chunks/vi-CdVRdKDw.js.map +1 -0
  148. package/dist/_chunks/vi-HW-EdMea.mjs +50 -0
  149. package/dist/_chunks/vi-HW-EdMea.mjs.map +1 -0
  150. package/dist/_chunks/zh-5hKkVPA4.mjs +86 -0
  151. package/dist/_chunks/zh-5hKkVPA4.mjs.map +1 -0
  152. package/dist/_chunks/zh-Cuq8gMnF.js +86 -0
  153. package/dist/_chunks/zh-Cuq8gMnF.js.map +1 -0
  154. package/dist/_chunks/zh-Hans-BHilK-yc.mjs +86 -0
  155. package/dist/_chunks/zh-Hans-BHilK-yc.mjs.map +1 -0
  156. package/dist/_chunks/zh-Hans-GQDMKtY4.js +86 -0
  157. package/dist/_chunks/zh-Hans-GQDMKtY4.js.map +1 -0
  158. package/dist/admin/index.js +4 -0
  159. package/dist/admin/index.js.map +1 -0
  160. package/dist/admin/index.mjs +5 -0
  161. package/dist/admin/index.mjs.map +1 -0
  162. package/jest.config.front.js +1 -1
  163. package/package.json +46 -27
  164. package/packup.config.ts +22 -0
  165. package/server/bootstrap/index.js +18 -15
  166. package/server/bootstrap/users-permissions-actions.js +6 -0
  167. package/server/config.js +29 -0
  168. package/server/content-types/user/index.js +0 -1
  169. package/server/controllers/auth.js +74 -38
  170. package/server/controllers/content-manager-user.js +28 -30
  171. package/server/controllers/role.js +17 -4
  172. package/server/controllers/user.js +18 -8
  173. package/server/controllers/validation/auth.js +81 -25
  174. package/server/middlewares/rateLimit.js +1 -1
  175. package/server/register.js +1 -1
  176. package/server/services/jwt.js +3 -3
  177. package/server/services/permission.js +3 -7
  178. package/server/services/providers-registry.js +469 -261
  179. package/server/services/providers.js +10 -5
  180. package/server/services/role.js +15 -13
  181. package/server/services/user.js +56 -19
  182. package/server/services/users-permissions.js +15 -13
  183. package/server/utils/index.d.ts +2 -1
  184. package/server/utils/sanitize/sanitizers.js +7 -3
  185. package/server/utils/sanitize/visitors/remove-user-relation-from-role-entities.js +2 -2
  186. package/.eslintrc.js +0 -14
  187. package/admin/src/components/FormModal/index.js +0 -126
  188. package/admin/src/components/Permissions/index.js +0 -57
  189. package/admin/src/hooks/index.js +0 -5
  190. package/admin/src/hooks/useFetchRole/index.js +0 -67
  191. package/admin/src/hooks/useFetchRole/reducer.js +0 -31
  192. package/admin/src/hooks/useForm/index.js +0 -68
  193. package/admin/src/hooks/useForm/reducer.js +0 -40
  194. package/admin/src/hooks/useRolesList/index.js +0 -65
  195. package/admin/src/hooks/useRolesList/init.js +0 -5
  196. package/admin/src/hooks/useRolesList/reducer.js +0 -31
  197. package/admin/src/pages/AdvancedSettings/index.js +0 -242
  198. package/admin/src/pages/AdvancedSettings/utils/api.js +0 -16
  199. package/admin/src/pages/EmailTemplates/components/EmailForm.js +0 -176
  200. package/admin/src/pages/EmailTemplates/index.js +0 -159
  201. package/admin/src/pages/EmailTemplates/utils/api.js +0 -16
  202. package/admin/src/pages/Providers/index.js +0 -271
  203. package/admin/src/pages/Providers/reducer.js +0 -54
  204. package/admin/src/pages/Providers/utils/api.js +0 -24
  205. package/admin/src/pages/Providers/utils/createProvidersArray.js +0 -21
  206. package/admin/src/pages/Roles/CreatePage.js +0 -185
  207. package/admin/src/pages/Roles/EditPage.js +0 -197
  208. package/admin/src/pages/Roles/ListPage/components/TableBody.js +0 -93
  209. package/admin/src/pages/Roles/ListPage/utils/api.js +0 -30
  210. package/admin/src/pages/Roles/ProtectedCreatePage.js +0 -15
  211. package/admin/src/pages/Roles/ProtectedEditPage.js +0 -15
  212. package/admin/src/pages/Roles/ProtectedListPage.js +0 -17
  213. package/admin/src/pages/Roles/index.js +0 -30
  214. package/server/bootstrap/grant-config.js +0 -131
  215. package/strapi-admin.js +0 -3
  216. package/strapi-server.js +0 -3
  217. /package/admin/src/components/Permissions/PermissionRow/{index.js → index.jsx} +0 -0
  218. /package/admin/src/contexts/UsersPermissionsContext/{index.js → index.jsx} +0 -0
@@ -0,0 +1,194 @@
1
+ import * as React from 'react';
2
+
3
+ import {
4
+ Button,
5
+ Flex,
6
+ Grid,
7
+ Main,
8
+ Textarea,
9
+ TextInput,
10
+ Typography,
11
+ Field,
12
+ } from '@strapi/design-system';
13
+ import { Check } from '@strapi/icons';
14
+ import { Page, useTracking, useNotification, useFetchClient, Layouts } from '@strapi/strapi/admin';
15
+ import { Formik, Form } from 'formik';
16
+ import { useIntl } from 'react-intl';
17
+ import { useMutation } from 'react-query';
18
+ import { useNavigate } from 'react-router-dom';
19
+
20
+ import UsersPermissions from '../../../components/UsersPermissions';
21
+ import { PERMISSIONS } from '../../../constants';
22
+ import getTrad from '../../../utils/getTrad';
23
+ import { createRoleSchema } from '../constants';
24
+ import { usePlugins } from '../hooks/usePlugins';
25
+
26
+ export const CreatePage = () => {
27
+ const { formatMessage } = useIntl();
28
+ const { toggleNotification } = useNotification();
29
+ const navigate = useNavigate();
30
+ const { isLoading: isLoadingPlugins, permissions, routes } = usePlugins();
31
+ const { trackUsage } = useTracking();
32
+ const permissionsRef = React.useRef();
33
+ const { post } = useFetchClient();
34
+ const mutation = useMutation((body) => post(`/users-permissions/roles`, body), {
35
+ onError() {
36
+ toggleNotification({
37
+ type: 'danger',
38
+ message: formatMessage({
39
+ id: 'notification.error',
40
+ defaultMessage: 'An error occurred',
41
+ }),
42
+ });
43
+ },
44
+
45
+ onSuccess() {
46
+ trackUsage('didCreateRole');
47
+
48
+ toggleNotification({
49
+ type: 'success',
50
+ message: formatMessage({
51
+ id: getTrad('Settings.roles.created'),
52
+ defaultMessage: 'Role created',
53
+ }),
54
+ });
55
+
56
+ // Forcing redirecting since we don't have the id in the response
57
+ navigate(-1);
58
+ },
59
+ });
60
+
61
+ const handleCreateRoleSubmit = async (data) => {
62
+ // TODO: refactor. Child -> parent component communication is evil;
63
+ // We should either move the provider one level up or move the state
64
+ // straight into redux.
65
+ const permissions = permissionsRef.current.getPermissions();
66
+
67
+ await mutation.mutate({ ...data, ...permissions, users: [] });
68
+ };
69
+
70
+ return (
71
+ <Main>
72
+ <Page.Title>
73
+ {formatMessage(
74
+ { id: 'Settings.PageTitle', defaultMessage: 'Settings - {name}' },
75
+ { name: 'Roles' }
76
+ )}
77
+ </Page.Title>
78
+ <Formik
79
+ enableReinitialize
80
+ initialValues={{ name: '', description: '' }}
81
+ onSubmit={handleCreateRoleSubmit}
82
+ validationSchema={createRoleSchema}
83
+ >
84
+ {({ handleSubmit, values, handleChange, errors }) => (
85
+ <Form noValidate onSubmit={handleSubmit}>
86
+ <Layouts.Header
87
+ primaryAction={
88
+ !isLoadingPlugins && (
89
+ <Button type="submit" loading={mutation.isLoading} startIcon={<Check />}>
90
+ {formatMessage({
91
+ id: 'global.save',
92
+ defaultMessage: 'Save',
93
+ })}
94
+ </Button>
95
+ )
96
+ }
97
+ title={formatMessage({
98
+ id: 'Settings.roles.create.title',
99
+ defaultMessage: 'Create a role',
100
+ })}
101
+ subtitle={formatMessage({
102
+ id: 'Settings.roles.create.description',
103
+ defaultMessage: 'Define the rights given to the role',
104
+ })}
105
+ />
106
+ <Layouts.Content>
107
+ <Flex
108
+ background="neutral0"
109
+ direction="column"
110
+ alignItems="stretch"
111
+ gap={7}
112
+ hasRadius
113
+ paddingTop={6}
114
+ paddingBottom={6}
115
+ paddingLeft={7}
116
+ paddingRight={7}
117
+ shadow="filterShadow"
118
+ >
119
+ <Flex direction="column" alignItems="stretch">
120
+ <Typography variant="delta" tag="h2">
121
+ {formatMessage({
122
+ id: getTrad('EditPage.form.roles'),
123
+ defaultMessage: 'Role details',
124
+ })}
125
+ </Typography>
126
+
127
+ <Grid.Root gap={4}>
128
+ <Grid.Item col={6} direction="column" alignItems="stretch">
129
+ <Field.Root
130
+ name="name"
131
+ error={
132
+ errors?.name
133
+ ? formatMessage({ id: errors.name, defaultMessage: 'Name is required' })
134
+ : false
135
+ }
136
+ required
137
+ >
138
+ <Field.Label>
139
+ {formatMessage({
140
+ id: 'global.name',
141
+ defaultMessage: 'Name',
142
+ })}
143
+ </Field.Label>
144
+ <TextInput value={values.name || ''} onChange={handleChange} />
145
+ <Field.Error />
146
+ </Field.Root>
147
+ </Grid.Item>
148
+ <Grid.Item col={6} direction="column" alignItems="stretch">
149
+ <Field.Root
150
+ name="description"
151
+ error={
152
+ errors?.description
153
+ ? formatMessage({
154
+ id: errors.description,
155
+ defaultMessage: 'Description is required',
156
+ })
157
+ : false
158
+ }
159
+ required
160
+ >
161
+ <Field.Label>
162
+ {formatMessage({
163
+ id: 'global.description',
164
+ defaultMessage: 'Description',
165
+ })}
166
+ </Field.Label>
167
+ <Textarea value={values.description || ''} onChange={handleChange} />
168
+ <Field.Error />
169
+ </Field.Root>
170
+ </Grid.Item>
171
+ </Grid.Root>
172
+ </Flex>
173
+
174
+ {!isLoadingPlugins && (
175
+ <UsersPermissions
176
+ ref={permissionsRef}
177
+ permissions={permissions}
178
+ routes={routes}
179
+ />
180
+ )}
181
+ </Flex>
182
+ </Layouts.Content>
183
+ </Form>
184
+ )}
185
+ </Formik>
186
+ </Main>
187
+ );
188
+ };
189
+
190
+ export const ProtectedRolesCreatePage = () => (
191
+ <Page.Protect permissions={PERMISSIONS.createRole}>
192
+ <CreatePage />
193
+ </Page.Protect>
194
+ );
@@ -0,0 +1,215 @@
1
+ import * as React from 'react';
2
+
3
+ import {
4
+ Main,
5
+ Button,
6
+ Flex,
7
+ TextInput,
8
+ Textarea,
9
+ Typography,
10
+ Grid,
11
+ Field,
12
+ } from '@strapi/design-system';
13
+ import { Check } from '@strapi/icons';
14
+ import {
15
+ Page,
16
+ BackButton,
17
+ useAPIErrorHandler,
18
+ useNotification,
19
+ useFetchClient,
20
+ Layouts,
21
+ } from '@strapi/strapi/admin';
22
+ import { Formik, Form } from 'formik';
23
+ import { useIntl } from 'react-intl';
24
+ import { useQuery, useMutation } from 'react-query';
25
+ import { useMatch } from 'react-router-dom';
26
+
27
+ import UsersPermissions from '../../../components/UsersPermissions';
28
+ import { PERMISSIONS } from '../../../constants';
29
+ import getTrad from '../../../utils/getTrad';
30
+ import { createRoleSchema } from '../constants';
31
+ import { usePlugins } from '../hooks/usePlugins';
32
+
33
+ export const EditPage = () => {
34
+ const { formatMessage } = useIntl();
35
+ const { toggleNotification } = useNotification();
36
+ const {
37
+ params: { id },
38
+ } = useMatch(`/settings/users-permissions/roles/:id`);
39
+ const { get } = useFetchClient();
40
+ const { isLoading: isLoadingPlugins, routes } = usePlugins();
41
+ const {
42
+ data: role,
43
+ isLoading: isLoadingRole,
44
+ refetch: refetchRole,
45
+ } = useQuery(['users-permissions', 'role', id], async () => {
46
+ // TODO: why doesn't this endpoint follow the admin API conventions?
47
+ const {
48
+ data: { role },
49
+ } = await get(`/users-permissions/roles/${id}`);
50
+
51
+ return role;
52
+ });
53
+
54
+ const permissionsRef = React.useRef();
55
+ const { put } = useFetchClient();
56
+ const { formatAPIError } = useAPIErrorHandler();
57
+ const mutation = useMutation((body) => put(`/users-permissions/roles/${id}`, body), {
58
+ onError(error) {
59
+ toggleNotification({
60
+ type: 'danger',
61
+ message: formatAPIError(error),
62
+ });
63
+ },
64
+
65
+ async onSuccess() {
66
+ toggleNotification({
67
+ type: 'success',
68
+ message: formatMessage({
69
+ id: getTrad('Settings.roles.created'),
70
+ defaultMessage: 'Role edited',
71
+ }),
72
+ });
73
+
74
+ await refetchRole();
75
+ },
76
+ });
77
+
78
+ const handleEditRoleSubmit = async (data) => {
79
+ const permissions = permissionsRef.current.getPermissions();
80
+
81
+ await mutation.mutate({ ...data, ...permissions, users: [] });
82
+ };
83
+
84
+ if (isLoadingRole) {
85
+ return <Page.Loading />;
86
+ }
87
+
88
+ return (
89
+ <Main>
90
+ <Page.Title>
91
+ {formatMessage(
92
+ { id: 'Settings.PageTitle', defaultMessage: 'Settings - {name}' },
93
+ { name: 'Roles' }
94
+ )}
95
+ </Page.Title>
96
+ <Formik
97
+ enableReinitialize
98
+ initialValues={{ name: role.name, description: role.description }}
99
+ onSubmit={handleEditRoleSubmit}
100
+ validationSchema={createRoleSchema}
101
+ >
102
+ {({ handleSubmit, values, handleChange, errors }) => (
103
+ <Form noValidate onSubmit={handleSubmit}>
104
+ <Layouts.Header
105
+ primaryAction={
106
+ !isLoadingPlugins ? (
107
+ <Button
108
+ disabled={role.code === 'strapi-super-admin'}
109
+ type="submit"
110
+ loading={mutation.isLoading}
111
+ startIcon={<Check />}
112
+ >
113
+ {formatMessage({
114
+ id: 'global.save',
115
+ defaultMessage: 'Save',
116
+ })}
117
+ </Button>
118
+ ) : null
119
+ }
120
+ title={role.name}
121
+ subtitle={role.description}
122
+ navigationAction={<BackButton fallback=".." />}
123
+ />
124
+ <Layouts.Content>
125
+ <Flex
126
+ background="neutral0"
127
+ direction="column"
128
+ alignItems="stretch"
129
+ gap={7}
130
+ hasRadius
131
+ paddingTop={6}
132
+ paddingBottom={6}
133
+ paddingLeft={7}
134
+ paddingRight={7}
135
+ shadow="filterShadow"
136
+ >
137
+ <Flex direction="column" alignItems="stretch" gap={4}>
138
+ <Typography variant="delta" tag="h2">
139
+ {formatMessage({
140
+ id: getTrad('EditPage.form.roles'),
141
+ defaultMessage: 'Role details',
142
+ })}
143
+ </Typography>
144
+
145
+ <Grid.Root gap={4}>
146
+ <Grid.Item col={6} direction="column" alignItems="stretch">
147
+ <Field.Root
148
+ name="name"
149
+ error={
150
+ errors?.name
151
+ ? formatMessage({
152
+ id: errors.name,
153
+ defaultMessage: 'Name is required',
154
+ })
155
+ : false
156
+ }
157
+ required
158
+ >
159
+ <Field.Label>
160
+ {formatMessage({
161
+ id: 'global.name',
162
+ defaultMessage: 'Name',
163
+ })}
164
+ </Field.Label>
165
+ <TextInput value={values.name || ''} onChange={handleChange} />
166
+ <Field.Error />
167
+ </Field.Root>
168
+ </Grid.Item>
169
+ <Grid.Item col={6} direction="column" alignItems="stretch">
170
+ <Field.Root
171
+ name="description"
172
+ error={
173
+ errors?.description
174
+ ? formatMessage({
175
+ id: errors.description,
176
+ defaultMessage: 'Description is required',
177
+ })
178
+ : false
179
+ }
180
+ required
181
+ >
182
+ <Field.Label>
183
+ {formatMessage({
184
+ id: 'global.description',
185
+ defaultMessage: 'Description',
186
+ })}
187
+ </Field.Label>
188
+ <Textarea value={values.description || ''} onChange={handleChange} />
189
+ <Field.Error />
190
+ </Field.Root>
191
+ </Grid.Item>
192
+ </Grid.Root>
193
+ </Flex>
194
+
195
+ {!isLoadingPlugins && (
196
+ <UsersPermissions
197
+ ref={permissionsRef}
198
+ permissions={role.permissions}
199
+ routes={routes}
200
+ />
201
+ )}
202
+ </Flex>
203
+ </Layouts.Content>
204
+ </Form>
205
+ )}
206
+ </Formik>
207
+ </Main>
208
+ );
209
+ };
210
+
211
+ export const ProtectedRolesEditPage = () => (
212
+ <Page.Protect permissions={PERMISSIONS.updateRole}>
213
+ <EditPage />
214
+ </Page.Protect>
215
+ );
@@ -0,0 +1,119 @@
1
+ import React from 'react';
2
+
3
+ import { Flex, IconButton, Link, Tbody, Td, Tr, Typography } from '@strapi/design-system';
4
+ import { Pencil, Trash } from '@strapi/icons';
5
+ import PropTypes from 'prop-types';
6
+ import { useIntl } from 'react-intl';
7
+ import { useNavigate, NavLink } from 'react-router-dom';
8
+ import { styled } from 'styled-components';
9
+
10
+ const EditLink = styled(Link)`
11
+ align-items: center;
12
+ height: 3.2rem;
13
+ width: 3.2rem;
14
+ display: flex;
15
+ justify-content: center;
16
+ padding: ${({ theme }) => `${theme.spaces[2]}`};
17
+
18
+ svg {
19
+ height: 1.6rem;
20
+ width: 1.6rem;
21
+
22
+ path {
23
+ fill: ${({ theme }) => theme.colors.neutral500};
24
+ }
25
+ }
26
+
27
+ &:hover,
28
+ &:focus {
29
+ svg {
30
+ path {
31
+ fill: ${({ theme }) => theme.colors.neutral800};
32
+ }
33
+ }
34
+ }
35
+ `;
36
+
37
+ const TableBody = ({ sortedRoles, canDelete, canUpdate, setRoleToDelete, onDelete }) => {
38
+ const { formatMessage } = useIntl();
39
+ const navigate = useNavigate();
40
+ const [showConfirmDelete, setShowConfirmDelete] = onDelete;
41
+
42
+ const checkCanDeleteRole = (role) =>
43
+ canDelete && !['public', 'authenticated'].includes(role.type);
44
+
45
+ const handleClickDelete = (id) => {
46
+ setRoleToDelete(id);
47
+ setShowConfirmDelete(!showConfirmDelete);
48
+ };
49
+
50
+ return (
51
+ <Tbody>
52
+ {sortedRoles?.map((role) => (
53
+ <Tr key={role.name} onClick={() => navigate(role.id.toString())}>
54
+ <Td width="20%">
55
+ <Typography>{role.name}</Typography>
56
+ </Td>
57
+ <Td width="50%">
58
+ <Typography>{role.description}</Typography>
59
+ </Td>
60
+ <Td width="30%">
61
+ <Typography>
62
+ {formatMessage(
63
+ {
64
+ id: 'Roles.RoleRow.user-count',
65
+ defaultMessage: '{number, plural, =0 {# user} one {# user} other {# users}}',
66
+ },
67
+ { number: role.nb_users }
68
+ )}
69
+ </Typography>
70
+ </Td>
71
+ <Td>
72
+ <Flex justifyContent="end" onClick={(e) => e.stopPropagation()}>
73
+ {canUpdate ? (
74
+ <EditLink
75
+ tag={NavLink}
76
+ to={role.id.toString()}
77
+ aria-label={formatMessage(
78
+ { id: 'app.component.table.edit', defaultMessage: 'Edit {target}' },
79
+ { target: `${role.name}` }
80
+ )}
81
+ >
82
+ <Pencil />
83
+ </EditLink>
84
+ ) : null}
85
+
86
+ {checkCanDeleteRole(role) && (
87
+ <IconButton
88
+ onClick={() => handleClickDelete(role.id.toString())}
89
+ variant="ghost"
90
+ label={formatMessage(
91
+ { id: 'global.delete-target', defaultMessage: 'Delete {target}' },
92
+ { target: `${role.name}` }
93
+ )}
94
+ >
95
+ <Trash />
96
+ </IconButton>
97
+ )}
98
+ </Flex>
99
+ </Td>
100
+ </Tr>
101
+ ))}
102
+ </Tbody>
103
+ );
104
+ };
105
+
106
+ export default TableBody;
107
+
108
+ TableBody.defaultProps = {
109
+ canDelete: false,
110
+ canUpdate: false,
111
+ };
112
+
113
+ TableBody.propTypes = {
114
+ onDelete: PropTypes.array.isRequired,
115
+ setRoleToDelete: PropTypes.func.isRequired,
116
+ sortedRoles: PropTypes.array.isRequired,
117
+ canDelete: PropTypes.bool,
118
+ canUpdate: PropTypes.bool,
119
+ };