mirror of
https://github.com/plankanban/planka.git
synced 2025-07-19 05:09:43 +02:00
parent
ad7fb51cfa
commit
2ee1166747
1557 changed files with 76832 additions and 47042 deletions
809
client/src/components/cards/CardModal/ProjectContent.jsx
Normal file
809
client/src/components/cards/CardModal/ProjectContent.jsx
Normal file
|
@ -0,0 +1,809 @@
|
|||
/*!
|
||||
* Copyright (c) 2024 PLANKA Software GmbH
|
||||
* Licensed under the Fair Use License: https://github.com/plankanban/planka/blob/master/LICENSE.md
|
||||
*/
|
||||
|
||||
import React, { useCallback, useContext, useMemo, useState } from 'react';
|
||||
import PropTypes from 'prop-types';
|
||||
import classNames from 'classnames';
|
||||
import { shallowEqual, useDispatch, useSelector } from 'react-redux';
|
||||
import { useTranslation } from 'react-i18next';
|
||||
import { Button, Grid, Icon } from 'semantic-ui-react';
|
||||
import { useDidUpdate } from '../../../lib/hooks';
|
||||
|
||||
import selectors from '../../../selectors';
|
||||
import entryActions from '../../../entry-actions';
|
||||
import { usePopupInClosableContext } from '../../../hooks';
|
||||
import { startStopwatch, stopStopwatch } from '../../../utils/stopwatch';
|
||||
import { isUsableMarkdownElement } from '../../../utils/element-helpers';
|
||||
import { BoardMembershipRoles, CardTypes, ListTypes } from '../../../constants/Enums';
|
||||
import { CardTypeIcons } from '../../../constants/Icons';
|
||||
import { ClosableContext } from '../../../contexts';
|
||||
import NameField from './NameField';
|
||||
import TaskLists from './TaskLists';
|
||||
import CustomFieldGroups from './CustomFieldGroups';
|
||||
import Communication from './Communication';
|
||||
import CreationDetailsStep from './CreationDetailsStep';
|
||||
import DueDateChip from '../DueDateChip';
|
||||
import StopwatchChip from '../StopwatchChip';
|
||||
import SelectCardTypeStep from '../SelectCardTypeStep';
|
||||
import EditDueDateStep from '../EditDueDateStep';
|
||||
import EditStopwatchStep from '../EditStopwatchStep';
|
||||
import MoveCardStep from '../MoveCardStep';
|
||||
import Markdown from '../../common/Markdown';
|
||||
import EditMarkdown from '../../common/EditMarkdown';
|
||||
import ConfirmationStep from '../../common/ConfirmationStep';
|
||||
import UserAvatar from '../../users/UserAvatar';
|
||||
import BoardMembershipsStep from '../../board-memberships/BoardMembershipsStep';
|
||||
import LabelChip from '../../labels/LabelChip';
|
||||
import LabelsStep from '../../labels/LabelsStep';
|
||||
import ListsStep from '../../lists/ListsStep';
|
||||
import AddTaskListStep from '../../task-lists/AddTaskListStep';
|
||||
import Attachments from '../../attachments/Attachments';
|
||||
import AddAttachmentStep from '../../attachments/AddAttachmentStep';
|
||||
import AddCustomFieldGroupStep from '../../custom-field-groups/AddCustomFieldGroupStep';
|
||||
|
||||
import styles from './ProjectContent.module.scss';
|
||||
|
||||
const ProjectContent = React.memo(({ onClose }) => {
|
||||
const selectListById = useMemo(() => selectors.makeSelectListById(), []);
|
||||
const selectPrevListById = useMemo(() => selectors.makeSelectListById(), []);
|
||||
|
||||
const card = useSelector(selectors.selectCurrentCard);
|
||||
const board = useSelector(selectors.selectCurrentBoard);
|
||||
const userIds = useSelector(selectors.selectUserIdsForCurrentCard);
|
||||
const labelIds = useSelector(selectors.selectLabelIdsForCurrentCard);
|
||||
const attachmentIds = useSelector(selectors.selectAttachmentIdsForCurrentCard);
|
||||
|
||||
const isJoined = useSelector(selectors.selectIsCurrentUserInCurrentCard);
|
||||
|
||||
const list = useSelector((state) => selectListById(state, card.listId));
|
||||
|
||||
// TODO: check availability?
|
||||
const prevList = useSelector(
|
||||
(state) => card.prevListId && selectPrevListById(state, card.prevListId),
|
||||
);
|
||||
|
||||
const isInArchiveList = list.type === ListTypes.ARCHIVE;
|
||||
const isInTrashList = list.type === ListTypes.TRASH;
|
||||
|
||||
const {
|
||||
canEditType,
|
||||
canEditName,
|
||||
canEditDescription,
|
||||
canEditDueDate,
|
||||
canEditStopwatch,
|
||||
canSubscribe,
|
||||
canJoin,
|
||||
canDuplicate,
|
||||
canMove,
|
||||
canRestore,
|
||||
canArchive,
|
||||
canDelete,
|
||||
canUseLists,
|
||||
canUseMembers,
|
||||
canUseLabels,
|
||||
canAddTaskList,
|
||||
canAddAttachment,
|
||||
canAddCustomFieldGroup,
|
||||
} = useSelector((state) => {
|
||||
const boardMembership = selectors.selectCurrentUserMembershipForCurrentBoard(state);
|
||||
|
||||
let isMember = false;
|
||||
let isEditor = false;
|
||||
|
||||
if (boardMembership) {
|
||||
isMember = true;
|
||||
isEditor = boardMembership.role === BoardMembershipRoles.EDITOR;
|
||||
}
|
||||
|
||||
if (isInArchiveList || isInTrashList) {
|
||||
return {
|
||||
canEditType: false,
|
||||
canEditName: false,
|
||||
canEditDescription: false,
|
||||
canEditDueDate: false,
|
||||
canEditStopwatch: false,
|
||||
canSubscribe: isMember,
|
||||
canJoin: false,
|
||||
canDuplicate: false,
|
||||
canMove: false,
|
||||
canRestore: isEditor,
|
||||
canArchive: isEditor,
|
||||
canDelete: isEditor,
|
||||
canUseLists: isEditor,
|
||||
canUseMembers: false,
|
||||
canUseLabels: false,
|
||||
canAddTaskList: false,
|
||||
canAddAttachment: false,
|
||||
canAddCustomFieldGroup: false,
|
||||
};
|
||||
}
|
||||
|
||||
return {
|
||||
canEditType: isEditor,
|
||||
canEditName: isEditor,
|
||||
canEditDescription: isEditor,
|
||||
canEditDueDate: isEditor,
|
||||
canEditStopwatch: isEditor,
|
||||
canSubscribe: isMember,
|
||||
canJoin: isEditor,
|
||||
canDuplicate: isEditor,
|
||||
canMove: isEditor,
|
||||
canRestore: null,
|
||||
canArchive: isEditor,
|
||||
canDelete: isEditor,
|
||||
canUseLists: isEditor,
|
||||
canUseMembers: isEditor,
|
||||
canUseLabels: isEditor,
|
||||
canAddTaskList: isEditor,
|
||||
canAddAttachment: isEditor,
|
||||
canAddCustomFieldGroup: isEditor,
|
||||
};
|
||||
}, shallowEqual);
|
||||
|
||||
const dispatch = useDispatch();
|
||||
const [t] = useTranslation();
|
||||
const [descriptionDraft, setDescriptionDraft] = useState(null);
|
||||
const [isEditDescriptionOpened, setIsEditDescriptionOpened] = useState(false);
|
||||
const [, , setIsClosableActive] = useContext(ClosableContext);
|
||||
|
||||
const handleListSelect = useCallback(
|
||||
(listId) => {
|
||||
dispatch(entryActions.moveCurrentCard(listId));
|
||||
},
|
||||
[dispatch],
|
||||
);
|
||||
|
||||
const handleTypeSelect = useCallback(
|
||||
(type) => {
|
||||
dispatch(
|
||||
entryActions.updateCurrentCard({
|
||||
type,
|
||||
}),
|
||||
);
|
||||
},
|
||||
[dispatch],
|
||||
);
|
||||
|
||||
const handleNameUpdate = useCallback(
|
||||
(name) => {
|
||||
dispatch(
|
||||
entryActions.updateCurrentCard({
|
||||
name,
|
||||
}),
|
||||
);
|
||||
},
|
||||
[dispatch],
|
||||
);
|
||||
|
||||
const handleDescriptionUpdate = useCallback(
|
||||
(description) => {
|
||||
dispatch(
|
||||
entryActions.updateCurrentCard({
|
||||
description,
|
||||
}),
|
||||
);
|
||||
},
|
||||
[dispatch],
|
||||
);
|
||||
|
||||
const handleToggleStopwatchClick = useCallback(() => {
|
||||
dispatch(
|
||||
entryActions.updateCurrentCard({
|
||||
stopwatch: card.stopwatch.startedAt
|
||||
? stopStopwatch(card.stopwatch)
|
||||
: startStopwatch(card.stopwatch),
|
||||
}),
|
||||
);
|
||||
}, [card.stopwatch, dispatch]);
|
||||
|
||||
const handleDuplicateClick = useCallback(() => {
|
||||
dispatch(
|
||||
entryActions.duplicateCurrentCard({
|
||||
name: `${card.name} (${t('common.copy', {
|
||||
context: 'inline',
|
||||
})})`,
|
||||
}),
|
||||
);
|
||||
|
||||
onClose();
|
||||
}, [onClose, card.name, dispatch, t]);
|
||||
|
||||
const handleRestoreClick = useCallback(() => {
|
||||
dispatch(entryActions.moveCurrentCard(card.prevListId, undefined, true));
|
||||
}, [card.prevListId, dispatch]);
|
||||
|
||||
const handleArchiveConfirm = useCallback(() => {
|
||||
dispatch(entryActions.moveCurrentCardToArchive());
|
||||
}, [dispatch]);
|
||||
|
||||
const handleDeleteConfirm = useCallback(() => {
|
||||
if (isInTrashList) {
|
||||
dispatch(entryActions.deleteCurrentCard());
|
||||
} else {
|
||||
dispatch(entryActions.moveCurrentCardToTrash());
|
||||
}
|
||||
}, [isInTrashList, dispatch]);
|
||||
|
||||
const handleUserSelect = useCallback(
|
||||
(userId) => {
|
||||
dispatch(entryActions.addUserToCurrentCard(userId));
|
||||
},
|
||||
[dispatch],
|
||||
);
|
||||
|
||||
const handleUserDeselect = useCallback(
|
||||
(userId) => {
|
||||
dispatch(entryActions.removeUserFromCurrentCard(userId));
|
||||
},
|
||||
[dispatch],
|
||||
);
|
||||
|
||||
const handleLabelSelect = useCallback(
|
||||
(labelId) => {
|
||||
dispatch(entryActions.addLabelToCurrentCard(labelId));
|
||||
},
|
||||
[dispatch],
|
||||
);
|
||||
|
||||
const handleLabelDeselect = useCallback(
|
||||
(labelId) => {
|
||||
dispatch(entryActions.removeLabelFromCurrentCard(labelId));
|
||||
},
|
||||
[dispatch],
|
||||
);
|
||||
|
||||
const handleCustomFieldGroupCreate = useCallback(
|
||||
(data) => {
|
||||
dispatch(entryActions.createCustomFieldGroupInCurrentCard(data));
|
||||
},
|
||||
[dispatch],
|
||||
);
|
||||
|
||||
const handleToggleJointClick = useCallback(() => {
|
||||
if (isJoined) {
|
||||
dispatch(entryActions.removeCurrentUserFromCurrentCard());
|
||||
} else {
|
||||
dispatch(entryActions.addCurrentUserToCurrentCard());
|
||||
}
|
||||
}, [isJoined, dispatch]);
|
||||
|
||||
const handleToggleSubscriptionClick = useCallback(() => {
|
||||
dispatch(
|
||||
entryActions.updateCurrentCard({
|
||||
isSubscribed: !card.isSubscribed,
|
||||
}),
|
||||
);
|
||||
}, [card.isSubscribed, dispatch]);
|
||||
|
||||
const handleEditDescriptionClick = useCallback((event) => {
|
||||
if (window.getSelection().toString() || isUsableMarkdownElement(event.target)) {
|
||||
return;
|
||||
}
|
||||
|
||||
setIsEditDescriptionOpened(true);
|
||||
}, []);
|
||||
|
||||
const handleEditDescriptionClose = useCallback((nextDescriptionDraft) => {
|
||||
setDescriptionDraft(nextDescriptionDraft);
|
||||
setIsEditDescriptionOpened(false);
|
||||
}, []);
|
||||
|
||||
useDidUpdate(() => {
|
||||
if (!canEditDescription) {
|
||||
setIsEditDescriptionOpened(false);
|
||||
}
|
||||
}, [canEditDescription]);
|
||||
|
||||
useDidUpdate(() => {
|
||||
setIsClosableActive(isEditDescriptionOpened);
|
||||
}, [isEditDescriptionOpened]);
|
||||
|
||||
const CreationDetailsPopup = usePopupInClosableContext(CreationDetailsStep);
|
||||
const BoardMembershipsPopup = usePopupInClosableContext(BoardMembershipsStep);
|
||||
const LabelsPopup = usePopupInClosableContext(LabelsStep);
|
||||
const ListsPopup = usePopupInClosableContext(ListsStep);
|
||||
const SelectCardTypePopup = usePopupInClosableContext(SelectCardTypeStep);
|
||||
const EditDueDatePopup = usePopupInClosableContext(EditDueDateStep);
|
||||
const EditStopwatchPopup = usePopupInClosableContext(EditStopwatchStep);
|
||||
const AddTaskListPopup = usePopupInClosableContext(AddTaskListStep);
|
||||
const AddAttachmentPopup = usePopupInClosableContext(AddAttachmentStep);
|
||||
const AddCustomFieldGroupPopup = usePopupInClosableContext(AddCustomFieldGroupStep);
|
||||
const MoveCardPopup = usePopupInClosableContext(MoveCardStep);
|
||||
const ConfirmationPopup = usePopupInClosableContext(ConfirmationStep);
|
||||
|
||||
return (
|
||||
<Grid className={styles.wrapper}>
|
||||
<Grid.Row className={styles.headerPadding}>
|
||||
<Grid.Column width={16} className={styles.headerPadding}>
|
||||
<div className={styles.headerWrapper}>
|
||||
<Icon name={CardTypeIcons[CardTypes.PROJECT]} className={styles.moduleIcon} />
|
||||
<div className={styles.headerTitleWrapper}>
|
||||
{canEditName ? (
|
||||
<NameField defaultValue={card.name} onUpdate={handleNameUpdate} />
|
||||
) : (
|
||||
<div className={styles.headerTitle}>{card.name}</div>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
</Grid.Column>
|
||||
</Grid.Row>
|
||||
<Grid.Row className={styles.modalPadding}>
|
||||
<Grid.Column width={12} className={styles.contentPadding}>
|
||||
{(card.dueDate ||
|
||||
card.stopwatch ||
|
||||
board.alwaysDisplayCardCreator ||
|
||||
userIds.length > 0 ||
|
||||
labelIds.length > 0) && (
|
||||
<div className={styles.moduleWrapper}>
|
||||
{board.alwaysDisplayCardCreator && (
|
||||
<div className={styles.attachments}>
|
||||
<div className={styles.text}>
|
||||
{t('common.creator', {
|
||||
context: 'title',
|
||||
})}
|
||||
</div>
|
||||
<span className={styles.attachment}>
|
||||
<CreationDetailsPopup userId={card.creatorUserId}>
|
||||
<UserAvatar withCreatorIndicator id={card.creatorUserId} />
|
||||
</CreationDetailsPopup>
|
||||
</span>
|
||||
</div>
|
||||
)}
|
||||
{userIds.length > 0 && (
|
||||
<div className={styles.attachments}>
|
||||
<div className={styles.text}>
|
||||
{t('common.members', {
|
||||
context: 'title',
|
||||
})}
|
||||
</div>
|
||||
{userIds.map((userId) => (
|
||||
<span key={userId} className={styles.attachment}>
|
||||
{canUseMembers ? (
|
||||
<BoardMembershipsPopup
|
||||
currentUserIds={userIds}
|
||||
onUserSelect={handleUserSelect}
|
||||
onUserDeselect={handleUserDeselect}
|
||||
>
|
||||
<UserAvatar id={userId} />
|
||||
</BoardMembershipsPopup>
|
||||
) : (
|
||||
<UserAvatar id={userId} />
|
||||
)}
|
||||
</span>
|
||||
))}
|
||||
{canUseMembers && (
|
||||
<BoardMembershipsPopup
|
||||
currentUserIds={userIds}
|
||||
onUserSelect={handleUserSelect}
|
||||
onUserDeselect={handleUserDeselect}
|
||||
>
|
||||
<button
|
||||
type="button"
|
||||
className={classNames(styles.attachment, styles.dueDate)}
|
||||
>
|
||||
<Icon name="add" size="small" className={styles.addAttachment} />
|
||||
</button>
|
||||
</BoardMembershipsPopup>
|
||||
)}
|
||||
</div>
|
||||
)}
|
||||
{labelIds.length > 0 && (
|
||||
<div className={styles.attachments}>
|
||||
<div className={styles.text}>
|
||||
{t('common.labels', {
|
||||
context: 'title',
|
||||
})}
|
||||
</div>
|
||||
{labelIds.map((labelId) => (
|
||||
<span key={labelId} className={styles.attachment}>
|
||||
{canUseLabels ? (
|
||||
<LabelsPopup
|
||||
currentIds={labelIds}
|
||||
cardId={card.id}
|
||||
onSelect={handleLabelSelect}
|
||||
onDeselect={handleLabelDeselect}
|
||||
>
|
||||
<LabelChip id={labelId} />
|
||||
</LabelsPopup>
|
||||
) : (
|
||||
<LabelChip id={labelId} />
|
||||
)}
|
||||
</span>
|
||||
))}
|
||||
{canUseLabels && (
|
||||
<LabelsPopup
|
||||
currentIds={labelIds}
|
||||
cardId={card.id}
|
||||
onSelect={handleLabelSelect}
|
||||
onDeselect={handleLabelDeselect}
|
||||
>
|
||||
<button
|
||||
type="button"
|
||||
className={classNames(styles.attachment, styles.dueDate)}
|
||||
>
|
||||
<Icon name="add" size="small" className={styles.addAttachment} />
|
||||
</button>
|
||||
</LabelsPopup>
|
||||
)}
|
||||
</div>
|
||||
)}
|
||||
{card.dueDate && (
|
||||
<div className={styles.attachments}>
|
||||
<div className={styles.text}>
|
||||
{t('common.dueDate', {
|
||||
context: 'title',
|
||||
})}
|
||||
</div>
|
||||
<span className={styles.attachment}>
|
||||
{canEditDueDate ? (
|
||||
<EditDueDatePopup cardId={card.id}>
|
||||
<DueDateChip
|
||||
withStatusIcon
|
||||
value={card.dueDate}
|
||||
withStatus={
|
||||
list.type !== ListTypes.CLOSED && !isInArchiveList && !isInTrashList
|
||||
}
|
||||
/>
|
||||
</EditDueDatePopup>
|
||||
) : (
|
||||
<DueDateChip
|
||||
withStatusIcon
|
||||
value={card.dueDate}
|
||||
withStatus={
|
||||
list.type !== ListTypes.CLOSED && !isInArchiveList && !isInTrashList
|
||||
}
|
||||
/>
|
||||
)}
|
||||
</span>
|
||||
</div>
|
||||
)}
|
||||
{card.stopwatch && (
|
||||
<div className={styles.attachments}>
|
||||
<div className={styles.text}>
|
||||
{t('common.stopwatch', {
|
||||
context: 'title',
|
||||
})}
|
||||
</div>
|
||||
<span className={styles.attachment}>
|
||||
{canEditStopwatch ? (
|
||||
<EditStopwatchPopup cardId={card.id}>
|
||||
<StopwatchChip value={card.stopwatch} />
|
||||
</EditStopwatchPopup>
|
||||
) : (
|
||||
<StopwatchChip value={card.stopwatch} />
|
||||
)}
|
||||
</span>
|
||||
{canEditStopwatch && (
|
||||
<button
|
||||
type="button"
|
||||
className={classNames(styles.attachment, styles.dueDate)}
|
||||
onClick={handleToggleStopwatchClick}
|
||||
>
|
||||
<Icon
|
||||
name={card.stopwatch.startedAt ? 'pause' : 'play'}
|
||||
size="small"
|
||||
className={styles.addAttachment}
|
||||
/>
|
||||
</button>
|
||||
)}
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
)}
|
||||
{(card.description || canEditDescription) && (
|
||||
<div className={classNames(styles.contentModule, styles.contentModuleDescription)}>
|
||||
<div className={styles.moduleWrapper}>
|
||||
<Icon name="align left" className={styles.moduleIcon} />
|
||||
<div className={styles.moduleHeader}>
|
||||
{t('common.description')}
|
||||
{canEditDescription && !isEditDescriptionOpened && descriptionDraft && (
|
||||
<span className={styles.draftChip}>{t('common.unsavedChanges')}</span>
|
||||
)}
|
||||
</div>
|
||||
{canEditDescription && (
|
||||
<>
|
||||
{isEditDescriptionOpened && (
|
||||
<EditMarkdown
|
||||
defaultValue={card.description}
|
||||
draftValue={descriptionDraft}
|
||||
placeholder="common.enterDescription"
|
||||
onUpdate={handleDescriptionUpdate}
|
||||
onClose={handleEditDescriptionClose}
|
||||
/>
|
||||
)}
|
||||
{!isEditDescriptionOpened &&
|
||||
(card.description ? (
|
||||
/* eslint-disable-next-line jsx-a11y/click-events-have-key-events,
|
||||
jsx-a11y/no-static-element-interactions */
|
||||
<div className={styles.cursorPointer} onClick={handleEditDescriptionClick}>
|
||||
<Button className={styles.editButton}>
|
||||
<Icon fitted name="pencil" size="small" />
|
||||
</Button>
|
||||
<Markdown>{card.description}</Markdown>
|
||||
</div>
|
||||
) : (
|
||||
<button
|
||||
type="button"
|
||||
className={styles.descriptionButton}
|
||||
onClick={handleEditDescriptionClick}
|
||||
>
|
||||
<span className={styles.descriptionButtonText}>
|
||||
{t('action.addMoreDetailedDescription')}
|
||||
</span>
|
||||
</button>
|
||||
))}
|
||||
</>
|
||||
)}
|
||||
{!canEditDescription && <Markdown>{card.description}</Markdown>}
|
||||
</div>
|
||||
</div>
|
||||
)}
|
||||
<CustomFieldGroups />
|
||||
<TaskLists />
|
||||
{attachmentIds.length > 0 && (
|
||||
<div className={styles.contentModule}>
|
||||
<div className={styles.moduleWrapper}>
|
||||
<Icon name="attach" className={styles.moduleIcon} />
|
||||
<div className={styles.moduleHeader}>{t('common.attachments')}</div>
|
||||
<Attachments />
|
||||
</div>
|
||||
</div>
|
||||
)}
|
||||
<div className={styles.contentModule}>
|
||||
<div className={styles.moduleWrapper}>
|
||||
<Icon name="list ul" className={styles.moduleIcon} />
|
||||
<Communication />
|
||||
</div>
|
||||
</div>
|
||||
</Grid.Column>
|
||||
<Grid.Column width={4} className={styles.sidebarPadding}>
|
||||
<div className={styles.sticky}>
|
||||
<div className={styles.actions}>
|
||||
<div className={classNames(styles.attachments, styles.attachmentsList)}>
|
||||
<div className={classNames(styles.text, styles.textList)}>{t('common.list')}</div>
|
||||
{canUseLists ? (
|
||||
<ListsPopup currentId={list.id} onSelect={handleListSelect}>
|
||||
<button type="button" className={styles.listButton}>
|
||||
<span className={classNames(styles.list, styles.listHoverable)}>
|
||||
<Icon name="columns" size="small" className={styles.listIcon} />
|
||||
<span className={styles.hidable}>
|
||||
{list.name || t(`common.${list.type}`)}
|
||||
</span>
|
||||
</span>
|
||||
</button>
|
||||
</ListsPopup>
|
||||
) : (
|
||||
<span className={styles.list}>
|
||||
<Icon name="columns" size="small" className={styles.listIcon} />
|
||||
<span className={styles.hidable}>{list.name || t(`common.${list.type}`)}</span>
|
||||
</span>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
{(canEditDueDate ||
|
||||
canEditStopwatch ||
|
||||
canUseMembers ||
|
||||
canUseLabels ||
|
||||
canAddTaskList ||
|
||||
canAddAttachment ||
|
||||
canAddCustomFieldGroup) && (
|
||||
<div className={styles.actions}>
|
||||
<span className={styles.actionsTitle}>{t('action.addToCard')}</span>
|
||||
{canUseMembers && (
|
||||
<BoardMembershipsPopup
|
||||
currentUserIds={userIds}
|
||||
onUserSelect={handleUserSelect}
|
||||
onUserDeselect={handleUserDeselect}
|
||||
>
|
||||
<Button fluid className={classNames(styles.actionButton, styles.hidable)}>
|
||||
<Icon name="user outline" className={styles.actionIcon} />
|
||||
{t('common.members')}
|
||||
</Button>
|
||||
</BoardMembershipsPopup>
|
||||
)}
|
||||
{canUseLabels && (
|
||||
<LabelsPopup
|
||||
currentIds={labelIds}
|
||||
cardId={card.id}
|
||||
onSelect={handleLabelSelect}
|
||||
onDeselect={handleLabelDeselect}
|
||||
>
|
||||
<Button fluid className={classNames(styles.actionButton, styles.hidable)}>
|
||||
<Icon name="bookmark outline" className={styles.actionIcon} />
|
||||
{t('common.labels')}
|
||||
</Button>
|
||||
</LabelsPopup>
|
||||
)}
|
||||
{canEditDueDate && (
|
||||
<EditDueDatePopup cardId={card.id}>
|
||||
<Button fluid className={classNames(styles.actionButton, styles.hidable)}>
|
||||
<Icon name="calendar check outline" className={styles.actionIcon} />
|
||||
{t('common.dueDate', {
|
||||
context: 'title',
|
||||
})}
|
||||
</Button>
|
||||
</EditDueDatePopup>
|
||||
)}
|
||||
{canEditStopwatch && (
|
||||
<EditStopwatchPopup cardId={card.id}>
|
||||
<Button fluid className={classNames(styles.actionButton, styles.hidable)}>
|
||||
<Icon name="clock outline" className={styles.actionIcon} />
|
||||
{t('common.stopwatch')}
|
||||
</Button>
|
||||
</EditStopwatchPopup>
|
||||
)}
|
||||
{canAddTaskList && (
|
||||
<AddTaskListPopup>
|
||||
<Button fluid className={classNames(styles.actionButton, styles.hidable)}>
|
||||
<Icon name="check square outline" className={styles.actionIcon} />
|
||||
{t('common.taskList', {
|
||||
context: 'title',
|
||||
})}
|
||||
</Button>
|
||||
</AddTaskListPopup>
|
||||
)}
|
||||
{canAddAttachment && (
|
||||
<AddAttachmentPopup>
|
||||
<Button fluid className={classNames(styles.actionButton, styles.hidable)}>
|
||||
<Icon name="attach" className={styles.actionIcon} />
|
||||
{t('common.attachment')}
|
||||
</Button>
|
||||
</AddAttachmentPopup>
|
||||
)}
|
||||
{canAddCustomFieldGroup && (
|
||||
<AddCustomFieldGroupPopup onCreate={handleCustomFieldGroupCreate}>
|
||||
<Button fluid className={classNames(styles.actionButton, styles.hidable)}>
|
||||
<Icon name="sticky note outline" className={styles.actionIcon} />
|
||||
{t('common.customField', {
|
||||
context: 'title',
|
||||
})}
|
||||
</Button>
|
||||
</AddCustomFieldGroupPopup>
|
||||
)}
|
||||
</div>
|
||||
)}
|
||||
{((!board.limitCardTypesToDefaultOne && canEditType) ||
|
||||
canSubscribe ||
|
||||
canJoin ||
|
||||
canDuplicate ||
|
||||
canMove ||
|
||||
(canRestore && (isInArchiveList || isInTrashList)) ||
|
||||
(canArchive && !isInArchiveList) ||
|
||||
canDelete) && (
|
||||
<div className={styles.actions}>
|
||||
<span className={styles.actionsTitle}>{t('common.actions')}</span>
|
||||
{canJoin && (
|
||||
<Button
|
||||
fluid
|
||||
className={classNames(styles.actionButton, styles.hidable)}
|
||||
onClick={handleToggleJointClick}
|
||||
>
|
||||
<Icon
|
||||
name={isJoined ? 'flag outline' : 'flag checkered'}
|
||||
className={styles.actionIcon}
|
||||
/>
|
||||
{isJoined ? t('action.leave') : t('action.join')}
|
||||
</Button>
|
||||
)}
|
||||
{canSubscribe && (
|
||||
<Button
|
||||
fluid
|
||||
disabled={board.isSubscribed}
|
||||
className={classNames(styles.actionButton, styles.hidable)}
|
||||
onClick={handleToggleSubscriptionClick}
|
||||
>
|
||||
{board.isSubscribed ? (
|
||||
<>
|
||||
<Icon name="bell slash outline" className={styles.actionIcon} />
|
||||
{t('common.boardSubscribed')}
|
||||
</>
|
||||
) : (
|
||||
<>
|
||||
<Icon
|
||||
name={card.isSubscribed ? 'bell slash outline' : 'bell outline'}
|
||||
className={styles.actionIcon}
|
||||
/>
|
||||
{card.isSubscribed ? t('action.unsubscribe') : t('action.subscribe')}
|
||||
</>
|
||||
)}
|
||||
</Button>
|
||||
)}
|
||||
{!board.limitCardTypesToDefaultOne && canEditType && (
|
||||
<SelectCardTypePopup
|
||||
withButton
|
||||
defaultValue={card.type}
|
||||
title="common.editType"
|
||||
buttonContent="action.save"
|
||||
onSelect={handleTypeSelect}
|
||||
>
|
||||
<Button fluid className={classNames(styles.actionButton, styles.hidable)}>
|
||||
<Icon name="map outline" className={styles.actionIcon} />
|
||||
{t('action.editType', {
|
||||
context: 'title',
|
||||
})}
|
||||
</Button>
|
||||
</SelectCardTypePopup>
|
||||
)}
|
||||
{canDuplicate && (
|
||||
<Button
|
||||
fluid
|
||||
className={classNames(styles.actionButton, styles.hidable)}
|
||||
onClick={handleDuplicateClick}
|
||||
>
|
||||
<Icon name="copy outline" className={styles.actionIcon} />
|
||||
{t('action.duplicate')}
|
||||
</Button>
|
||||
)}
|
||||
{canMove && (
|
||||
<MoveCardPopup id={card.id}>
|
||||
<Button fluid className={classNames(styles.actionButton, styles.hidable)}>
|
||||
<Icon name="share square outline" className={styles.actionIcon} />
|
||||
{t('action.move')}
|
||||
</Button>
|
||||
</MoveCardPopup>
|
||||
)}
|
||||
{canRestore && (isInArchiveList || isInTrashList) && (
|
||||
<Button
|
||||
fluid
|
||||
disabled={!prevList}
|
||||
className={classNames(styles.actionButton, styles.hidable)}
|
||||
onClick={handleRestoreClick}
|
||||
>
|
||||
<Icon name="undo alternate" className={styles.actionIcon} />
|
||||
{prevList
|
||||
? t('action.restoreToList', {
|
||||
list: prevList.name || t(`common.${prevList.type}`),
|
||||
})
|
||||
: t('common.selectListToRestoreThisCard')}
|
||||
</Button>
|
||||
)}
|
||||
{canArchive && !isInArchiveList && (
|
||||
<ConfirmationPopup
|
||||
title="common.archiveCard"
|
||||
content="common.areYouSureYouWantToArchiveThisCard"
|
||||
buttonContent="action.archiveCard"
|
||||
onConfirm={handleArchiveConfirm}
|
||||
>
|
||||
<Button fluid className={classNames(styles.actionButton, styles.hidable)}>
|
||||
<Icon name="folder open outline" className={styles.actionIcon} />
|
||||
{t('action.archive')}
|
||||
</Button>
|
||||
</ConfirmationPopup>
|
||||
)}
|
||||
{canDelete && (
|
||||
<ConfirmationPopup
|
||||
title={isInTrashList ? 'common.deleteCardForever' : 'common.deleteCard'}
|
||||
content={
|
||||
isInTrashList
|
||||
? 'common.areYouSureYouWantToDeleteThisCardForever'
|
||||
: 'common.areYouSureYouWantToDeleteThisCard'
|
||||
}
|
||||
buttonContent={isInTrashList ? 'action.deleteCardForever' : 'action.deleteCard'}
|
||||
onConfirm={handleDeleteConfirm}
|
||||
>
|
||||
<Button fluid className={classNames(styles.actionButton, styles.hidable)}>
|
||||
<Icon name="trash alternate outline" className={styles.actionIcon} />
|
||||
{isInTrashList
|
||||
? t('action.deleteForever', {
|
||||
context: 'title',
|
||||
})
|
||||
: t('action.delete')}
|
||||
</Button>
|
||||
</ConfirmationPopup>
|
||||
)}
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
</Grid.Column>
|
||||
</Grid.Row>
|
||||
</Grid>
|
||||
);
|
||||
});
|
||||
|
||||
ProjectContent.propTypes = {
|
||||
onClose: PropTypes.func.isRequired,
|
||||
};
|
||||
|
||||
export default ProjectContent;
|
Loading…
Add table
Add a link
Reference in a new issue