2020-06-13 00:21:58 +02:00
|
|
|
import React, { useState, useContext, useEffect } from 'react';
|
2020-04-10 04:40:22 +02:00
|
|
|
import styled from 'styled-components/macro';
|
2020-05-27 02:53:31 +02:00
|
|
|
import GlobalTopNavbar from 'App/TopNavbar';
|
2020-07-05 01:02:57 +02:00
|
|
|
import Empty from 'shared/undraw/Empty';
|
2020-06-21 00:49:11 +02:00
|
|
|
import {
|
|
|
|
useCreateTeamMutation,
|
|
|
|
useGetProjectsQuery,
|
|
|
|
useCreateProjectMutation,
|
|
|
|
GetProjectsDocument,
|
2020-06-23 23:55:17 +02:00
|
|
|
GetProjectsQuery,
|
2020-06-21 00:49:11 +02:00
|
|
|
} from 'shared/generated/graphql';
|
2020-04-10 04:40:22 +02:00
|
|
|
|
2020-06-01 04:20:03 +02:00
|
|
|
import ProjectGridItem, { AddProjectItem } from 'shared/components/ProjectGridItem';
|
2020-04-10 04:40:22 +02:00
|
|
|
import { Link } from 'react-router-dom';
|
2020-06-01 04:20:03 +02:00
|
|
|
import NewProject from 'shared/components/NewProject';
|
2020-08-01 03:01:14 +02:00
|
|
|
import UserContext, { PermissionLevel, PermissionObjectType, useCurrentUser } from 'App/context';
|
2020-06-21 00:49:11 +02:00
|
|
|
import Button from 'shared/components/Button';
|
|
|
|
import { usePopup, Popup } from 'shared/components/PopupMenu';
|
|
|
|
import { useForm } from 'react-hook-form';
|
|
|
|
import Input from 'shared/components/Input';
|
2020-06-23 23:55:17 +02:00
|
|
|
import updateApolloCache from 'shared/utils/cache';
|
|
|
|
import produce from 'immer';
|
2020-07-05 01:02:57 +02:00
|
|
|
const EmptyStateContent = styled.div`
|
|
|
|
display: flex;
|
|
|
|
justy-content: center;
|
|
|
|
align-items: center;
|
|
|
|
flex-direction: column;
|
|
|
|
`;
|
2020-04-10 04:40:22 +02:00
|
|
|
|
2020-07-05 01:02:57 +02:00
|
|
|
const EmptyStateTitle = styled.h3`
|
|
|
|
color: #fff;
|
|
|
|
font-size: 18px;
|
|
|
|
`;
|
|
|
|
|
|
|
|
const EmptyStatePrompt = styled.span`
|
|
|
|
color: rgba(${props => props.theme.colors.text.primary});
|
|
|
|
font-size: 16px;
|
|
|
|
margin-top: 8px;
|
|
|
|
`;
|
|
|
|
const EmptyState = styled(Empty)`
|
|
|
|
display: block;
|
|
|
|
margin: 0 auto;
|
|
|
|
`;
|
2020-06-21 00:49:11 +02:00
|
|
|
const CreateTeamButton = styled(Button)`
|
|
|
|
width: 100%;
|
|
|
|
`;
|
|
|
|
type CreateTeamData = { teamName: string };
|
|
|
|
type CreateTeamFormProps = {
|
|
|
|
onCreateTeam: (teamName: string) => void;
|
|
|
|
};
|
|
|
|
const CreateTeamFormContainer = styled.form``;
|
|
|
|
|
|
|
|
const CreateTeamForm: React.FC<CreateTeamFormProps> = ({ onCreateTeam }) => {
|
|
|
|
const { register, handleSubmit, errors } = useForm<CreateTeamData>();
|
|
|
|
const createTeam = (data: CreateTeamData) => {
|
|
|
|
onCreateTeam(data.teamName);
|
|
|
|
};
|
|
|
|
return (
|
|
|
|
<CreateTeamFormContainer onSubmit={handleSubmit(createTeam)}>
|
|
|
|
<Input
|
|
|
|
width="100%"
|
|
|
|
label="Team name"
|
|
|
|
id="teamName"
|
|
|
|
name="teamName"
|
|
|
|
variant="alternate"
|
|
|
|
ref={register({ required: 'Team name is required' })}
|
|
|
|
/>
|
|
|
|
<CreateTeamButton type="submit">Create</CreateTeamButton>
|
|
|
|
</CreateTeamFormContainer>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
const ProjectAddTile = styled.div`
|
|
|
|
background-color: rgba(${props => props.theme.colors.bg.primary}, 0.4);
|
|
|
|
background-size: cover;
|
|
|
|
background-position: 50%;
|
|
|
|
color: #fff;
|
|
|
|
line-height: 20px;
|
|
|
|
padding: 8px;
|
|
|
|
position: relative;
|
|
|
|
text-decoration: none;
|
|
|
|
|
|
|
|
border-radius: 3px;
|
|
|
|
display: block;
|
|
|
|
`;
|
|
|
|
|
|
|
|
const ProjectTile = styled(Link)<{ color: string }>`
|
|
|
|
background-color: ${props => props.color};
|
|
|
|
background-size: cover;
|
|
|
|
background-position: 50%;
|
|
|
|
color: #fff;
|
|
|
|
line-height: 20px;
|
|
|
|
padding: 8px;
|
|
|
|
position: relative;
|
|
|
|
text-decoration: none;
|
|
|
|
|
|
|
|
border-radius: 3px;
|
|
|
|
display: block;
|
|
|
|
`;
|
|
|
|
|
|
|
|
const ProjectTileFade = styled.div`
|
|
|
|
background-color: rgba(0, 0, 0, 0.15);
|
|
|
|
bottom: 0;
|
|
|
|
left: 0;
|
|
|
|
position: absolute;
|
|
|
|
right: 0;
|
|
|
|
top: 0;
|
|
|
|
`;
|
|
|
|
|
|
|
|
const ProjectListItem = styled.li`
|
|
|
|
width: 23.5%;
|
|
|
|
padding: 0;
|
|
|
|
margin: 0 2% 2% 0;
|
|
|
|
|
|
|
|
box-sizing: border-box;
|
|
|
|
position: relative;
|
|
|
|
cursor: pointer;
|
|
|
|
|
|
|
|
&:hover ${ProjectTileFade} {
|
|
|
|
background-color: rgba(0, 0, 0, 0.25);
|
|
|
|
}
|
|
|
|
`;
|
|
|
|
|
|
|
|
const ProjectList = styled.ul`
|
|
|
|
display: flex;
|
|
|
|
flex-wrap: wrap;
|
|
|
|
|
|
|
|
& ${ProjectListItem}:nth-of-type(4n) {
|
|
|
|
margin-right: 0;
|
|
|
|
}
|
|
|
|
`;
|
|
|
|
|
|
|
|
const ProjectTileDetails = styled.div`
|
|
|
|
display: flex;
|
|
|
|
height: 80px;
|
|
|
|
position: relative;
|
|
|
|
flex-direction: column;
|
|
|
|
justify-content: space-between;
|
|
|
|
`;
|
|
|
|
|
|
|
|
const ProjectAddTileDetails = styled.div`
|
|
|
|
display: flex;
|
|
|
|
height: 80px;
|
|
|
|
position: relative;
|
|
|
|
flex-direction: column;
|
|
|
|
align-items: center;
|
|
|
|
justify-content: center;
|
|
|
|
`;
|
|
|
|
|
|
|
|
const ProjectTileName = styled.div<{ centered?: boolean }>`
|
|
|
|
flex: 0 0 auto;
|
|
|
|
font-size: 16px;
|
|
|
|
font-weight: 700;
|
|
|
|
display: inline-block;
|
|
|
|
overflow: hidden;
|
|
|
|
max-height: 40px;
|
|
|
|
width: 100%;
|
|
|
|
word-wrap: break-word;
|
|
|
|
${props => props.centered && 'text-align: center;'}
|
|
|
|
`;
|
|
|
|
|
|
|
|
const Wrapper = styled.div`
|
2020-06-23 22:20:53 +02:00
|
|
|
position: relative;
|
2020-06-21 00:49:11 +02:00
|
|
|
display: flex;
|
|
|
|
flex-direction: row;
|
|
|
|
align-items: flex-start;
|
|
|
|
justify-content: center;
|
|
|
|
`;
|
|
|
|
|
|
|
|
const ProjectSectionTitleWrapper = styled.div`
|
|
|
|
align-items: center;
|
|
|
|
display: flex;
|
2020-06-23 22:20:53 +02:00
|
|
|
justify-content: space-between;
|
2020-06-21 00:49:11 +02:00
|
|
|
height: 32px;
|
|
|
|
margin-bottom: 24px;
|
|
|
|
padding: 8px 0;
|
|
|
|
position: relative;
|
2020-06-23 22:20:53 +02:00
|
|
|
margin-top: 16px;
|
|
|
|
`;
|
|
|
|
|
|
|
|
const SectionActions = styled.div`
|
|
|
|
display: flex;
|
|
|
|
align-items: center;
|
|
|
|
`;
|
|
|
|
|
|
|
|
const SectionAction = styled(Button)`
|
|
|
|
padding: 6px 12px;
|
|
|
|
`;
|
|
|
|
const SectionActionLink = styled(Link)`
|
|
|
|
margin-right: 8px;
|
2020-06-21 00:49:11 +02:00
|
|
|
`;
|
|
|
|
|
|
|
|
const ProjectSectionTitle = styled.h3`
|
|
|
|
font-size: 16px;
|
|
|
|
color: rgba(${props => props.theme.colors.text.primary});
|
2020-04-10 04:40:22 +02:00
|
|
|
`;
|
|
|
|
|
2020-06-21 00:49:11 +02:00
|
|
|
const ProjectsContainer = styled.div`
|
|
|
|
margin: 40px 16px 0;
|
|
|
|
width: 100%;
|
|
|
|
max-width: 825px;
|
|
|
|
min-width: 288px;
|
|
|
|
`;
|
2020-04-10 04:40:22 +02:00
|
|
|
const ProjectGrid = styled.div`
|
2020-04-20 05:02:55 +02:00
|
|
|
max-width: 780px;
|
2020-06-01 04:20:03 +02:00
|
|
|
display: grid;
|
|
|
|
grid-template-columns: 240px 240px 240px;
|
|
|
|
gap: 20px 10px;
|
2020-04-10 04:40:22 +02:00
|
|
|
`;
|
2020-06-21 00:49:11 +02:00
|
|
|
const AddTeamButton = styled(Button)`
|
|
|
|
padding: 6px 12px;
|
2020-06-23 22:20:53 +02:00
|
|
|
position: absolute;
|
|
|
|
top: 6px;
|
|
|
|
right: 12px;
|
2020-06-21 00:49:11 +02:00
|
|
|
`;
|
2020-07-05 01:02:57 +02:00
|
|
|
|
|
|
|
const CreateFirstTeam = styled(Button)`
|
|
|
|
margin-top: 8px;
|
|
|
|
`;
|
2020-06-23 22:20:53 +02:00
|
|
|
type ShowNewProject = {
|
|
|
|
open: boolean;
|
|
|
|
initialTeamID: null | string;
|
|
|
|
};
|
2020-04-20 05:02:55 +02:00
|
|
|
|
2020-06-01 04:20:03 +02:00
|
|
|
const ProjectLink = styled(Link)``;
|
2020-04-20 05:02:55 +02:00
|
|
|
|
2020-04-10 04:40:22 +02:00
|
|
|
const Projects = () => {
|
2020-06-23 22:20:53 +02:00
|
|
|
const { showPopup, hidePopup } = usePopup();
|
2020-08-01 03:01:14 +02:00
|
|
|
const { loading, data } = useGetProjectsQuery({ fetchPolicy: 'network-only' });
|
2020-06-13 00:21:58 +02:00
|
|
|
useEffect(() => {
|
2020-08-07 03:50:35 +02:00
|
|
|
document.title = 'Taskcafé';
|
2020-06-13 00:21:58 +02:00
|
|
|
}, []);
|
2020-06-01 04:20:03 +02:00
|
|
|
const [createProject] = useCreateProjectMutation({
|
|
|
|
update: (client, newProject) => {
|
2020-06-23 23:55:17 +02:00
|
|
|
updateApolloCache<GetProjectsQuery>(client, GetProjectsDocument, cache =>
|
|
|
|
produce(cache, draftCache => {
|
|
|
|
draftCache.projects.push({ ...newProject.data.createProject });
|
|
|
|
}),
|
|
|
|
);
|
2020-06-01 04:20:03 +02:00
|
|
|
},
|
|
|
|
});
|
2020-06-23 22:20:53 +02:00
|
|
|
|
|
|
|
const [showNewProject, setShowNewProject] = useState<ShowNewProject>({ open: false, initialTeamID: null });
|
2020-08-01 03:01:14 +02:00
|
|
|
const { user, setUser } = useCurrentUser();
|
2020-06-23 22:20:53 +02:00
|
|
|
const [createTeam] = useCreateTeamMutation({
|
|
|
|
update: (client, createData) => {
|
2020-06-23 23:55:17 +02:00
|
|
|
updateApolloCache<GetProjectsQuery>(client, GetProjectsDocument, cache =>
|
|
|
|
produce(cache, draftCache => {
|
|
|
|
draftCache.teams.push({ ...createData.data.createTeam });
|
|
|
|
}),
|
|
|
|
);
|
2020-06-23 22:20:53 +02:00
|
|
|
},
|
|
|
|
});
|
2020-04-10 04:40:22 +02:00
|
|
|
if (loading) {
|
2020-04-10 18:31:29 +02:00
|
|
|
return (
|
|
|
|
<>
|
2020-04-20 05:02:55 +02:00
|
|
|
<span>loading</span>
|
2020-04-10 18:31:29 +02:00
|
|
|
</>
|
|
|
|
);
|
2020-04-10 04:40:22 +02:00
|
|
|
}
|
2020-06-21 00:49:11 +02:00
|
|
|
|
|
|
|
const colors = ['#e362e3', '#7a6ff0', '#37c5ab', '#aa62e3', '#e8384f'];
|
2020-08-01 03:01:14 +02:00
|
|
|
if (data && user) {
|
|
|
|
console.log(user);
|
2020-06-21 00:49:11 +02:00
|
|
|
const { projects, teams, organizations } = data;
|
|
|
|
const organizationID = organizations[0].id ?? null;
|
2020-08-01 03:01:14 +02:00
|
|
|
const projectTeams = teams
|
|
|
|
.sort((a, b) => {
|
|
|
|
const textA = a.name.toUpperCase();
|
|
|
|
const textB = b.name.toUpperCase();
|
|
|
|
return textA < textB ? -1 : textA > textB ? 1 : 0;
|
|
|
|
})
|
|
|
|
.map(team => {
|
|
|
|
return {
|
|
|
|
id: team.id,
|
|
|
|
name: team.name,
|
|
|
|
projects: projects
|
|
|
|
.filter(project => project.team.id === team.id)
|
|
|
|
.sort((a, b) => {
|
|
|
|
const textA = a.name.toUpperCase();
|
|
|
|
const textB = b.name.toUpperCase();
|
|
|
|
return textA < textB ? -1 : textA > textB ? 1 : 0;
|
|
|
|
}),
|
|
|
|
};
|
|
|
|
});
|
|
|
|
console.log(projectTeams);
|
2020-04-10 04:40:22 +02:00
|
|
|
return (
|
2020-05-27 02:53:31 +02:00
|
|
|
<>
|
2020-06-21 00:49:11 +02:00
|
|
|
<GlobalTopNavbar onSaveProjectName={() => {}} projectID={null} name={null} />
|
|
|
|
<Wrapper>
|
|
|
|
<ProjectsContainer>
|
2020-08-01 03:01:14 +02:00
|
|
|
{user.roles.org === 'admin' && (
|
|
|
|
<AddTeamButton
|
|
|
|
variant="outline"
|
|
|
|
onClick={$target => {
|
|
|
|
showPopup(
|
|
|
|
$target,
|
|
|
|
<Popup
|
|
|
|
title="Create team"
|
|
|
|
tab={0}
|
|
|
|
onClose={() => {
|
|
|
|
hidePopup();
|
2020-06-21 00:49:11 +02:00
|
|
|
}}
|
2020-08-01 03:01:14 +02:00
|
|
|
>
|
|
|
|
<CreateTeamForm
|
|
|
|
onCreateTeam={teamName => {
|
|
|
|
if (organizationID) {
|
|
|
|
createTeam({ variables: { name: teamName, organizationID } });
|
|
|
|
hidePopup();
|
|
|
|
}
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
</Popup>,
|
|
|
|
);
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
Add Team
|
|
|
|
</AddTeamButton>
|
|
|
|
)}
|
2020-07-05 01:02:57 +02:00
|
|
|
{projectTeams.length === 0 && (
|
|
|
|
<EmptyStateContent>
|
|
|
|
<EmptyState width={425} height={425} />
|
|
|
|
<EmptyStateTitle>No teams exist</EmptyStateTitle>
|
|
|
|
<EmptyStatePrompt>Create a new team to get started</EmptyStatePrompt>
|
|
|
|
<CreateFirstTeam
|
|
|
|
variant="outline"
|
|
|
|
onClick={$target => {
|
|
|
|
showPopup(
|
|
|
|
$target,
|
|
|
|
<Popup
|
|
|
|
title="Create team"
|
|
|
|
tab={0}
|
|
|
|
onClose={() => {
|
|
|
|
hidePopup();
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<CreateTeamForm
|
|
|
|
onCreateTeam={teamName => {
|
|
|
|
if (organizationID) {
|
|
|
|
createTeam({ variables: { name: teamName, organizationID } });
|
|
|
|
hidePopup();
|
|
|
|
}
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
</Popup>,
|
|
|
|
);
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
Create new team
|
|
|
|
</CreateFirstTeam>
|
|
|
|
</EmptyStateContent>
|
|
|
|
)}
|
2020-06-21 00:49:11 +02:00
|
|
|
{projectTeams.map(team => {
|
|
|
|
return (
|
|
|
|
<div key={team.id}>
|
|
|
|
<ProjectSectionTitleWrapper>
|
|
|
|
<ProjectSectionTitle>{team.name}</ProjectSectionTitle>
|
2020-08-01 03:01:14 +02:00
|
|
|
{user.isAdmin(PermissionLevel.TEAM, PermissionObjectType.TEAM, team.id) && (
|
|
|
|
<SectionActions>
|
|
|
|
<SectionActionLink to={`/teams/${team.id}`}>
|
|
|
|
<SectionAction variant="outline">Projects</SectionAction>
|
|
|
|
</SectionActionLink>
|
|
|
|
<SectionActionLink to={`/teams/${team.id}/members`}>
|
|
|
|
<SectionAction variant="outline">Members</SectionAction>
|
|
|
|
</SectionActionLink>
|
|
|
|
<SectionActionLink to={`/teams/${team.id}/settings`}>
|
|
|
|
<SectionAction variant="outline">Settings</SectionAction>
|
|
|
|
</SectionActionLink>
|
|
|
|
</SectionActions>
|
|
|
|
)}
|
2020-06-21 00:49:11 +02:00
|
|
|
</ProjectSectionTitleWrapper>
|
|
|
|
<ProjectList>
|
|
|
|
{team.projects.map((project, idx) => (
|
|
|
|
<ProjectListItem key={project.id}>
|
|
|
|
<ProjectTile color={colors[idx % 5]} to={`/projects/${project.id}`}>
|
|
|
|
<ProjectTileFade />
|
|
|
|
<ProjectTileDetails>
|
|
|
|
<ProjectTileName>{project.name}</ProjectTileName>
|
|
|
|
</ProjectTileDetails>
|
|
|
|
</ProjectTile>
|
|
|
|
</ProjectListItem>
|
|
|
|
))}
|
2020-08-01 03:01:14 +02:00
|
|
|
{user.isAdmin(PermissionLevel.TEAM, PermissionObjectType.TEAM, team.id) && (
|
|
|
|
<ProjectListItem>
|
|
|
|
<ProjectAddTile
|
|
|
|
onClick={() => {
|
|
|
|
setShowNewProject({ open: true, initialTeamID: team.id });
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<ProjectTileFade />
|
|
|
|
<ProjectAddTileDetails>
|
|
|
|
<ProjectTileName centered>Create new project</ProjectTileName>
|
|
|
|
</ProjectAddTileDetails>
|
|
|
|
</ProjectAddTile>
|
|
|
|
</ProjectListItem>
|
|
|
|
)}
|
2020-06-21 00:49:11 +02:00
|
|
|
</ProjectList>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
})}
|
2020-06-23 22:20:53 +02:00
|
|
|
{showNewProject.open && (
|
2020-06-21 00:49:11 +02:00
|
|
|
<NewProject
|
2020-06-23 22:20:53 +02:00
|
|
|
initialTeamID={showNewProject.initialTeamID}
|
2020-06-21 00:49:11 +02:00
|
|
|
onCreateProject={(name, teamID) => {
|
2020-08-01 03:01:14 +02:00
|
|
|
if (user) {
|
|
|
|
createProject({ variables: { teamID, name, userID: user.id } });
|
2020-06-23 22:20:53 +02:00
|
|
|
setShowNewProject({ open: false, initialTeamID: null });
|
2020-06-21 00:49:11 +02:00
|
|
|
}
|
|
|
|
}}
|
|
|
|
onClose={() => {
|
2020-06-23 22:20:53 +02:00
|
|
|
setShowNewProject({ open: false, initialTeamID: null });
|
2020-06-21 00:49:11 +02:00
|
|
|
}}
|
|
|
|
teams={teams}
|
2020-05-27 02:53:31 +02:00
|
|
|
/>
|
2020-06-21 00:49:11 +02:00
|
|
|
)}
|
|
|
|
</ProjectsContainer>
|
|
|
|
</Wrapper>
|
2020-05-27 02:53:31 +02:00
|
|
|
</>
|
2020-04-10 04:40:22 +02:00
|
|
|
);
|
|
|
|
}
|
2020-04-20 05:02:55 +02:00
|
|
|
return <div>Error!</div>;
|
2020-04-10 04:40:22 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
export default Projects;
|