2023-08-10 09:28:10 +02:00
|
|
|
import { useState } from 'react';
|
|
|
|
import { ConditionallyRender } from 'component/common/ConditionallyRender/ConditionallyRender';
|
|
|
|
import { ADMIN } from 'component/providers/AccessProvider/permissions';
|
|
|
|
import { RolesTable } from './RolesTable/RolesTable';
|
|
|
|
import { PageContent } from 'component/common/PageContent/PageContent';
|
|
|
|
import { Tab, Tabs, styled, useMediaQuery } from '@mui/material';
|
|
|
|
import { Route, Routes, useLocation } from 'react-router-dom';
|
|
|
|
import { PROJECT_ROLE_TYPE, ROOT_ROLE_TYPE } from '@server/util/constants';
|
|
|
|
import { useRoles } from 'hooks/api/getters/useRoles/useRoles';
|
|
|
|
import { Search } from 'component/common/Search/Search';
|
|
|
|
import theme from 'themes/theme';
|
|
|
|
import { PageHeader } from 'component/common/PageHeader/PageHeader';
|
2024-03-12 10:56:10 +01:00
|
|
|
import Add from '@mui/icons-material/Add';
|
2023-08-10 09:28:10 +02:00
|
|
|
import ResponsiveButton from 'component/common/ResponsiveButton/ResponsiveButton';
|
2024-03-18 13:58:05 +01:00
|
|
|
import type { IRole } from 'interfaces/role';
|
2024-01-10 11:33:51 +01:00
|
|
|
import { TabLink } from 'component/common/TabNav/TabLink';
|
2023-08-10 09:28:10 +02:00
|
|
|
|
|
|
|
const StyledHeader = styled('div')(() => ({
|
|
|
|
display: 'flex',
|
|
|
|
justifyContent: 'space-between',
|
|
|
|
alignItems: 'center',
|
|
|
|
}));
|
|
|
|
|
|
|
|
const StyledTabsContainer = styled('div')({
|
|
|
|
flex: 1,
|
|
|
|
});
|
|
|
|
|
|
|
|
const StyledActions = styled('div')({
|
|
|
|
display: 'flex',
|
|
|
|
alignItems: 'center',
|
|
|
|
});
|
|
|
|
|
|
|
|
export const RolesPage = () => {
|
|
|
|
const { pathname } = useLocation();
|
|
|
|
|
|
|
|
const { roles, projectRoles, loading } = useRoles();
|
|
|
|
|
|
|
|
const [searchValue, setSearchValue] = useState('');
|
|
|
|
const [modalOpen, setModalOpen] = useState(false);
|
|
|
|
const [selectedRole, setSelectedRole] = useState<IRole>();
|
|
|
|
|
2024-02-09 09:41:40 +01:00
|
|
|
const tabs = [
|
|
|
|
{
|
|
|
|
label: 'Root roles',
|
|
|
|
path: '/admin/roles',
|
|
|
|
total: roles.length,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
label: 'Project roles',
|
|
|
|
path: '/admin/roles/project-roles',
|
|
|
|
total: projectRoles.length,
|
|
|
|
},
|
|
|
|
];
|
2023-08-10 09:28:10 +02:00
|
|
|
|
|
|
|
const isSmallScreen = useMediaQuery(theme.breakpoints.down('md'));
|
|
|
|
|
2024-02-09 09:41:40 +01:00
|
|
|
const type = pathname.includes('project-roles')
|
|
|
|
? PROJECT_ROLE_TYPE
|
|
|
|
: ROOT_ROLE_TYPE;
|
2023-08-10 09:28:10 +02:00
|
|
|
|
|
|
|
return (
|
|
|
|
<PageContent
|
|
|
|
withTabs
|
2023-10-02 14:25:46 +02:00
|
|
|
bodyClass='page-body'
|
2023-08-10 09:28:10 +02:00
|
|
|
isLoading={loading}
|
|
|
|
header={
|
|
|
|
<>
|
|
|
|
<StyledHeader>
|
|
|
|
<StyledTabsContainer>
|
|
|
|
<Tabs
|
|
|
|
value={pathname}
|
2023-10-02 14:25:46 +02:00
|
|
|
indicatorColor='primary'
|
|
|
|
textColor='primary'
|
|
|
|
variant='scrollable'
|
2023-08-10 09:28:10 +02:00
|
|
|
allowScrollButtonsMobile
|
|
|
|
>
|
|
|
|
{tabs.map(({ label, path, total }) => (
|
|
|
|
<Tab
|
|
|
|
key={label}
|
|
|
|
value={path}
|
|
|
|
label={
|
2024-01-10 11:33:51 +01:00
|
|
|
<TabLink to={path}>
|
|
|
|
{label} ({total})
|
|
|
|
</TabLink>
|
2023-08-10 09:28:10 +02:00
|
|
|
}
|
2023-08-18 10:40:35 +02:00
|
|
|
sx={{ padding: 0 }}
|
2023-08-10 09:28:10 +02:00
|
|
|
/>
|
|
|
|
))}
|
|
|
|
</Tabs>
|
|
|
|
</StyledTabsContainer>
|
|
|
|
<StyledActions>
|
|
|
|
<ConditionallyRender
|
|
|
|
condition={!isSmallScreen}
|
|
|
|
show={
|
|
|
|
<>
|
|
|
|
<Search
|
|
|
|
initialValue={searchValue}
|
|
|
|
onChange={setSearchValue}
|
|
|
|
/>
|
|
|
|
<PageHeader.Divider />
|
|
|
|
</>
|
|
|
|
}
|
|
|
|
/>
|
|
|
|
<ResponsiveButton
|
|
|
|
onClick={() => {
|
|
|
|
setSelectedRole(undefined);
|
|
|
|
setModalOpen(true);
|
|
|
|
}}
|
2023-10-02 14:25:46 +02:00
|
|
|
maxWidth={`${theme.breakpoints.values.sm}px`}
|
2023-08-10 09:28:10 +02:00
|
|
|
Icon={Add}
|
|
|
|
permission={ADMIN}
|
|
|
|
>
|
|
|
|
New {type} role
|
|
|
|
</ResponsiveButton>
|
|
|
|
</StyledActions>
|
|
|
|
</StyledHeader>
|
|
|
|
<ConditionallyRender
|
|
|
|
condition={isSmallScreen}
|
|
|
|
show={
|
|
|
|
<Search
|
|
|
|
initialValue={searchValue}
|
|
|
|
onChange={setSearchValue}
|
|
|
|
/>
|
|
|
|
}
|
|
|
|
/>
|
|
|
|
</>
|
|
|
|
}
|
|
|
|
>
|
|
|
|
<Routes>
|
|
|
|
<Route
|
2023-10-02 14:25:46 +02:00
|
|
|
path='project-roles'
|
2023-08-10 09:28:10 +02:00
|
|
|
element={
|
|
|
|
<RolesTable
|
|
|
|
type={PROJECT_ROLE_TYPE}
|
|
|
|
searchValue={searchValue}
|
|
|
|
modalOpen={modalOpen}
|
|
|
|
setModalOpen={setModalOpen}
|
|
|
|
selectedRole={selectedRole}
|
|
|
|
setSelectedRole={setSelectedRole}
|
|
|
|
/>
|
|
|
|
}
|
|
|
|
/>
|
|
|
|
<Route
|
2023-10-02 14:25:46 +02:00
|
|
|
path='*'
|
2023-08-10 09:28:10 +02:00
|
|
|
element={
|
|
|
|
<RolesTable
|
2024-02-09 09:41:40 +01:00
|
|
|
type={ROOT_ROLE_TYPE}
|
2023-08-10 09:28:10 +02:00
|
|
|
searchValue={searchValue}
|
|
|
|
modalOpen={modalOpen}
|
|
|
|
setModalOpen={setModalOpen}
|
|
|
|
selectedRole={selectedRole}
|
|
|
|
setSelectedRole={setSelectedRole}
|
|
|
|
/>
|
|
|
|
}
|
|
|
|
/>
|
|
|
|
</Routes>
|
|
|
|
</PageContent>
|
|
|
|
);
|
|
|
|
};
|