1
0
mirror of https://github.com/Unleash/unleash.git synced 2024-10-18 20:09:08 +02:00
unleash.unleash/frontend/src/component/project/ProjectList/ProjectList.tsx

172 lines
5.9 KiB
TypeScript
Raw Normal View History

2022-02-28 17:00:12 +01:00
import { useContext, useMemo, useState } from 'react';
import { Link, useNavigate } from 'react-router-dom';
import { mutate } from 'swr';
2022-02-28 17:00:12 +01:00
import { getProjectFetcher } from 'hooks/api/getters/useProject/getProjectFetcher';
import useProjects from 'hooks/api/getters/useProjects/useProjects';
import { ConditionallyRender } from 'component/common/ConditionallyRender/ConditionallyRender';
2022-03-01 22:23:23 +01:00
import { ProjectCard } from '../ProjectCard/ProjectCard';
import { useStyles } from './ProjectList.styles';
2022-02-28 17:00:12 +01:00
import { IProjectCard } from 'interfaces/project';
import loadingData from './loadingData';
2022-02-28 17:00:12 +01:00
import useLoading from 'hooks/useLoading';
import { PageContent } from 'component/common/PageContent/PageContent';
2022-02-28 17:00:12 +01:00
import AccessContext from 'contexts/AccessContext';
import { PageHeader } from 'component/common/PageHeader/PageHeader';
2022-02-28 17:00:12 +01:00
import ResponsiveButton from 'component/common/ResponsiveButton/ResponsiveButton';
import { CREATE_PROJECT } from 'component/providers/AccessProvider/permissions';
import { Add } from '@mui/icons-material';
2022-02-28 17:00:12 +01:00
import ApiError from 'component/common/ApiError/ApiError';
import useUiConfig from 'hooks/api/getters/useUiConfig/useUiConfig';
import { SearchField } from 'component/common/SearchField/SearchField';
import classnames from 'classnames';
type projectMap = {
[index: string]: boolean;
};
function resolveCreateButtonData(isOss: boolean, hasAccess: boolean) {
if (isOss) {
return {
title: 'You must be on a paid subscription to create new projects',
disabled: true,
};
} else if (!hasAccess) {
return {
title: 'You do not have permission to create new projects',
disabled: true,
};
} else {
return {
title: 'Click to create a new project',
disabled: false,
};
}
}
2022-02-28 17:00:12 +01:00
export const ProjectListNew = () => {
const { hasAccess } = useContext(AccessContext);
const navigate = useNavigate();
const { classes: styles } = useStyles();
const { projects, loading, error, refetch } = useProjects();
const [fetchedProjects, setFetchedProjects] = useState<projectMap>({});
const ref = useLoading(loading);
const { isOss } = useUiConfig();
2022-02-28 17:00:12 +01:00
const [filter, setFilter] = useState('');
const filteredProjects = useMemo(() => {
const regExp = new RegExp(filter, 'i');
return filter
2022-03-01 09:50:29 +01:00
? projects.filter(project => regExp.test(project.name))
2022-02-28 17:00:12 +01:00
: projects;
}, [projects, filter]);
const handleHover = (projectId: string) => {
if (fetchedProjects[projectId]) {
return;
}
const { KEY, fetcher } = getProjectFetcher(projectId);
mutate(KEY, fetcher);
setFetchedProjects(prev => ({ ...prev, [projectId]: true }));
};
const createButtonData = resolveCreateButtonData(
isOss(),
hasAccess(CREATE_PROJECT)
);
const renderError = () => {
return (
<ApiError
onClick={refetch}
className={styles.apiError}
text="Error fetching projects"
/>
);
};
const renderProjects = () => {
if (loading) {
return renderLoading();
}
2022-03-01 09:50:29 +01:00
return filteredProjects.map((project: IProjectCard) => {
return (
<Link
key={project.id}
to={`/projects/${project.id}`}
className={styles.cardLink}
>
<ProjectCard
onHover={() => handleHover(project?.id)}
name={project?.name}
memberCount={project?.memberCount ?? 0}
health={project?.health}
id={project?.id}
featureCount={project?.featureCount}
/>
</Link>
);
});
};
const renderLoading = () => {
return loadingData.map((project: IProjectCard) => {
return (
<ProjectCard
data-loading
onHover={() => {}}
key={project.id}
name={project.name}
id={project.id}
memberCount={2}
health={95}
featureCount={4}
/>
);
});
};
return (
<div ref={ref}>
2022-02-28 17:00:12 +01:00
<div className={styles.searchBarContainer}>
<SearchField
initialValue={filter}
updateValue={setFilter}
showValueChip
className={classnames(styles.searchBar, {
skeleton: loading,
})}
/>
</div>
<PageContent
header={
<PageHeader
title="Projects"
actions={
<ResponsiveButton
Icon={Add}
onClick={() => navigate('/projects/create')}
maxWidth="700px"
permission={CREATE_PROJECT}
disabled={createButtonData.disabled}
>
2022-02-23 00:10:48 +01:00
New project
</ResponsiveButton>
}
/>
}
>
<ConditionallyRender condition={error} show={renderError()} />
<div className={styles.container}>
<ConditionallyRender
2022-03-01 09:50:29 +01:00
condition={filteredProjects.length < 1 && !loading}
show={<div>No projects available.</div>}
elseShow={renderProjects()}
/>
</div>
</PageContent>
</div>
);
};