Skip to content

Commit

Permalink
Merge pull request #10278 from marmelab/canAccess-demos
Browse files Browse the repository at this point in the history
Fix `useCanAccess` causes extra rerender + use `useCanAccess` in the simple demo
  • Loading branch information
djhi authored Oct 17, 2024
2 parents 4c8cc23 + ecb319b commit 4842169
Show file tree
Hide file tree
Showing 12 changed files with 143 additions and 86 deletions.
4 changes: 4 additions & 0 deletions cypress/e2e/navigation.cy.js
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,10 @@ describe('Navigation', () => {
ListPage.navigate();

ListPage.waitUntilVisible();
// We need to wait for 'John Doe' and 'Posts' to be visible, because enabling canAccess triggers
// additional rerenders, and otherwise it's the 'Skip to content' button that gets focused
cy.contains('John Doe');
cy.contains('Posts');
cy.get(ListPage.elements.profile).focus().tab();

cy.get(`${ListPage.elements.menuItems}:first-child`).should(
Expand Down
35 changes: 31 additions & 4 deletions examples/simple/src/authProvider.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,33 @@ const DEFAULT_IDENTITY = {
avatar: '',
};

const accessControlStrategies = {
admin: () => {
return true;
},
user: ({ resource, action }) => {
const deniedResources = ['posts.authors', 'users.role', 'users.id'];
const deniedActions = ['batch_create'];
return (
!deniedResources.includes(resource) &&
!deniedActions.includes(action)
);
},
default: ({ resource, action }) => {
const deniedResources = [
'users',
'posts.authors',
'users.role',
'users.id',
];
const deniedActions = ['batch_create'];
return (
!deniedResources.includes(resource) &&
!deniedActions.includes(action)
);
},
};

// Authenticated by default
const authProvider: AuthProvider = {
login: ({ username, password }) => {
Expand Down Expand Up @@ -60,10 +87,6 @@ const authProvider: AuthProvider = {
? Promise.reject()
: Promise.resolve();
},
getPermissions: () => {
const role = localStorage.getItem('role');
return Promise.resolve(role ?? '');
},
getIdentity: () => {
const id = localStorage.getItem('login');
if (!id) {
Expand All @@ -75,6 +98,10 @@ const authProvider: AuthProvider = {
avatar: localStorage.getItem('avatar') ?? '',
});
},
canAccess: async ({ resource, action }) => {
const role = localStorage.getItem('role') || 'default';
return accessControlStrategies[role]({ resource, action });
},
};

export default authProvider;
40 changes: 16 additions & 24 deletions examples/simple/src/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -29,36 +29,28 @@ root.render(
title="Example Admin"
layout={Layout}
>
<Resource name="posts" {...posts} />
<Resource name="comments" {...comments} />
<Resource name="tags" {...tags} />
<Resource name="users" {...users} />
<CustomRoutes noLayout>
<Route
path="/custom"
element={<CustomRouteNoLayout title="Posts from /custom" />}
/>
<Route
path="/custom1"
element={
<CustomRouteNoLayout title="Posts from /custom1" />
}
/>
</CustomRoutes>
<CustomRoutes>
<Route
path="/custom2"
element={<CustomRouteLayout title="Posts from /custom2" />}
/>
</CustomRoutes>
<Resource name="posts" {...posts} />
<Resource name="comments" {...comments} />
<Resource name="tags" {...tags} />
{permissions => (
<>
{permissions ? <Resource name="users" {...users} /> : null}
<CustomRoutes noLayout>
<Route
path="/custom1"
element={
<CustomRouteNoLayout title="Posts from /custom1" />
}
/>
</CustomRoutes>
<CustomRoutes>
<Route
path="/custom2"
element={
<CustomRouteLayout title="Posts from /custom2" />
}
/>
</CustomRoutes>
</>
)}
<CustomRoutes>
<Route
path="/custom3"
Expand Down
7 changes: 3 additions & 4 deletions examples/simple/src/posts/PostCreate.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -22,10 +22,10 @@ import {
TextInput,
Toolbar,
useNotify,
usePermissions,
useRedirect,
useCreate,
useCreateSuggestionContext,
CanAccess,
} from 'react-admin';
import { useFormContext, useWatch } from 'react-hook-form';
import { Button, Dialog, DialogActions, DialogContent } from '@mui/material';
Expand Down Expand Up @@ -101,7 +101,6 @@ const PostCreate = () => {
}),
[]
);
const { permissions } = usePermissions();
const dateDefaultValue = useMemo(() => new Date(), []);
return (
<Create redirect="edit">
Expand Down Expand Up @@ -153,7 +152,7 @@ const PostCreate = () => {
<TextInput source="url" defaultValue="" />
</SimpleFormIterator>
</ArrayInput>
{permissions === 'admin' && (
<CanAccess action="edit" resource="posts.authors">
<ArrayInput source="authors">
<SimpleFormIterator>
<ReferenceInput source="user_id" reference="users">
Expand Down Expand Up @@ -188,7 +187,7 @@ const PostCreate = () => {
</FormDataConsumer>
</SimpleFormIterator>
</ArrayInput>
)}
</CanAccess>
</SimpleFormConfigurable>
</Create>
);
Expand Down
7 changes: 3 additions & 4 deletions examples/simple/src/posts/PostEdit.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,7 @@ import {
FormDataConsumer,
useCreateSuggestionContext,
EditActionsProps,
usePermissions,
CanAccess,
} from 'react-admin'; // eslint-disable-line import/no-unresolved
import {
Box,
Expand Down Expand Up @@ -101,7 +101,6 @@ const categories = [
];

const PostEdit = () => {
const { permissions } = usePermissions();
return (
<Edit title={<PostTitle />} actions={<EditActions />}>
<TabbedForm
Expand Down Expand Up @@ -148,7 +147,7 @@ const PostEdit = () => {
>
<ImageField source="src" title="title" />
</ImageInput>
{permissions === 'admin' && (
<CanAccess action="edit" resource="posts.authors">
<ArrayInput source="authors">
<SimpleFormIterator inline>
<ReferenceInput
Expand Down Expand Up @@ -184,7 +183,7 @@ const PostEdit = () => {
</FormDataConsumer>
</SimpleFormIterator>
</ArrayInput>
)}
</CanAccess>
</TabbedForm.Tab>
<TabbedForm.Tab label="post.form.body">
<RichTextInput
Expand Down
25 changes: 16 additions & 9 deletions examples/simple/src/users/UserCreate.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -10,20 +10,21 @@ import {
Toolbar,
required,
useNotify,
usePermissions,
useUnique,
CanAccess,
useCanAccess,
} from 'react-admin';

import Aside from './Aside';

const UserCreateToolbar = ({ permissions, ...props }) => {
const UserCreateToolbar = () => {
const notify = useNotify();
const { reset } = useFormContext();

return (
<Toolbar {...props}>
<Toolbar>
<SaveButton label="user.action.save_and_show" />
{permissions === 'admin' && (
<CanAccess action="batch_create">
<SaveButton
label="user.action.save_and_add"
mutationOptions={{
Expand All @@ -40,7 +41,7 @@ const UserCreateToolbar = ({ permissions, ...props }) => {
type="button"
variant="text"
/>
)}
</CanAccess>
</Toolbar>
);
};
Expand All @@ -53,14 +54,20 @@ const isValidName = async value =>
);

const UserCreate = () => {
const { permissions } = usePermissions();
const unique = useUnique();
const { isPending, canAccess: canEditRole } = useCanAccess({
action: 'edit',
resource: 'users.role',
});
if (isPending) {
return null;
}
return (
<Create aside={<Aside />} redirect="show">
<TabbedForm
mode="onBlur"
warnWhenUnsavedChanges
toolbar={<UserCreateToolbar permissions={permissions} />}
toolbar={<UserCreateToolbar />}
>
<TabbedForm.Tab label="user.form.summary" path="">
<TextInput
Expand All @@ -70,7 +77,7 @@ const UserCreate = () => {
validate={[required(), isValidName, unique()]}
/>
</TabbedForm.Tab>
{permissions === 'admin' && (
{canEditRole ? (
<TabbedForm.Tab label="user.form.security" path="security">
<AutocompleteInput
source="role"
Expand All @@ -83,7 +90,7 @@ const UserCreate = () => {
validate={[required()]}
/>
</TabbedForm.Tab>
)}
) : null}
</TabbedForm>
</Create>
);
Expand Down
19 changes: 13 additions & 6 deletions examples/simple/src/users/UserEdit.tsx
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
/* eslint react/jsx-key: off */
import * as React from 'react';
import {
CanAccess,
CloneButton,
DeleteWithConfirmButton,
Edit,
Expand All @@ -12,7 +13,7 @@ import {
TextInput,
Toolbar,
TopToolbar,
usePermissions,
useCanAccess,
useSaveContext,
} from 'react-admin';

Expand Down Expand Up @@ -43,8 +44,14 @@ const EditActions = () => (
);

const UserEditForm = () => {
const { permissions } = usePermissions();
const { isPending, canAccess: canEditRole } = useCanAccess({
action: 'edit',
resource: 'users.role',
});
const { save } = useSaveContext();
if (isPending) {
return null;
}
if (!save) return null;

const newSave = values =>
Expand All @@ -67,16 +74,16 @@ const UserEditForm = () => {
onSubmit={newSave}
>
<TabbedForm.Tab label="user.form.summary" path="">
{permissions === 'admin' && (
<CanAccess action="edit" resource="users.id">
<TextInput source="id" InputProps={{ disabled: true }} />
)}
</CanAccess>
<TextInput
source="name"
defaultValue="slim shady"
validate={required()}
/>
</TabbedForm.Tab>
{permissions === 'admin' && (
{canEditRole ? (
<TabbedForm.Tab label="user.form.security" path="security">
<SelectInput
source="role"
Expand All @@ -89,7 +96,7 @@ const UserEditForm = () => {
defaultValue={'user'}
/>
</TabbedForm.Tab>
)}
) : null}
</TabbedForm>
);
};
Expand Down
Loading

0 comments on commit 4842169

Please sign in to comment.