refactor(Project): split out components into their own files
This commit is contained in:
parent
bd34f4b3ad
commit
3e72271d9b
@ -0,0 +1,16 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import { Cross } from 'shared/icons';
|
||||||
|
import * as S from './Styles';
|
||||||
|
|
||||||
|
const OptionValue = ({ data, removeProps }: any) => {
|
||||||
|
return (
|
||||||
|
<S.OptionValueWrapper>
|
||||||
|
<S.OptionValueLabel>{data.label}</S.OptionValueLabel>
|
||||||
|
<S.OptionValueRemove {...removeProps}>
|
||||||
|
<Cross width={14} height={14} />
|
||||||
|
</S.OptionValueRemove>
|
||||||
|
</S.OptionValueWrapper>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default OptionValue;
|
64
frontend/src/Projects/Project/UserManagementPopup/Styles.ts
Normal file
64
frontend/src/Projects/Project/UserManagementPopup/Styles.ts
Normal file
@ -0,0 +1,64 @@
|
|||||||
|
import styled from 'styled-components';
|
||||||
|
import Button from 'shared/components/Button';
|
||||||
|
|
||||||
|
export const OptionWrapper = styled.div<{ isFocused: boolean }>`
|
||||||
|
cursor: pointer;
|
||||||
|
padding: 4px 8px;
|
||||||
|
${props => props.isFocused && `background: rgba(${props.theme.colors.primary});`}
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
`;
|
||||||
|
|
||||||
|
export const OptionContent = styled.div`
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
margin-left: 12px;
|
||||||
|
`;
|
||||||
|
|
||||||
|
export const OptionLabel = styled.span<{ fontSize: number; quiet: boolean }>`
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
font-size: ${p => p.fontSize}px;
|
||||||
|
color: rgba(${p => (p.quiet ? p.theme.colors.text.primary : p.theme.colors.text.primary)});
|
||||||
|
`;
|
||||||
|
|
||||||
|
export const OptionValueWrapper = styled.div`
|
||||||
|
background: rgba(${props => props.theme.colors.bg.primary});
|
||||||
|
border-radius: 4px;
|
||||||
|
margin: 2px;
|
||||||
|
padding: 3px 6px 3px 4px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
`;
|
||||||
|
|
||||||
|
export const OptionValueLabel = styled.span`
|
||||||
|
font-size: 12px;
|
||||||
|
color: rgba(${props => props.theme.colors.text.secondary});
|
||||||
|
`;
|
||||||
|
|
||||||
|
export const OptionValueRemove = styled.button`
|
||||||
|
cursor: pointer;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
background: none;
|
||||||
|
border: none;
|
||||||
|
outline: none;
|
||||||
|
padding: 0;
|
||||||
|
margin: 0;
|
||||||
|
margin-left: 4px;
|
||||||
|
`;
|
||||||
|
|
||||||
|
export const InviteButton = styled(Button)`
|
||||||
|
margin-top: 12px;
|
||||||
|
height: 32px;
|
||||||
|
padding: 4px 12px;
|
||||||
|
width: 100%;
|
||||||
|
justify-content: center;
|
||||||
|
`;
|
||||||
|
|
||||||
|
export const InviteContainer = styled.div`
|
||||||
|
min-height: 300px;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
`;
|
@ -0,0 +1,39 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import TaskAssignee from 'shared/components/TaskAssignee';
|
||||||
|
import * as S from './Styles';
|
||||||
|
|
||||||
|
type UserOptionProps = {
|
||||||
|
innerProps: any;
|
||||||
|
isDisabled: boolean;
|
||||||
|
isFocused: boolean;
|
||||||
|
label: string;
|
||||||
|
data: any;
|
||||||
|
getValue: any;
|
||||||
|
};
|
||||||
|
|
||||||
|
const UserOption: React.FC<UserOptionProps> = ({ isDisabled, isFocused, innerProps, label, data }) => {
|
||||||
|
return !isDisabled ? (
|
||||||
|
<S.OptionWrapper {...innerProps} isFocused={isFocused}>
|
||||||
|
<TaskAssignee
|
||||||
|
size={32}
|
||||||
|
member={{
|
||||||
|
id: '',
|
||||||
|
fullName: data.value.label,
|
||||||
|
profileIcon: data.value.profileIcon,
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<S.OptionContent>
|
||||||
|
<S.OptionLabel fontSize={16} quiet={false}>
|
||||||
|
{label}
|
||||||
|
</S.OptionLabel>
|
||||||
|
{data.value.type === 2 && (
|
||||||
|
<S.OptionLabel fontSize={14} quiet>
|
||||||
|
Joined
|
||||||
|
</S.OptionLabel>
|
||||||
|
)}
|
||||||
|
</S.OptionContent>
|
||||||
|
</S.OptionWrapper>
|
||||||
|
) : null;
|
||||||
|
};
|
||||||
|
|
||||||
|
export default UserOption;
|
@ -0,0 +1,82 @@
|
|||||||
|
import gql from 'graphql-tag';
|
||||||
|
import isValidEmail from 'shared/utils/email';
|
||||||
|
|
||||||
|
type MemberFilterOptions = {
|
||||||
|
projectID?: null | string;
|
||||||
|
teamID?: null | string;
|
||||||
|
organization?: boolean;
|
||||||
|
};
|
||||||
|
|
||||||
|
export default async function(client: any, projectID: string, options: MemberFilterOptions, input: string, cb: any) {
|
||||||
|
if (input && input.trim().length < 3) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
const res = await client.query({
|
||||||
|
query: gql`
|
||||||
|
query {
|
||||||
|
searchMembers(input: {searchFilter:"${input}", projectID:"${projectID}"}) {
|
||||||
|
id
|
||||||
|
similarity
|
||||||
|
status
|
||||||
|
user {
|
||||||
|
id
|
||||||
|
fullName
|
||||||
|
email
|
||||||
|
profileIcon {
|
||||||
|
url
|
||||||
|
initials
|
||||||
|
bgColor
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
`,
|
||||||
|
});
|
||||||
|
|
||||||
|
let results: any = [];
|
||||||
|
const emails: Array<string> = [];
|
||||||
|
if (res.data && res.data.searchMembers) {
|
||||||
|
results = [
|
||||||
|
...res.data.searchMembers.map((m: any) => {
|
||||||
|
if (m.status === 'INVITED') {
|
||||||
|
return {
|
||||||
|
label: m.id,
|
||||||
|
value: {
|
||||||
|
id: m.id,
|
||||||
|
type: 2,
|
||||||
|
profileIcon: {
|
||||||
|
bgColor: '#ccc',
|
||||||
|
initials: m.id.charAt(0),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
emails.push(m.user.email);
|
||||||
|
return {
|
||||||
|
label: m.user.fullName,
|
||||||
|
value: { id: m.user.id, type: 0, profileIcon: m.user.profileIcon },
|
||||||
|
};
|
||||||
|
}),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isValidEmail(input) && !emails.find(e => e === input)) {
|
||||||
|
results = [
|
||||||
|
...results,
|
||||||
|
{
|
||||||
|
label: input,
|
||||||
|
value: {
|
||||||
|
id: input,
|
||||||
|
type: 1,
|
||||||
|
profileIcon: {
|
||||||
|
bgColor: '#ccc',
|
||||||
|
initials: input.charAt(0),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
return results;
|
||||||
|
}
|
82
frontend/src/Projects/Project/UserManagementPopup/index.tsx
Normal file
82
frontend/src/Projects/Project/UserManagementPopup/index.tsx
Normal file
@ -0,0 +1,82 @@
|
|||||||
|
import React, { useState } from 'react';
|
||||||
|
import AsyncSelect from 'react-select/async';
|
||||||
|
import { useApolloClient } from '@apollo/react-hooks';
|
||||||
|
import { colourStyles } from 'shared/components/Select';
|
||||||
|
import { Popup } from 'shared/components/PopupMenu';
|
||||||
|
import OptionValue from './OptionValue';
|
||||||
|
import UserOption from './UserOption';
|
||||||
|
import fetchMembers from './fetchMembers';
|
||||||
|
import * as S from './Styles';
|
||||||
|
|
||||||
|
type InviteUserData = {
|
||||||
|
email?: string;
|
||||||
|
userID?: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
type UserManagementPopupProps = {
|
||||||
|
projectID: string;
|
||||||
|
users: Array<User>;
|
||||||
|
projectMembers: Array<TaskUser>;
|
||||||
|
onInviteProjectMembers: (data: Array<InviteUserData>) => void;
|
||||||
|
};
|
||||||
|
|
||||||
|
const UserManagementPopup: React.FC<UserManagementPopupProps> = ({
|
||||||
|
projectID,
|
||||||
|
users,
|
||||||
|
projectMembers,
|
||||||
|
onInviteProjectMembers,
|
||||||
|
}) => {
|
||||||
|
const client = useApolloClient();
|
||||||
|
const [invitedUsers, setInvitedUsers] = useState<Array<any> | null>(null);
|
||||||
|
return (
|
||||||
|
<Popup tab={0} title="Invite a user">
|
||||||
|
<S.InviteContainer>
|
||||||
|
<AsyncSelect
|
||||||
|
getOptionValue={option => option.value.id}
|
||||||
|
placeholder="Email address or username"
|
||||||
|
noOptionsMessage={() => null}
|
||||||
|
onChange={(e: any) => {
|
||||||
|
setInvitedUsers(e);
|
||||||
|
}}
|
||||||
|
isMulti
|
||||||
|
autoFocus
|
||||||
|
cacheOptions
|
||||||
|
styles={colourStyles}
|
||||||
|
defaultOption
|
||||||
|
components={{
|
||||||
|
MultiValue: OptionValue,
|
||||||
|
Option: UserOption,
|
||||||
|
IndicatorSeparator: null,
|
||||||
|
DropdownIndicator: null,
|
||||||
|
}}
|
||||||
|
loadOptions={(i, cb) => fetchMembers(client, projectID, {}, i, cb)}
|
||||||
|
/>
|
||||||
|
</S.InviteContainer>
|
||||||
|
<S.InviteButton
|
||||||
|
onClick={() => {
|
||||||
|
if (invitedUsers) {
|
||||||
|
onInviteProjectMembers(
|
||||||
|
invitedUsers.map(user => {
|
||||||
|
if (user.value.type === 0) {
|
||||||
|
return {
|
||||||
|
userID: user.value.id,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
return {
|
||||||
|
email: user.value.id,
|
||||||
|
};
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
disabled={invitedUsers === null}
|
||||||
|
hoverVariant="none"
|
||||||
|
fontSize="16px"
|
||||||
|
>
|
||||||
|
Send Invite
|
||||||
|
</S.InviteButton>
|
||||||
|
</Popup>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default UserManagementPopup;
|
@ -1,11 +1,9 @@
|
|||||||
// LOC830
|
// LOC830
|
||||||
import React, { useState, useRef, useEffect, useContext } from 'react';
|
import React, { useRef, useEffect } from 'react';
|
||||||
import updateApolloCache from 'shared/utils/cache';
|
import updateApolloCache from 'shared/utils/cache';
|
||||||
import GlobalTopNavbar from 'App/TopNavbar';
|
import GlobalTopNavbar from 'App/TopNavbar';
|
||||||
import ProjectPopup from 'App/TopNavbar/ProjectPopup';
|
import ProjectPopup from 'App/TopNavbar/ProjectPopup';
|
||||||
import styled from 'styled-components/macro';
|
import { usePopup } from 'shared/components/PopupMenu';
|
||||||
import AsyncSelect from 'react-select/async';
|
|
||||||
import { usePopup, Popup } from 'shared/components/PopupMenu';
|
|
||||||
import {
|
import {
|
||||||
useParams,
|
useParams,
|
||||||
Route,
|
Route,
|
||||||
@ -24,392 +22,51 @@ import {
|
|||||||
useFindProjectQuery,
|
useFindProjectQuery,
|
||||||
useDeleteInvitedProjectMemberMutation,
|
useDeleteInvitedProjectMemberMutation,
|
||||||
useUpdateTaskNameMutation,
|
useUpdateTaskNameMutation,
|
||||||
useCreateTaskMutation,
|
|
||||||
useDeleteTaskMutation,
|
useDeleteTaskMutation,
|
||||||
useUpdateTaskLocationMutation,
|
|
||||||
useUpdateTaskGroupLocationMutation,
|
|
||||||
useCreateTaskGroupMutation,
|
|
||||||
useUpdateTaskDescriptionMutation,
|
useUpdateTaskDescriptionMutation,
|
||||||
FindProjectDocument,
|
FindProjectDocument,
|
||||||
FindProjectQuery,
|
FindProjectQuery,
|
||||||
} from 'shared/generated/graphql';
|
} from 'shared/generated/graphql';
|
||||||
import produce from 'immer';
|
import produce from 'immer';
|
||||||
import UserContext, { useCurrentUser } from 'App/context';
|
|
||||||
import Input from 'shared/components/Input';
|
|
||||||
import Member from 'shared/components/Member';
|
|
||||||
import EmptyBoard from 'shared/components/EmptyBoard';
|
|
||||||
import NOOP from 'shared/utils/noop';
|
import NOOP from 'shared/utils/noop';
|
||||||
import { Lock, Cross } from 'shared/icons';
|
import useStateWithLocalStorage from 'shared/hooks/useStateWithLocalStorage';
|
||||||
import Button from 'shared/components/Button';
|
import localStorage from 'shared/utils/localStorage';
|
||||||
import { useApolloClient } from '@apollo/react-hooks';
|
|
||||||
import TaskAssignee from 'shared/components/TaskAssignee';
|
|
||||||
import gql from 'graphql-tag';
|
|
||||||
import { colourStyles } from 'shared/components/Select';
|
|
||||||
import Board, { BoardLoading } from './Board';
|
import Board, { BoardLoading } from './Board';
|
||||||
import Details from './Details';
|
import Details from './Details';
|
||||||
import LabelManagerEditor from './LabelManagerEditor';
|
import LabelManagerEditor from './LabelManagerEditor';
|
||||||
import { mixin } from '../../shared/utils/styles';
|
import UserManagementPopup from './UserManagementPopup';
|
||||||
|
|
||||||
const CARD_LABEL_VARIANT_STORAGE_KEY = 'card_label_variant';
|
|
||||||
|
|
||||||
const RFC2822_EMAIL = /[a-z0-9!#$%&'*+/=?^_`{|}~-]+(?:\.[a-z0-9!#$%&'*+/=?^_`{|}~-]+)*@(?:[a-z0-9](?:[a-z0-9-]*[a-z0-9])?\.)+[a-z0-9](?:[a-z0-9-]*[a-z0-9])?/;
|
|
||||||
|
|
||||||
const useStateWithLocalStorage = (localStorageKey: string): [string, React.Dispatch<React.SetStateAction<string>>] => {
|
|
||||||
const [value, setValue] = React.useState<string>(localStorage.getItem(localStorageKey) || '');
|
|
||||||
|
|
||||||
React.useEffect(() => {
|
|
||||||
localStorage.setItem(localStorageKey, value);
|
|
||||||
}, [value]);
|
|
||||||
|
|
||||||
return [value, setValue];
|
|
||||||
};
|
|
||||||
|
|
||||||
const SearchInput = styled(Input)`
|
|
||||||
margin: 0;
|
|
||||||
`;
|
|
||||||
|
|
||||||
const UserMember = styled(Member)`
|
|
||||||
padding: 4px 0;
|
|
||||||
cursor: pointer;
|
|
||||||
&:hover {
|
|
||||||
background: ${props => mixin.rgba(props.theme.colors.bg.primary, 0.4)};
|
|
||||||
}
|
|
||||||
border-radius: 6px;
|
|
||||||
`;
|
|
||||||
|
|
||||||
const MemberList = styled.div`
|
|
||||||
margin: 8px 0;
|
|
||||||
`;
|
|
||||||
|
|
||||||
type InviteUserData = {
|
|
||||||
email?: string;
|
|
||||||
suerID?: string;
|
|
||||||
};
|
|
||||||
type UserManagementPopupProps = {
|
|
||||||
projectID: string;
|
|
||||||
users: Array<User>;
|
|
||||||
projectMembers: Array<TaskUser>;
|
|
||||||
onInviteProjectMembers: (data: Array<InviteUserData>) => void;
|
|
||||||
};
|
|
||||||
|
|
||||||
const VisibiltyPrivateIcon = styled(Lock)`
|
|
||||||
padding-right: 4px;
|
|
||||||
`;
|
|
||||||
|
|
||||||
const VisibiltyButtonText = styled.span`
|
|
||||||
color: rgba(${props => props.theme.colors.text.primary});
|
|
||||||
`;
|
|
||||||
|
|
||||||
const ShareActions = styled.div`
|
|
||||||
border-top: 1px solid #414561;
|
|
||||||
margin-top: 8px;
|
|
||||||
padding-top: 8px;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: space-between;
|
|
||||||
`;
|
|
||||||
|
|
||||||
const VisibiltyButton = styled.button`
|
|
||||||
cursor: pointer;
|
|
||||||
margin: 2px 4px;
|
|
||||||
padding: 2px 4px;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: center;
|
|
||||||
border-bottom: 1px solid transparent;
|
|
||||||
&:hover ${VisibiltyButtonText} {
|
|
||||||
color: rgba(${props => props.theme.colors.text.secondary});
|
|
||||||
}
|
|
||||||
&:hover ${VisibiltyPrivateIcon} {
|
|
||||||
fill: rgba(${props => props.theme.colors.text.secondary});
|
|
||||||
stroke: rgba(${props => props.theme.colors.text.secondary});
|
|
||||||
}
|
|
||||||
&:hover {
|
|
||||||
border-bottom: 1px solid rgba(${props => props.theme.colors.primary});
|
|
||||||
}
|
|
||||||
`;
|
|
||||||
|
|
||||||
type MemberFilterOptions = {
|
|
||||||
projectID?: null | string;
|
|
||||||
teamID?: null | string;
|
|
||||||
organization?: boolean;
|
|
||||||
};
|
|
||||||
|
|
||||||
const fetchMembers = async (client: any, projectID: string, options: MemberFilterOptions, input: string, cb: any) => {
|
|
||||||
if (input && input.trim().length < 3) {
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
const res = await client.query({
|
|
||||||
query: gql`
|
|
||||||
query {
|
|
||||||
searchMembers(input: {searchFilter:"${input}", projectID:"${projectID}"}) {
|
|
||||||
id
|
|
||||||
similarity
|
|
||||||
status
|
|
||||||
user {
|
|
||||||
id
|
|
||||||
fullName
|
|
||||||
email
|
|
||||||
profileIcon {
|
|
||||||
url
|
|
||||||
initials
|
|
||||||
bgColor
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
`,
|
|
||||||
});
|
|
||||||
|
|
||||||
let results: any = [];
|
|
||||||
const emails: Array<string> = [];
|
|
||||||
if (res.data && res.data.searchMembers) {
|
|
||||||
results = [
|
|
||||||
...res.data.searchMembers.map((m: any) => {
|
|
||||||
if (m.status === 'INVITED') {
|
|
||||||
return {
|
|
||||||
label: m.id,
|
|
||||||
value: {
|
|
||||||
id: m.id,
|
|
||||||
type: 2,
|
|
||||||
profileIcon: {
|
|
||||||
bgColor: '#ccc',
|
|
||||||
initials: m.id.charAt(0),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
emails.push(m.user.email);
|
|
||||||
return {
|
|
||||||
label: m.user.fullName,
|
|
||||||
value: { id: m.user.id, type: 0, profileIcon: m.user.profileIcon },
|
|
||||||
};
|
|
||||||
}),
|
|
||||||
];
|
|
||||||
}
|
|
||||||
|
|
||||||
if (RFC2822_EMAIL.test(input) && !emails.find(e => e === input)) {
|
|
||||||
results = [
|
|
||||||
...results,
|
|
||||||
{
|
|
||||||
label: input,
|
|
||||||
value: {
|
|
||||||
id: input,
|
|
||||||
type: 1,
|
|
||||||
profileIcon: {
|
|
||||||
bgColor: '#ccc',
|
|
||||||
initials: input.charAt(0),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
];
|
|
||||||
}
|
|
||||||
|
|
||||||
return results;
|
|
||||||
};
|
|
||||||
|
|
||||||
type UserOptionProps = {
|
|
||||||
innerProps: any;
|
|
||||||
isDisabled: boolean;
|
|
||||||
isFocused: boolean;
|
|
||||||
label: string;
|
|
||||||
data: any;
|
|
||||||
getValue: any;
|
|
||||||
};
|
|
||||||
|
|
||||||
const OptionWrapper = styled.div<{ isFocused: boolean }>`
|
|
||||||
cursor: pointer;
|
|
||||||
padding: 4px 8px;
|
|
||||||
${props => props.isFocused && `background: rgba(${props.theme.colors.primary});`}
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
`;
|
|
||||||
const OptionContent = styled.div`
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
margin-left: 12px;
|
|
||||||
`;
|
|
||||||
|
|
||||||
const OptionLabel = styled.span<{ fontSize: number; quiet: boolean }>`
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
font-size: ${p => p.fontSize}px;
|
|
||||||
color: rgba(${p => (p.quiet ? p.theme.colors.text.primary : p.theme.colors.text.primary)});
|
|
||||||
`;
|
|
||||||
|
|
||||||
const UserOption: React.FC<UserOptionProps> = ({ isDisabled, isFocused, innerProps, label, data }) => {
|
|
||||||
return !isDisabled ? (
|
|
||||||
<OptionWrapper {...innerProps} isFocused={isFocused}>
|
|
||||||
<TaskAssignee
|
|
||||||
size={32}
|
|
||||||
member={{
|
|
||||||
id: '',
|
|
||||||
fullName: data.value.label,
|
|
||||||
profileIcon: data.value.profileIcon,
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
<OptionContent>
|
|
||||||
<OptionLabel fontSize={16} quiet={false}>
|
|
||||||
{label}
|
|
||||||
</OptionLabel>
|
|
||||||
{data.value.type === 2 && (
|
|
||||||
<OptionLabel fontSize={14} quiet>
|
|
||||||
Joined
|
|
||||||
</OptionLabel>
|
|
||||||
)}
|
|
||||||
</OptionContent>
|
|
||||||
</OptionWrapper>
|
|
||||||
) : null;
|
|
||||||
};
|
|
||||||
|
|
||||||
const OptionValueWrapper = styled.div`
|
|
||||||
background: rgba(${props => props.theme.colors.bg.primary});
|
|
||||||
border-radius: 4px;
|
|
||||||
margin: 2px;
|
|
||||||
padding: 3px 6px 3px 4px;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
`;
|
|
||||||
|
|
||||||
const OptionValueLabel = styled.span`
|
|
||||||
font-size: 12px;
|
|
||||||
color: rgba(${props => props.theme.colors.text.secondary});
|
|
||||||
`;
|
|
||||||
|
|
||||||
const OptionValueRemove = styled.button`
|
|
||||||
cursor: pointer;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: center;
|
|
||||||
background: none;
|
|
||||||
border: none;
|
|
||||||
outline: none;
|
|
||||||
padding: 0;
|
|
||||||
margin: 0;
|
|
||||||
margin-left: 4px;
|
|
||||||
`;
|
|
||||||
const OptionValue = ({ data, removeProps }: any) => {
|
|
||||||
return (
|
|
||||||
<OptionValueWrapper>
|
|
||||||
<OptionValueLabel>{data.label}</OptionValueLabel>
|
|
||||||
<OptionValueRemove {...removeProps}>
|
|
||||||
<Cross width={14} height={14} />
|
|
||||||
</OptionValueRemove>
|
|
||||||
</OptionValueWrapper>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
const InviteButton = styled(Button)`
|
|
||||||
margin-top: 12px;
|
|
||||||
height: 32px;
|
|
||||||
padding: 4px 12px;
|
|
||||||
width: 100%;
|
|
||||||
justify-content: center;
|
|
||||||
`;
|
|
||||||
|
|
||||||
const InviteContainer = styled.div`
|
|
||||||
min-height: 300px;
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
`;
|
|
||||||
|
|
||||||
const UserManagementPopup: React.FC<UserManagementPopupProps> = ({
|
|
||||||
projectID,
|
|
||||||
users,
|
|
||||||
projectMembers,
|
|
||||||
onInviteProjectMembers,
|
|
||||||
}) => {
|
|
||||||
const client = useApolloClient();
|
|
||||||
const [invitedUsers, setInvitedUsers] = useState<Array<any> | null>(null);
|
|
||||||
return (
|
|
||||||
<Popup tab={0} title="Invite a user">
|
|
||||||
<InviteContainer>
|
|
||||||
<AsyncSelect
|
|
||||||
getOptionValue={option => option.value.id}
|
|
||||||
placeholder="Email address or username"
|
|
||||||
noOptionsMessage={() => null}
|
|
||||||
onChange={(e: any) => {
|
|
||||||
setInvitedUsers(e);
|
|
||||||
}}
|
|
||||||
isMulti
|
|
||||||
autoFocus
|
|
||||||
cacheOptions
|
|
||||||
styles={colourStyles}
|
|
||||||
defaultOption
|
|
||||||
components={{
|
|
||||||
MultiValue: OptionValue,
|
|
||||||
Option: UserOption,
|
|
||||||
IndicatorSeparator: null,
|
|
||||||
DropdownIndicator: null,
|
|
||||||
}}
|
|
||||||
loadOptions={(i, cb) => fetchMembers(client, projectID, {}, i, cb)}
|
|
||||||
/>
|
|
||||||
</InviteContainer>
|
|
||||||
<InviteButton
|
|
||||||
onClick={() => {
|
|
||||||
if (invitedUsers) {
|
|
||||||
onInviteProjectMembers(
|
|
||||||
invitedUsers.map(user => {
|
|
||||||
if (user.value.type === 0) {
|
|
||||||
return {
|
|
||||||
userID: user.value.id,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
return {
|
|
||||||
email: user.value.id,
|
|
||||||
};
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}}
|
|
||||||
disabled={invitedUsers === null}
|
|
||||||
hoverVariant="none"
|
|
||||||
fontSize="16px"
|
|
||||||
>
|
|
||||||
Send Invite
|
|
||||||
</InviteButton>
|
|
||||||
</Popup>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
type TaskRouteProps = {
|
type TaskRouteProps = {
|
||||||
taskID: string;
|
taskID: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
interface QuickCardEditorState {
|
|
||||||
isOpen: boolean;
|
|
||||||
target: React.RefObject<HTMLElement> | null;
|
|
||||||
taskID: string | null;
|
|
||||||
taskGroupID: string | null;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface ProjectParams {
|
interface ProjectParams {
|
||||||
projectID: string;
|
projectID: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
const initialQuickCardEditorState: QuickCardEditorState = {
|
|
||||||
taskID: null,
|
|
||||||
taskGroupID: null,
|
|
||||||
isOpen: false,
|
|
||||||
target: null,
|
|
||||||
};
|
|
||||||
|
|
||||||
const Project = () => {
|
const Project = () => {
|
||||||
const { projectID } = useParams<ProjectParams>();
|
const { projectID } = useParams<ProjectParams>();
|
||||||
const history = useHistory();
|
const history = useHistory();
|
||||||
const match = useRouteMatch();
|
const match = useRouteMatch();
|
||||||
|
const location = useLocation();
|
||||||
|
|
||||||
|
const { showPopup, hidePopup } = usePopup();
|
||||||
|
const labelsRef = useRef<Array<ProjectLabel>>([]);
|
||||||
|
const [value, setValue] = useStateWithLocalStorage(localStorage.CARD_LABEL_VARIANT_STORAGE_KEY);
|
||||||
|
const taskLabelsRef = useRef<Array<TaskLabel>>([]);
|
||||||
|
|
||||||
const [updateTaskDescription] = useUpdateTaskDescriptionMutation();
|
const [updateTaskDescription] = useUpdateTaskDescriptionMutation();
|
||||||
const taskLabelsRef = useRef<Array<TaskLabel>>([]);
|
const [updateProjectMemberRole] = useUpdateProjectMemberRoleMutation();
|
||||||
|
const [updateTaskName] = useUpdateTaskNameMutation();
|
||||||
|
const { data, error } = useFindProjectQuery({
|
||||||
|
variables: { projectID },
|
||||||
|
pollInterval: 3000,
|
||||||
|
});
|
||||||
const [toggleTaskLabel] = useToggleTaskLabelMutation({
|
const [toggleTaskLabel] = useToggleTaskLabelMutation({
|
||||||
onCompleted: newTaskLabel => {
|
onCompleted: newTaskLabel => {
|
||||||
taskLabelsRef.current = newTaskLabel.toggleTaskLabel.task.labels;
|
taskLabelsRef.current = newTaskLabel.toggleTaskLabel.task.labels;
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
const [value, setValue] = useStateWithLocalStorage(CARD_LABEL_VARIANT_STORAGE_KEY);
|
|
||||||
const [updateProjectMemberRole] = useUpdateProjectMemberRoleMutation();
|
|
||||||
|
|
||||||
const [deleteTask] = useDeleteTaskMutation({
|
const [deleteTask] = useDeleteTaskMutation({
|
||||||
update: (client, resp) =>
|
update: (client, resp) =>
|
||||||
updateApolloCache<FindProjectQuery>(
|
updateApolloCache<FindProjectQuery>(
|
||||||
@ -433,13 +90,6 @@ const Project = () => {
|
|||||||
),
|
),
|
||||||
});
|
});
|
||||||
|
|
||||||
const [updateTaskName] = useUpdateTaskNameMutation();
|
|
||||||
|
|
||||||
const { loading, data, error } = useFindProjectQuery({
|
|
||||||
variables: { projectID },
|
|
||||||
pollInterval: 3000,
|
|
||||||
});
|
|
||||||
|
|
||||||
const [updateProjectName] = useUpdateProjectNameMutation({
|
const [updateProjectName] = useUpdateProjectNameMutation({
|
||||||
update: (client, newName) => {
|
update: (client, newName) => {
|
||||||
updateApolloCache<FindProjectQuery>(
|
updateApolloCache<FindProjectQuery>(
|
||||||
@ -507,20 +157,16 @@ const Project = () => {
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
const { user } = useCurrentUser();
|
|
||||||
const location = useLocation();
|
|
||||||
|
|
||||||
const { showPopup, hidePopup } = usePopup();
|
|
||||||
const $labelsRef = useRef<HTMLDivElement>(null);
|
|
||||||
const labelsRef = useRef<Array<ProjectLabel>>([]);
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (data) {
|
if (data) {
|
||||||
document.title = `${data.findProject.name} | Taskcafé`;
|
document.title = `${data.findProject.name} | Taskcafé`;
|
||||||
}
|
}
|
||||||
}, [data]);
|
}, [data]);
|
||||||
|
|
||||||
if (error) {
|
if (error) {
|
||||||
history.push('/projects');
|
history.push('/projects');
|
||||||
}
|
}
|
||||||
|
|
||||||
if (data) {
|
if (data) {
|
||||||
labelsRef.current = data.findProject.labels;
|
labelsRef.current = data.findProject.labels;
|
||||||
|
|
||||||
@ -530,7 +176,7 @@ const Project = () => {
|
|||||||
onChangeRole={(userID, roleCode) => {
|
onChangeRole={(userID, roleCode) => {
|
||||||
updateProjectMemberRole({ variables: { userID, roleCode, projectID } });
|
updateProjectMemberRole({ variables: { userID, roleCode, projectID } });
|
||||||
}}
|
}}
|
||||||
onChangeProjectOwner={uid => {
|
onChangeProjectOwner={() => {
|
||||||
hidePopup();
|
hidePopup();
|
||||||
}}
|
}}
|
||||||
onRemoveFromBoard={userID => {
|
onRemoveFromBoard={userID => {
|
||||||
|
13
frontend/src/shared/hooks/useStateWithLocalStorage.ts
Normal file
13
frontend/src/shared/hooks/useStateWithLocalStorage.ts
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
import React from 'react';
|
||||||
|
|
||||||
|
const useStateWithLocalStorage = (localStorageKey: string): [string, React.Dispatch<React.SetStateAction<string>>] => {
|
||||||
|
const [value, setValue] = React.useState<string>(localStorage.getItem(localStorageKey) || '');
|
||||||
|
|
||||||
|
React.useEffect(() => {
|
||||||
|
localStorage.setItem(localStorageKey, value);
|
||||||
|
}, [value]);
|
||||||
|
|
||||||
|
return [value, setValue];
|
||||||
|
};
|
||||||
|
|
||||||
|
export default useStateWithLocalStorage;
|
5
frontend/src/shared/utils/email.ts
Normal file
5
frontend/src/shared/utils/email.ts
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
const RFC2822_EMAIL = /[a-z0-9!#$%&'*+/=?^_`{|}~-]+(?:\.[a-z0-9!#$%&'*+/=?^_`{|}~-]+)*@(?:[a-z0-9](?:[a-z0-9-]*[a-z0-9])?\.)+[a-z0-9](?:[a-z0-9-]*[a-z0-9])?/;
|
||||||
|
|
||||||
|
export default function isValidEmail(target: string) {
|
||||||
|
return RFC2822_EMAIL.test(target);
|
||||||
|
}
|
5
frontend/src/shared/utils/localStorage.ts
Normal file
5
frontend/src/shared/utils/localStorage.ts
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
const localStorage = {
|
||||||
|
CARD_LABEL_VARIANT_STORAGE_KEY: 'card_label_variant',
|
||||||
|
};
|
||||||
|
|
||||||
|
export default localStorage;
|
Loading…
Reference in New Issue
Block a user