mirror of
https://github.com/plankanban/planka.git
synced 2025-07-19 05:09:43 +02:00
ref: Remove board types, refactoring
This commit is contained in:
parent
d39da61295
commit
5cd025ffb7
182 changed files with 1573 additions and 1239 deletions
|
@ -35,18 +35,8 @@ export const transformCardData = (data) => ({
|
||||||
|
|
||||||
/* Actions */
|
/* Actions */
|
||||||
|
|
||||||
const getCards = (boardId, data, headers) =>
|
const createCard = (listId, data, headers) =>
|
||||||
socket.get(`/board/${boardId}/cards`, data, headers).then((body) => ({
|
socket.post(`/lists/${listId}/cards`, transformCardData(data), headers).then((body) => ({
|
||||||
...body,
|
|
||||||
items: body.items.map(transformCard),
|
|
||||||
included: {
|
|
||||||
...body.included,
|
|
||||||
attachments: body.included.attachments.map(transformAttachment),
|
|
||||||
},
|
|
||||||
}));
|
|
||||||
|
|
||||||
const createCard = (boardId, data, headers) =>
|
|
||||||
socket.post(`/boards/${boardId}/cards`, transformCardData(data), headers).then((body) => ({
|
|
||||||
...body,
|
...body,
|
||||||
item: transformCard(body.item),
|
item: transformCard(body.item),
|
||||||
}));
|
}));
|
||||||
|
@ -55,6 +45,10 @@ const getCard = (id, headers) =>
|
||||||
socket.get(`/cards/${id}`, undefined, headers).then((body) => ({
|
socket.get(`/cards/${id}`, undefined, headers).then((body) => ({
|
||||||
...body,
|
...body,
|
||||||
item: transformCard(body.item),
|
item: transformCard(body.item),
|
||||||
|
included: {
|
||||||
|
...body.included,
|
||||||
|
attachments: body.included.attachments.map(transformAttachment),
|
||||||
|
},
|
||||||
}));
|
}));
|
||||||
|
|
||||||
const updateCard = (id, data, headers) =>
|
const updateCard = (id, data, headers) =>
|
||||||
|
@ -83,7 +77,6 @@ const makeHandleCardUpdate = makeHandleCardCreate;
|
||||||
const makeHandleCardDelete = makeHandleCardCreate;
|
const makeHandleCardDelete = makeHandleCardCreate;
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
getCards,
|
|
||||||
createCard,
|
createCard,
|
||||||
getCard,
|
getCard,
|
||||||
updateCard,
|
updateCard,
|
||||||
|
|
|
@ -10,11 +10,11 @@ import CardModalContainer from '../../containers/CardModalContainer';
|
||||||
import ListAdd from './ListAdd';
|
import ListAdd from './ListAdd';
|
||||||
import { ReactComponent as PlusMathIcon } from '../../assets/images/plus-math-icon.svg';
|
import { ReactComponent as PlusMathIcon } from '../../assets/images/plus-math-icon.svg';
|
||||||
|
|
||||||
import styles from './BoardKanban.module.scss';
|
import styles from './Board.module.scss';
|
||||||
|
|
||||||
const parseDndId = (dndId) => dndId.split(':')[1];
|
const parseDndId = (dndId) => dndId.split(':')[1];
|
||||||
|
|
||||||
const BoardKanban = React.memo(
|
const Board = React.memo(
|
||||||
({ listIds, isCardModalOpened, canEdit, onListCreate, onListMove, onCardMove }) => {
|
({ listIds, isCardModalOpened, canEdit, onListCreate, onListMove, onCardMove }) => {
|
||||||
const [t] = useTranslation();
|
const [t] = useTranslation();
|
||||||
const [isListAddOpened, setIsListAddOpened] = useState(false);
|
const [isListAddOpened, setIsListAddOpened] = useState(false);
|
||||||
|
@ -166,7 +166,7 @@ const BoardKanban = React.memo(
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
|
||||||
BoardKanban.propTypes = {
|
Board.propTypes = {
|
||||||
listIds: PropTypes.array.isRequired, // eslint-disable-line react/forbid-prop-types
|
listIds: PropTypes.array.isRequired, // eslint-disable-line react/forbid-prop-types
|
||||||
isCardModalOpened: PropTypes.bool.isRequired,
|
isCardModalOpened: PropTypes.bool.isRequired,
|
||||||
canEdit: PropTypes.bool.isRequired,
|
canEdit: PropTypes.bool.isRequired,
|
||||||
|
@ -175,4 +175,4 @@ BoardKanban.propTypes = {
|
||||||
onCardMove: PropTypes.func.isRequired,
|
onCardMove: PropTypes.func.isRequired,
|
||||||
};
|
};
|
||||||
|
|
||||||
export default BoardKanban;
|
export default Board;
|
|
@ -72,7 +72,7 @@ const ListAdd = React.memo(({ onCreate, onClose }) => {
|
||||||
<Button
|
<Button
|
||||||
positive
|
positive
|
||||||
content={t('action.addList')}
|
content={t('action.addList')}
|
||||||
className={styles.submitButton}
|
className={styles.button}
|
||||||
onMouseOver={handleControlMouseOver}
|
onMouseOver={handleControlMouseOver}
|
||||||
onMouseOut={handleControlMouseOut}
|
onMouseOut={handleControlMouseOut}
|
||||||
/>
|
/>
|
|
@ -1,4 +1,9 @@
|
||||||
:global(#app) {
|
:global(#app) {
|
||||||
|
.button {
|
||||||
|
min-height: 30px;
|
||||||
|
vertical-align: top;
|
||||||
|
}
|
||||||
|
|
||||||
.controls {
|
.controls {
|
||||||
margin-top: 4px;
|
margin-top: 4px;
|
||||||
}
|
}
|
||||||
|
@ -18,11 +23,6 @@
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.submitButton {
|
|
||||||
min-height: 30px;
|
|
||||||
vertical-align: top;
|
|
||||||
}
|
|
||||||
|
|
||||||
.wrapper {
|
.wrapper {
|
||||||
background: #e2e4e6;
|
background: #e2e4e6;
|
||||||
border-radius: 3px;
|
border-radius: 3px;
|
3
client/src/components/Board/index.js
Executable file
3
client/src/components/Board/index.js
Executable file
|
@ -0,0 +1,3 @@
|
||||||
|
import Board from './Board';
|
||||||
|
|
||||||
|
export default Board;
|
|
@ -1,3 +0,0 @@
|
||||||
import BoardKanban from './BoardKanban';
|
|
||||||
|
|
||||||
export default BoardKanban;
|
|
|
@ -1,3 +1,4 @@
|
||||||
|
import { dequal } from 'dequal';
|
||||||
import omit from 'lodash/omit';
|
import omit from 'lodash/omit';
|
||||||
import React, { useCallback, useState } from 'react';
|
import React, { useCallback, useState } from 'react';
|
||||||
import PropTypes from 'prop-types';
|
import PropTypes from 'prop-types';
|
||||||
|
@ -10,7 +11,7 @@ import { BoardMembershipRoles } from '../../constants/Enums';
|
||||||
import styles from './BoardMembershipPermissionsSelectStep.module.scss';
|
import styles from './BoardMembershipPermissionsSelectStep.module.scss';
|
||||||
|
|
||||||
const BoardMembershipPermissionsSelectStep = React.memo(
|
const BoardMembershipPermissionsSelectStep = React.memo(
|
||||||
({ defaultData, title, buttonContent, onSelect, onBack }) => {
|
({ defaultData, title, buttonContent, onSelect, onBack, onClose }) => {
|
||||||
const [t] = useTranslation();
|
const [t] = useTranslation();
|
||||||
|
|
||||||
const [data, setData] = useState(() => ({
|
const [data, setData] = useState(() => ({
|
||||||
|
@ -23,7 +24,7 @@ const BoardMembershipPermissionsSelectStep = React.memo(
|
||||||
setData((prevData) => ({
|
setData((prevData) => ({
|
||||||
...prevData,
|
...prevData,
|
||||||
role,
|
role,
|
||||||
canComment: role === BoardMembershipRoles.EDITOR ? null : !!prevData.canComment,
|
canComment: role === BoardMembershipRoles.VIEWER ? !!prevData.canComment : null,
|
||||||
}));
|
}));
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
@ -35,8 +36,12 @@ const BoardMembershipPermissionsSelectStep = React.memo(
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const handleSubmit = useCallback(() => {
|
const handleSubmit = useCallback(() => {
|
||||||
onSelect(data.role === BoardMembershipRoles.EDITOR ? omit(data, 'canComment') : data);
|
if (!dequal(data, defaultData)) {
|
||||||
}, [onSelect, data]);
|
onSelect(data.role === BoardMembershipRoles.VIEWER ? data : omit(data, 'canComment'));
|
||||||
|
}
|
||||||
|
|
||||||
|
onClose();
|
||||||
|
}, [defaultData, onSelect, onClose, data]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
|
@ -65,7 +70,7 @@ const BoardMembershipPermissionsSelectStep = React.memo(
|
||||||
<div className={styles.menuItemDescription}>{t('common.canOnlyViewBoard')}</div>
|
<div className={styles.menuItemDescription}>{t('common.canOnlyViewBoard')}</div>
|
||||||
</Menu.Item>
|
</Menu.Item>
|
||||||
</Menu>
|
</Menu>
|
||||||
{data.role !== BoardMembershipRoles.EDITOR && (
|
{data.role === BoardMembershipRoles.VIEWER && (
|
||||||
<Segment basic className={styles.settings}>
|
<Segment basic className={styles.settings}>
|
||||||
<Radio
|
<Radio
|
||||||
toggle
|
toggle
|
||||||
|
@ -90,6 +95,7 @@ BoardMembershipPermissionsSelectStep.propTypes = {
|
||||||
buttonContent: PropTypes.string,
|
buttonContent: PropTypes.string,
|
||||||
onSelect: PropTypes.func.isRequired,
|
onSelect: PropTypes.func.isRequired,
|
||||||
onBack: PropTypes.func.isRequired,
|
onBack: PropTypes.func.isRequired,
|
||||||
|
onClose: PropTypes.func.isRequired,
|
||||||
};
|
};
|
||||||
|
|
||||||
BoardMembershipPermissionsSelectStep.defaultProps = {
|
BoardMembershipPermissionsSelectStep.defaultProps = {
|
||||||
|
|
|
@ -1,25 +0,0 @@
|
||||||
import React from 'react';
|
|
||||||
import PropTypes from 'prop-types';
|
|
||||||
import { Loader } from 'semantic-ui-react';
|
|
||||||
|
|
||||||
import { BoardTypes } from '../constants/Enums';
|
|
||||||
import BoardKanbanContainer from '../containers/BoardKanbanContainer';
|
|
||||||
|
|
||||||
const BoardWrapper = React.memo(({ type, isFetching }) => {
|
|
||||||
if (isFetching) {
|
|
||||||
return <Loader active />;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (type === BoardTypes.KANBAN) {
|
|
||||||
return <BoardKanbanContainer />;
|
|
||||||
}
|
|
||||||
|
|
||||||
return null;
|
|
||||||
});
|
|
||||||
|
|
||||||
BoardWrapper.propTypes = {
|
|
||||||
type: PropTypes.string.isRequired,
|
|
||||||
isFetching: PropTypes.bool.isRequired,
|
|
||||||
};
|
|
||||||
|
|
||||||
export default BoardWrapper;
|
|
|
@ -46,7 +46,6 @@ const AddStep = React.memo(({ onCreate, onClose }) => {
|
||||||
const handleSubmit = useCallback(() => {
|
const handleSubmit = useCallback(() => {
|
||||||
const cleanData = {
|
const cleanData = {
|
||||||
...data,
|
...data,
|
||||||
type: 'kanban',
|
|
||||||
name: data.name.trim(),
|
name: data.name.trim(),
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -14,7 +14,6 @@ const Item = React.memo(({ type, data, createdAt, user }) => {
|
||||||
const [t] = useTranslation();
|
const [t] = useTranslation();
|
||||||
|
|
||||||
let contentNode;
|
let contentNode;
|
||||||
|
|
||||||
switch (type) {
|
switch (type) {
|
||||||
case ActivityTypes.CREATE_CARD:
|
case ActivityTypes.CREATE_CARD:
|
||||||
contentNode = (
|
contentNode = (
|
||||||
|
|
|
@ -120,6 +120,7 @@ const Attachments = React.memo(
|
||||||
withCaption
|
withCaption
|
||||||
withDownloadButton
|
withDownloadButton
|
||||||
options={{
|
options={{
|
||||||
|
wheelToZoom: true,
|
||||||
showHideAnimationType: 'none',
|
showHideAnimationType: 'none',
|
||||||
closeTitle: '',
|
closeTitle: '',
|
||||||
zoomTitle: '',
|
zoomTitle: '',
|
||||||
|
|
|
@ -9,6 +9,7 @@
|
||||||
text-align: left;
|
text-align: left;
|
||||||
text-overflow: ellipsis;
|
text-overflow: ellipsis;
|
||||||
transition: background 85ms ease;
|
transition: background 85ms ease;
|
||||||
|
white-space: nowrap;
|
||||||
|
|
||||||
&:hover {
|
&:hover {
|
||||||
background: #dfe3e6;
|
background: #dfe3e6;
|
||||||
|
|
|
@ -1,41 +0,0 @@
|
||||||
import React from 'react';
|
|
||||||
import PropTypes from 'prop-types';
|
|
||||||
|
|
||||||
import ModalTypes from '../constants/ModalTypes';
|
|
||||||
import FixedContainer from '../containers/FixedContainer';
|
|
||||||
import StaticContainer from '../containers/StaticContainer';
|
|
||||||
import UsersModalContainer from '../containers/UsersModalContainer';
|
|
||||||
import UserSettingsModalContainer from '../containers/UserSettingsModalContainer';
|
|
||||||
import ProjectAddModalContainer from '../containers/ProjectAddModalContainer';
|
|
||||||
import Background from './Background';
|
|
||||||
|
|
||||||
function Core({ currentModal, currentProject }) {
|
|
||||||
return (
|
|
||||||
<>
|
|
||||||
{currentProject && currentProject.background && (
|
|
||||||
<Background
|
|
||||||
type={currentProject.background.type}
|
|
||||||
name={currentProject.background.name}
|
|
||||||
imageUrl={currentProject.backgroundImage && currentProject.backgroundImage.url}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
<FixedContainer />
|
|
||||||
<StaticContainer />
|
|
||||||
{currentModal === ModalTypes.USERS && <UsersModalContainer />}
|
|
||||||
{currentModal === ModalTypes.USER_SETTINGS && <UserSettingsModalContainer />}
|
|
||||||
{currentModal === ModalTypes.PROJECT_ADD && <ProjectAddModalContainer />}
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Core.propTypes = {
|
|
||||||
currentModal: PropTypes.oneOf(Object.values(ModalTypes)),
|
|
||||||
currentProject: PropTypes.object, // eslint-disable-line react/forbid-prop-types
|
|
||||||
};
|
|
||||||
|
|
||||||
Core.defaultProps = {
|
|
||||||
currentModal: undefined,
|
|
||||||
currentProject: undefined,
|
|
||||||
};
|
|
||||||
|
|
||||||
export default Core;
|
|
69
client/src/components/Core/Core.jsx
Executable file
69
client/src/components/Core/Core.jsx
Executable file
|
@ -0,0 +1,69 @@
|
||||||
|
import React from 'react';
|
||||||
|
import PropTypes from 'prop-types';
|
||||||
|
import { useTranslation, Trans } from 'react-i18next';
|
||||||
|
import { Loader } from 'semantic-ui-react';
|
||||||
|
|
||||||
|
import ModalTypes from '../../constants/ModalTypes';
|
||||||
|
import FixedContainer from '../../containers/FixedContainer';
|
||||||
|
import StaticContainer from '../../containers/StaticContainer';
|
||||||
|
import UsersModalContainer from '../../containers/UsersModalContainer';
|
||||||
|
import UserSettingsModalContainer from '../../containers/UserSettingsModalContainer';
|
||||||
|
import ProjectAddModalContainer from '../../containers/ProjectAddModalContainer';
|
||||||
|
import Background from '../Background';
|
||||||
|
|
||||||
|
import styles from './Core.module.scss';
|
||||||
|
|
||||||
|
const Core = React.memo(
|
||||||
|
({ isInitializing, isSocketDisconnected, currentModal, currentProject }) => {
|
||||||
|
const [t] = useTranslation();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
{isInitializing ? (
|
||||||
|
<Loader active size="massive" />
|
||||||
|
) : (
|
||||||
|
<>
|
||||||
|
{currentProject && currentProject.background && (
|
||||||
|
<Background
|
||||||
|
type={currentProject.background.type}
|
||||||
|
name={currentProject.background.name}
|
||||||
|
imageUrl={currentProject.backgroundImage && currentProject.backgroundImage.url}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
<FixedContainer />
|
||||||
|
<StaticContainer />
|
||||||
|
{currentModal === ModalTypes.USERS && <UsersModalContainer />}
|
||||||
|
{currentModal === ModalTypes.USER_SETTINGS && <UserSettingsModalContainer />}
|
||||||
|
{currentModal === ModalTypes.PROJECT_ADD && <ProjectAddModalContainer />}
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
{isSocketDisconnected && (
|
||||||
|
<div className={styles.message}>
|
||||||
|
<div className={styles.messageHeader}>{t('common.noConnectionToServer')}</div>
|
||||||
|
<div className={styles.messageContent}>
|
||||||
|
<Trans i18nKey="common.allChangesWillBeAutomaticallySavedAfterConnectionRestored">
|
||||||
|
All changes will be automatically saved
|
||||||
|
<br />
|
||||||
|
after connection restored
|
||||||
|
</Trans>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
Core.propTypes = {
|
||||||
|
isInitializing: PropTypes.bool.isRequired,
|
||||||
|
isSocketDisconnected: PropTypes.bool.isRequired,
|
||||||
|
currentModal: PropTypes.oneOf(Object.values(ModalTypes)),
|
||||||
|
currentProject: PropTypes.object, // eslint-disable-line react/forbid-prop-types
|
||||||
|
};
|
||||||
|
|
||||||
|
Core.defaultProps = {
|
||||||
|
currentModal: undefined,
|
||||||
|
currentProject: undefined,
|
||||||
|
};
|
||||||
|
|
||||||
|
export default Core;
|
3
client/src/components/Core/index.js
Normal file
3
client/src/components/Core/index.js
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
import Core from './Core';
|
||||||
|
|
||||||
|
export default Core;
|
|
@ -1,37 +0,0 @@
|
||||||
import React from 'react';
|
|
||||||
import PropTypes from 'prop-types';
|
|
||||||
import { useTranslation, Trans } from 'react-i18next';
|
|
||||||
import { Loader } from 'semantic-ui-react';
|
|
||||||
|
|
||||||
import CoreContainer from '../../containers/CoreContainer';
|
|
||||||
|
|
||||||
import styles from './CoreWrapper.module.scss';
|
|
||||||
|
|
||||||
const CoreWrapper = React.memo(({ isInitializing, isSocketDisconnected }) => {
|
|
||||||
const [t] = useTranslation();
|
|
||||||
|
|
||||||
return (
|
|
||||||
<>
|
|
||||||
{isInitializing ? <Loader active size="massive" /> : <CoreContainer />}
|
|
||||||
{isSocketDisconnected && (
|
|
||||||
<div className={styles.message}>
|
|
||||||
<div className={styles.messageHeader}>{t('common.noConnectionToServer')}</div>
|
|
||||||
<div className={styles.messageContent}>
|
|
||||||
<Trans i18nKey="common.allChangesWillBeAutomaticallySavedAfterConnectionRestored">
|
|
||||||
All changes will be automatically saved
|
|
||||||
<br />
|
|
||||||
after connection restored
|
|
||||||
</Trans>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
CoreWrapper.propTypes = {
|
|
||||||
isInitializing: PropTypes.bool.isRequired,
|
|
||||||
isSocketDisconnected: PropTypes.bool.isRequired,
|
|
||||||
};
|
|
||||||
|
|
||||||
export default CoreWrapper;
|
|
|
@ -1,3 +0,0 @@
|
||||||
import CoreWrapper from './CoreWrapper';
|
|
||||||
|
|
||||||
export default CoreWrapper;
|
|
|
@ -28,8 +28,8 @@ const ActionsStep = React.memo(
|
||||||
deleteConfirmationTitle,
|
deleteConfirmationTitle,
|
||||||
deleteConfirmationContent,
|
deleteConfirmationContent,
|
||||||
deleteConfirmationButtonContent,
|
deleteConfirmationButtonContent,
|
||||||
canLeave,
|
|
||||||
canEdit,
|
canEdit,
|
||||||
|
canLeave,
|
||||||
onUpdate,
|
onUpdate,
|
||||||
onDelete,
|
onDelete,
|
||||||
onClose,
|
onClose,
|
||||||
|
@ -50,10 +50,8 @@ const ActionsStep = React.memo(
|
||||||
if (onUpdate) {
|
if (onUpdate) {
|
||||||
onUpdate(data);
|
onUpdate(data);
|
||||||
}
|
}
|
||||||
|
|
||||||
onClose();
|
|
||||||
},
|
},
|
||||||
[onUpdate, onClose],
|
[onUpdate],
|
||||||
);
|
);
|
||||||
|
|
||||||
if (step) {
|
if (step) {
|
||||||
|
@ -68,6 +66,7 @@ const ActionsStep = React.memo(
|
||||||
buttonContent="action.save"
|
buttonContent="action.save"
|
||||||
onSelect={handleRoleSelect}
|
onSelect={handleRoleSelect}
|
||||||
onBack={handleBack}
|
onBack={handleBack}
|
||||||
|
onClose={onClose}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -146,8 +145,8 @@ ActionsStep.propTypes = {
|
||||||
deleteConfirmationTitle: PropTypes.string,
|
deleteConfirmationTitle: PropTypes.string,
|
||||||
deleteConfirmationContent: PropTypes.string,
|
deleteConfirmationContent: PropTypes.string,
|
||||||
deleteConfirmationButtonContent: PropTypes.string,
|
deleteConfirmationButtonContent: PropTypes.string,
|
||||||
canLeave: PropTypes.bool.isRequired,
|
|
||||||
canEdit: PropTypes.bool.isRequired,
|
canEdit: PropTypes.bool.isRequired,
|
||||||
|
canLeave: PropTypes.bool.isRequired,
|
||||||
onUpdate: PropTypes.func,
|
onUpdate: PropTypes.func,
|
||||||
onDelete: PropTypes.func.isRequired,
|
onDelete: PropTypes.func.isRequired,
|
||||||
onClose: PropTypes.func.isRequired,
|
onClose: PropTypes.func.isRequired,
|
||||||
|
|
|
@ -56,10 +56,8 @@ const AddStep = React.memo(
|
||||||
userId: step.params.userId,
|
userId: step.params.userId,
|
||||||
...data,
|
...data,
|
||||||
});
|
});
|
||||||
|
|
||||||
onClose();
|
|
||||||
},
|
},
|
||||||
[onCreate, onClose, step],
|
[onCreate, step],
|
||||||
);
|
);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
|
@ -79,6 +77,7 @@ const AddStep = React.memo(
|
||||||
buttonContent="action.addMember"
|
buttonContent="action.addMember"
|
||||||
onSelect={handleRoleSelect}
|
onSelect={handleRoleSelect}
|
||||||
onBack={handleBack}
|
onBack={handleBack}
|
||||||
|
onClose={onClose}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
|
@ -44,8 +44,8 @@ const Memberships = React.memo(
|
||||||
deleteConfirmationTitle={deleteConfirmationTitle}
|
deleteConfirmationTitle={deleteConfirmationTitle}
|
||||||
deleteConfirmationContent={deleteConfirmationContent}
|
deleteConfirmationContent={deleteConfirmationContent}
|
||||||
deleteConfirmationButtonContent={deleteConfirmationButtonContent}
|
deleteConfirmationButtonContent={deleteConfirmationButtonContent}
|
||||||
canLeave={items.length > 1 || canLeaveIfLast}
|
|
||||||
canEdit={canEdit}
|
canEdit={canEdit}
|
||||||
|
canLeave={items.length > 1 || canLeaveIfLast}
|
||||||
onUpdate={(data) => onUpdate(item.id, data)}
|
onUpdate={(data) => onUpdate(item.id, data)}
|
||||||
onDelete={() => onDelete(item.id)}
|
onDelete={() => onDelete(item.id)}
|
||||||
>
|
>
|
||||||
|
|
|
@ -6,7 +6,7 @@ import { ReduxRouter } from '../lib/redux-router';
|
||||||
|
|
||||||
import Paths from '../constants/Paths';
|
import Paths from '../constants/Paths';
|
||||||
import LoginContainer from '../containers/LoginContainer';
|
import LoginContainer from '../containers/LoginContainer';
|
||||||
import CoreWrapperContainer from '../containers/CoreWrapperContainer';
|
import CoreContainer from '../containers/CoreContainer';
|
||||||
import NotFound from './NotFound';
|
import NotFound from './NotFound';
|
||||||
|
|
||||||
import 'react-datepicker/dist/react-datepicker.css';
|
import 'react-datepicker/dist/react-datepicker.css';
|
||||||
|
@ -22,10 +22,10 @@ function Root({ store, history }) {
|
||||||
<ReduxRouter history={history}>
|
<ReduxRouter history={history}>
|
||||||
<Routes>
|
<Routes>
|
||||||
<Route path={Paths.LOGIN} element={<LoginContainer />} />
|
<Route path={Paths.LOGIN} element={<LoginContainer />} />
|
||||||
<Route path={Paths.ROOT} element={<CoreWrapperContainer />} />
|
<Route path={Paths.ROOT} element={<CoreContainer />} />
|
||||||
<Route path={Paths.PROJECTS} element={<CoreWrapperContainer />} />
|
<Route path={Paths.PROJECTS} element={<CoreContainer />} />
|
||||||
<Route path={Paths.BOARDS} element={<CoreWrapperContainer />} />
|
<Route path={Paths.BOARDS} element={<CoreContainer />} />
|
||||||
<Route path={Paths.CARDS} element={<CoreWrapperContainer />} />
|
<Route path={Paths.CARDS} element={<CoreContainer />} />
|
||||||
<Route path="*" element={<NotFound />} />
|
<Route path="*" element={<NotFound />} />
|
||||||
</Routes>
|
</Routes>
|
||||||
</ReduxRouter>
|
</ReduxRouter>
|
||||||
|
|
|
@ -2,19 +2,19 @@ import React from 'react';
|
||||||
import PropTypes from 'prop-types';
|
import PropTypes from 'prop-types';
|
||||||
import classNames from 'classnames';
|
import classNames from 'classnames';
|
||||||
import { useTranslation, Trans } from 'react-i18next';
|
import { useTranslation, Trans } from 'react-i18next';
|
||||||
import { Icon } from 'semantic-ui-react';
|
import { Icon, Loader } from 'semantic-ui-react';
|
||||||
|
|
||||||
import ProjectsContainer from '../../containers/ProjectsContainer';
|
import ProjectsContainer from '../../containers/ProjectsContainer';
|
||||||
import BoardWrapperContainer from '../../containers/BoardWrapperContainer';
|
import BoardContainer from '../../containers/BoardContainer';
|
||||||
|
|
||||||
import styles from './Static.module.scss';
|
import styles from './Static.module.scss';
|
||||||
|
|
||||||
function Static({ cardId, boardId, projectId }) {
|
function Static({ projectId, cardId, board }) {
|
||||||
const [t] = useTranslation();
|
const [t] = useTranslation();
|
||||||
|
|
||||||
if (projectId === undefined) {
|
if (projectId === undefined) {
|
||||||
return (
|
return (
|
||||||
<div className={styles.root}>
|
<div className={styles.wrapper}>
|
||||||
<ProjectsContainer />
|
<ProjectsContainer />
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
@ -22,7 +22,7 @@ function Static({ cardId, boardId, projectId }) {
|
||||||
|
|
||||||
if (cardId === null) {
|
if (cardId === null) {
|
||||||
return (
|
return (
|
||||||
<div className={classNames(styles.root, styles.flex)}>
|
<div className={classNames(styles.wrapper, styles.wrapperFlex)}>
|
||||||
<div className={styles.message}>
|
<div className={styles.message}>
|
||||||
<h1>
|
<h1>
|
||||||
{t('common.cardNotFound', {
|
{t('common.cardNotFound', {
|
||||||
|
@ -34,9 +34,9 @@ function Static({ cardId, boardId, projectId }) {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (boardId === null) {
|
if (board === null) {
|
||||||
return (
|
return (
|
||||||
<div className={classNames(styles.root, styles.flex)}>
|
<div className={classNames(styles.wrapper, styles.wrapperFlex)}>
|
||||||
<div className={styles.message}>
|
<div className={styles.message}>
|
||||||
<h1>
|
<h1>
|
||||||
{t('common.boardNotFound', {
|
{t('common.boardNotFound', {
|
||||||
|
@ -50,7 +50,7 @@ function Static({ cardId, boardId, projectId }) {
|
||||||
|
|
||||||
if (projectId === null) {
|
if (projectId === null) {
|
||||||
return (
|
return (
|
||||||
<div className={classNames(styles.root, styles.flex)}>
|
<div className={classNames(styles.wrapper, styles.wrapperFlex)}>
|
||||||
<div className={styles.message}>
|
<div className={styles.message}>
|
||||||
<h1>
|
<h1>
|
||||||
{t('common.projectNotFound', {
|
{t('common.projectNotFound', {
|
||||||
|
@ -62,9 +62,9 @@ function Static({ cardId, boardId, projectId }) {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (boardId === undefined) {
|
if (board === undefined) {
|
||||||
return (
|
return (
|
||||||
<div className={classNames(styles.board, styles.flex)}>
|
<div className={classNames(styles.wrapper, styles.wrapperFlex, styles.wrapperProject)}>
|
||||||
<div className={styles.message}>
|
<div className={styles.message}>
|
||||||
<Icon inverted name="hand point up outline" size="huge" className={styles.messageIcon} />
|
<Icon inverted name="hand point up outline" size="huge" className={styles.messageIcon} />
|
||||||
<h1 className={styles.messageTitle}>
|
<h1 className={styles.messageTitle}>
|
||||||
|
@ -80,23 +80,31 @@ function Static({ cardId, boardId, projectId }) {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (board.isFetching) {
|
||||||
|
return (
|
||||||
|
<div className={classNames(styles.wrapper, styles.wrapperLoader, styles.wrapperProject)}>
|
||||||
|
<Loader active size="big" />
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className={classNames(styles.board, styles.flex)}>
|
<div className={classNames(styles.wrapper, styles.wrapperFlex, styles.wrapperBoard)}>
|
||||||
<BoardWrapperContainer />
|
<BoardContainer />
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
Static.propTypes = {
|
Static.propTypes = {
|
||||||
cardId: PropTypes.string,
|
|
||||||
boardId: PropTypes.string,
|
|
||||||
projectId: PropTypes.string,
|
projectId: PropTypes.string,
|
||||||
|
cardId: PropTypes.string,
|
||||||
|
board: PropTypes.object, // eslint-disable-line react/forbid-prop-types
|
||||||
};
|
};
|
||||||
|
|
||||||
Static.defaultProps = {
|
Static.defaultProps = {
|
||||||
cardId: undefined,
|
|
||||||
boardId: undefined,
|
|
||||||
projectId: undefined,
|
projectId: undefined,
|
||||||
|
cardId: undefined,
|
||||||
|
board: undefined,
|
||||||
};
|
};
|
||||||
|
|
||||||
export default Static;
|
export default Static;
|
||||||
|
|
|
@ -1,13 +1,4 @@
|
||||||
:global(#app) {
|
:global(#app) {
|
||||||
.board {
|
|
||||||
margin-top: 174px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.flex {
|
|
||||||
display: flex;
|
|
||||||
height: 100%;
|
|
||||||
}
|
|
||||||
|
|
||||||
.message {
|
.message {
|
||||||
align-content: space-between;
|
align-content: space-between;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
|
@ -34,7 +25,24 @@
|
||||||
text-align: center;
|
text-align: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
.root {
|
.wrapper {
|
||||||
|
height: 100%;
|
||||||
margin-top: 50px;
|
margin-top: 50px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.wrapperBoard {
|
||||||
|
margin-top: 174px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.wrapperFlex {
|
||||||
|
display: flex;
|
||||||
|
}
|
||||||
|
|
||||||
|
.wrapperLoader {
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
.wrapperProject {
|
||||||
|
margin-top: 98px;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,4 +1,3 @@
|
||||||
import omit from 'lodash/omit';
|
|
||||||
import isEmail from 'validator/lib/isEmail';
|
import isEmail from 'validator/lib/isEmail';
|
||||||
import React, { useCallback, useEffect, useMemo, useRef } from 'react';
|
import React, { useCallback, useEffect, useMemo, useRef } from 'react';
|
||||||
import PropTypes from 'prop-types';
|
import PropTypes from 'prop-types';
|
||||||
|
@ -78,12 +77,16 @@ const UserEmailEditStep = React.memo(
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (usePasswordConfirmation && !cleanData.currentPassword) {
|
if (usePasswordConfirmation) {
|
||||||
currentPasswordField.current.focus();
|
if (!cleanData.currentPassword) {
|
||||||
return;
|
currentPasswordField.current.focus();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
delete cleanData.currentPassword;
|
||||||
}
|
}
|
||||||
|
|
||||||
onUpdate(usePasswordConfirmation ? cleanData : omit(cleanData, 'currentPassword'));
|
onUpdate(cleanData);
|
||||||
}, [email, usePasswordConfirmation, onUpdate, onClose, data]);
|
}, [email, usePasswordConfirmation, onUpdate, onClose, data]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
|
|
|
@ -1,4 +1,3 @@
|
||||||
import omit from 'lodash/omit';
|
|
||||||
import React, { useCallback, useEffect, useMemo, useRef } from 'react';
|
import React, { useCallback, useEffect, useMemo, useRef } from 'react';
|
||||||
import PropTypes from 'prop-types';
|
import PropTypes from 'prop-types';
|
||||||
import { useTranslation } from 'react-i18next';
|
import { useTranslation } from 'react-i18next';
|
||||||
|
@ -78,12 +77,16 @@ const UserUsernameEditStep = React.memo(
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (usePasswordConfirmation && !cleanData.currentPassword) {
|
if (usePasswordConfirmation) {
|
||||||
currentPasswordField.current.focus();
|
if (!cleanData.currentPassword) {
|
||||||
return;
|
currentPasswordField.current.focus();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
delete cleanData.currentPassword;
|
||||||
}
|
}
|
||||||
|
|
||||||
onUpdate(usePasswordConfirmation ? cleanData : omit(cleanData, 'currentPassword'));
|
onUpdate(cleanData);
|
||||||
}, [username, usePasswordConfirmation, onUpdate, onClose, data]);
|
}, [username, usePasswordConfirmation, onUpdate, onClose, data]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
|
|
|
@ -3,10 +3,6 @@ export const ProjectBackgroundTypes = {
|
||||||
IMAGE: 'image',
|
IMAGE: 'image',
|
||||||
};
|
};
|
||||||
|
|
||||||
export const BoardTypes = {
|
|
||||||
KANBAN: 'kanban',
|
|
||||||
};
|
|
||||||
|
|
||||||
export const BoardMembershipRoles = {
|
export const BoardMembershipRoles = {
|
||||||
EDITOR: 'editor',
|
EDITOR: 'editor',
|
||||||
VIEWER: 'viewer',
|
VIEWER: 'viewer',
|
||||||
|
|
|
@ -15,13 +15,16 @@ const mapStateToProps = (state) => {
|
||||||
const filterLabels = selectors.selectFilterLabelsForCurrentBoard(state);
|
const filterLabels = selectors.selectFilterLabelsForCurrentBoard(state);
|
||||||
const currentUserMembership = selectors.selectCurrentUserMembershipForCurrentBoard(state);
|
const currentUserMembership = selectors.selectCurrentUserMembershipForCurrentBoard(state);
|
||||||
|
|
||||||
|
const isCurrentUserEditor =
|
||||||
|
!!currentUserMembership && currentUserMembership.role === BoardMembershipRoles.EDITOR;
|
||||||
|
|
||||||
return {
|
return {
|
||||||
memberships,
|
memberships,
|
||||||
labels,
|
labels,
|
||||||
filterUsers,
|
filterUsers,
|
||||||
filterLabels,
|
filterLabels,
|
||||||
allUsers,
|
allUsers,
|
||||||
canEdit: !!currentUserMembership && currentUserMembership.role === BoardMembershipRoles.EDITOR,
|
canEdit: isCurrentUserEditor,
|
||||||
canEditMemberships: isCurrentUserManager,
|
canEditMemberships: isCurrentUserManager,
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
|
@ -4,17 +4,20 @@ import { connect } from 'react-redux';
|
||||||
import selectors from '../selectors';
|
import selectors from '../selectors';
|
||||||
import entryActions from '../entry-actions';
|
import entryActions from '../entry-actions';
|
||||||
import { BoardMembershipRoles } from '../constants/Enums';
|
import { BoardMembershipRoles } from '../constants/Enums';
|
||||||
import BoardKanban from '../components/BoardKanban';
|
import Board from '../components/Board';
|
||||||
|
|
||||||
const mapStateToProps = (state) => {
|
const mapStateToProps = (state) => {
|
||||||
const { cardId } = selectors.selectPath(state);
|
const { cardId } = selectors.selectPath(state);
|
||||||
const currentUserMembership = selectors.selectCurrentUserMembershipForCurrentBoard(state);
|
const currentUserMembership = selectors.selectCurrentUserMembershipForCurrentBoard(state);
|
||||||
const listIds = selectors.selectListIdsForCurrentBoard(state);
|
const listIds = selectors.selectListIdsForCurrentBoard(state);
|
||||||
|
|
||||||
|
const isCurrentUserEditor =
|
||||||
|
!!currentUserMembership && currentUserMembership.role === BoardMembershipRoles.EDITOR;
|
||||||
|
|
||||||
return {
|
return {
|
||||||
listIds,
|
listIds,
|
||||||
isCardModalOpened: !!cardId,
|
isCardModalOpened: !!cardId,
|
||||||
canEdit: !!currentUserMembership && currentUserMembership.role === BoardMembershipRoles.EDITOR,
|
canEdit: isCurrentUserEditor,
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -28,4 +31,4 @@ const mapDispatchToProps = (dispatch) =>
|
||||||
dispatch,
|
dispatch,
|
||||||
);
|
);
|
||||||
|
|
||||||
export default connect(mapStateToProps, mapDispatchToProps)(BoardKanban);
|
export default connect(mapStateToProps, mapDispatchToProps)(Board);
|
|
@ -1,15 +0,0 @@
|
||||||
import { connect } from 'react-redux';
|
|
||||||
|
|
||||||
import selectors from '../selectors';
|
|
||||||
import BoardWrapper from '../components/BoardWrapper';
|
|
||||||
|
|
||||||
const mapStateToProps = (state) => {
|
|
||||||
const { type, isFetching } = selectors.selectCurrentBoard(state);
|
|
||||||
|
|
||||||
return {
|
|
||||||
type,
|
|
||||||
isFetching,
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
export default connect(mapStateToProps)(BoardWrapper);
|
|
|
@ -30,6 +30,9 @@ const makeMapStateToProps = () => {
|
||||||
const tasks = selectTasksByCardId(state, id);
|
const tasks = selectTasksByCardId(state, id);
|
||||||
const notificationsTotal = selectNotificationsTotalByCardId(state, id);
|
const notificationsTotal = selectNotificationsTotalByCardId(state, id);
|
||||||
|
|
||||||
|
const isCurrentUserEditor =
|
||||||
|
!!currentUserMembership && currentUserMembership.role === BoardMembershipRoles.EDITOR;
|
||||||
|
|
||||||
return {
|
return {
|
||||||
id,
|
id,
|
||||||
index,
|
index,
|
||||||
|
@ -48,8 +51,7 @@ const makeMapStateToProps = () => {
|
||||||
allProjectsToLists,
|
allProjectsToLists,
|
||||||
allBoardMemberships,
|
allBoardMemberships,
|
||||||
allLabels,
|
allLabels,
|
||||||
canEdit:
|
canEdit: isCurrentUserEditor,
|
||||||
!!currentUserMembership && currentUserMembership.role === BoardMembershipRoles.EDITOR,
|
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
|
@ -37,6 +37,14 @@ const mapStateToProps = (state) => {
|
||||||
const attachments = selectors.selectAttachmentsForCurrentCard(state);
|
const attachments = selectors.selectAttachmentsForCurrentCard(state);
|
||||||
const activities = selectors.selectActivitiesForCurrentCard(state);
|
const activities = selectors.selectActivitiesForCurrentCard(state);
|
||||||
|
|
||||||
|
let isCurrentUserEditor = false;
|
||||||
|
let isCurrentUserEditorOrCanComment = false;
|
||||||
|
|
||||||
|
if (currentUserMembership) {
|
||||||
|
isCurrentUserEditor = currentUserMembership.role === BoardMembershipRoles.EDITOR;
|
||||||
|
isCurrentUserEditorOrCanComment = isCurrentUserEditor || currentUserMembership.canComment;
|
||||||
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
name,
|
name,
|
||||||
description,
|
description,
|
||||||
|
@ -58,11 +66,8 @@ const mapStateToProps = (state) => {
|
||||||
allProjectsToLists,
|
allProjectsToLists,
|
||||||
allBoardMemberships,
|
allBoardMemberships,
|
||||||
allLabels,
|
allLabels,
|
||||||
canEdit: !!currentUserMembership && currentUserMembership.role === BoardMembershipRoles.EDITOR,
|
canEdit: isCurrentUserEditor,
|
||||||
canEditCommentActivities:
|
canEditCommentActivities: isCurrentUserEditorOrCanComment,
|
||||||
!!currentUserMembership &&
|
|
||||||
(currentUserMembership.role === BoardMembershipRoles.EDITOR ||
|
|
||||||
currentUserMembership.canComment),
|
|
||||||
canEditAllCommentActivities: isCurrentUserManager,
|
canEditAllCommentActivities: isCurrentUserManager,
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
|
@ -4,12 +4,16 @@ import selectors from '../selectors';
|
||||||
import Core from '../components/Core';
|
import Core from '../components/Core';
|
||||||
|
|
||||||
const mapStateToProps = (state) => {
|
const mapStateToProps = (state) => {
|
||||||
|
const isCoreInitializing = selectors.selectIsCoreInitializing(state);
|
||||||
|
const isSocketDisconnected = selectors.selectIsSocketDisconnected(state);
|
||||||
const currentModal = selectors.selectCurrentModal(state);
|
const currentModal = selectors.selectCurrentModal(state);
|
||||||
const currentProject = selectors.selectCurrentProject(state);
|
const currentProject = selectors.selectCurrentProject(state);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
|
isSocketDisconnected,
|
||||||
currentModal,
|
currentModal,
|
||||||
currentProject,
|
currentProject,
|
||||||
|
isInitializing: isCoreInitializing,
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -1,15 +0,0 @@
|
||||||
import { connect } from 'react-redux';
|
|
||||||
|
|
||||||
import selectors from '../selectors';
|
|
||||||
import CoreWrapper from '../components/CoreWrapper';
|
|
||||||
|
|
||||||
const mapStateToProps = (state) => {
|
|
||||||
const isCoreInitializing = selectors.selectIsCoreInitializing(state);
|
|
||||||
|
|
||||||
return {
|
|
||||||
isInitializing: isCoreInitializing,
|
|
||||||
isSocketDisconnected: state.socket.isDisconnected,
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
export default connect(mapStateToProps)(CoreWrapper);
|
|
|
@ -15,14 +15,16 @@ const makeMapStateToProps = () => {
|
||||||
const cardIds = selectCardIdsByListId(state, id);
|
const cardIds = selectCardIdsByListId(state, id);
|
||||||
const currentUserMembership = selectors.selectCurrentUserMembershipForCurrentBoard(state);
|
const currentUserMembership = selectors.selectCurrentUserMembershipForCurrentBoard(state);
|
||||||
|
|
||||||
|
const isCurrentUserEditor =
|
||||||
|
!!currentUserMembership && currentUserMembership.role === BoardMembershipRoles.EDITOR;
|
||||||
|
|
||||||
return {
|
return {
|
||||||
id,
|
id,
|
||||||
index,
|
index,
|
||||||
name,
|
name,
|
||||||
isPersisted,
|
isPersisted,
|
||||||
cardIds,
|
cardIds,
|
||||||
canEdit:
|
canEdit: isCurrentUserEditor,
|
||||||
!!currentUserMembership && currentUserMembership.role === BoardMembershipRoles.EDITOR,
|
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
|
@ -4,12 +4,13 @@ import selectors from '../selectors';
|
||||||
import Static from '../components/Static';
|
import Static from '../components/Static';
|
||||||
|
|
||||||
const mapStateToProps = (state) => {
|
const mapStateToProps = (state) => {
|
||||||
const { cardId, boardId, projectId } = selectors.selectPath(state);
|
const { cardId, projectId } = selectors.selectPath(state);
|
||||||
|
const currentBoard = selectors.selectCurrentBoard(state);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
cardId,
|
|
||||||
boardId,
|
|
||||||
projectId,
|
projectId,
|
||||||
|
cardId,
|
||||||
|
board: currentBoard,
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -6,7 +6,7 @@ import { Button, Popup as SemanticUIPopup } from 'semantic-ui-react';
|
||||||
import styles from './Popup.module.css';
|
import styles from './Popup.module.css';
|
||||||
|
|
||||||
export default (WrappedComponent, defaultProps) => {
|
export default (WrappedComponent, defaultProps) => {
|
||||||
const Popup = React.memo(({ children, ...props }) => {
|
const Popup = React.memo(({ children, onClose, ...props }) => {
|
||||||
const [isOpened, setIsOpened] = useState(false);
|
const [isOpened, setIsOpened] = useState(false);
|
||||||
|
|
||||||
const wrapper = useRef(null);
|
const wrapper = useRef(null);
|
||||||
|
@ -18,7 +18,11 @@ export default (WrappedComponent, defaultProps) => {
|
||||||
|
|
||||||
const handleClose = useCallback(() => {
|
const handleClose = useCallback(() => {
|
||||||
setIsOpened(false);
|
setIsOpened(false);
|
||||||
}, []);
|
|
||||||
|
if (onClose) {
|
||||||
|
onClose();
|
||||||
|
}
|
||||||
|
}, [onClose]);
|
||||||
|
|
||||||
const handleMouseDown = useCallback((event) => {
|
const handleMouseDown = useCallback((event) => {
|
||||||
event.stopPropagation();
|
event.stopPropagation();
|
||||||
|
@ -105,6 +109,11 @@ export default (WrappedComponent, defaultProps) => {
|
||||||
|
|
||||||
Popup.propTypes = {
|
Popup.propTypes = {
|
||||||
children: PropTypes.node.isRequired,
|
children: PropTypes.node.isRequired,
|
||||||
|
onClose: PropTypes.func,
|
||||||
|
};
|
||||||
|
|
||||||
|
Popup.defaultProps = {
|
||||||
|
onClose: undefined,
|
||||||
};
|
};
|
||||||
|
|
||||||
return Popup;
|
return Popup;
|
||||||
|
|
|
@ -1,8 +1,9 @@
|
||||||
import { Model, attr, fk } from 'redux-orm';
|
import { attr, fk } from 'redux-orm';
|
||||||
|
|
||||||
|
import BaseModel from './BaseModel';
|
||||||
import ActionTypes from '../constants/ActionTypes';
|
import ActionTypes from '../constants/ActionTypes';
|
||||||
|
|
||||||
export default class extends Model {
|
export default class extends BaseModel {
|
||||||
static modelName = 'Activity';
|
static modelName = 'Activity';
|
||||||
|
|
||||||
static fields = {
|
static fields = {
|
||||||
|
|
|
@ -1,8 +1,9 @@
|
||||||
import { Model, attr, fk } from 'redux-orm';
|
import { attr, fk } from 'redux-orm';
|
||||||
|
|
||||||
|
import BaseModel from './BaseModel';
|
||||||
import ActionTypes from '../constants/ActionTypes';
|
import ActionTypes from '../constants/ActionTypes';
|
||||||
|
|
||||||
export default class extends Model {
|
export default class extends BaseModel {
|
||||||
static modelName = 'Attachment';
|
static modelName = 'Attachment';
|
||||||
|
|
||||||
static fields = {
|
static fields = {
|
||||||
|
|
6
client/src/models/BaseModel.js
Normal file
6
client/src/models/BaseModel.js
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
import { Model } from 'redux-orm';
|
||||||
|
|
||||||
|
export default class BaseModel extends Model {
|
||||||
|
// eslint-disable-next-line no-underscore-dangle, class-methods-use-this
|
||||||
|
_onDelete() {}
|
||||||
|
}
|
|
@ -1,13 +1,13 @@
|
||||||
import { Model, attr, fk, many } from 'redux-orm';
|
import { attr, fk, many } from 'redux-orm';
|
||||||
|
|
||||||
|
import BaseModel from './BaseModel';
|
||||||
import ActionTypes from '../constants/ActionTypes';
|
import ActionTypes from '../constants/ActionTypes';
|
||||||
|
|
||||||
export default class extends Model {
|
export default class extends BaseModel {
|
||||||
static modelName = 'Board';
|
static modelName = 'Board';
|
||||||
|
|
||||||
static fields = {
|
static fields = {
|
||||||
id: attr(),
|
id: attr(),
|
||||||
type: attr(),
|
|
||||||
position: attr(),
|
position: attr(),
|
||||||
name: attr(),
|
name: attr(),
|
||||||
isFetching: attr({
|
isFetching: attr({
|
||||||
|
@ -179,7 +179,7 @@ export default class extends Model {
|
||||||
return this.lists.orderBy('position');
|
return this.lists.orderBy('position');
|
||||||
}
|
}
|
||||||
|
|
||||||
getMembershipModel(userId) {
|
getMembershipModelForUser(userId) {
|
||||||
return this.memberships
|
return this.memberships
|
||||||
.filter({
|
.filter({
|
||||||
userId,
|
userId,
|
||||||
|
@ -187,7 +187,7 @@ export default class extends Model {
|
||||||
.first();
|
.first();
|
||||||
}
|
}
|
||||||
|
|
||||||
hasMemberUser(userId) {
|
hasMembershipForUser(userId) {
|
||||||
return this.memberships
|
return this.memberships
|
||||||
.filter({
|
.filter({
|
||||||
userId,
|
userId,
|
||||||
|
@ -195,6 +195,12 @@ export default class extends Model {
|
||||||
.exists();
|
.exists();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
isAvailableForUser(userId) {
|
||||||
|
return (
|
||||||
|
this.project && (this.project.hasManagerForUser(userId) || this.hasMembershipForUser(userId))
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
deleteRelated(exceptMemberUserId) {
|
deleteRelated(exceptMemberUserId) {
|
||||||
this.memberships.toModelArray().forEach((boardMembershipModel) => {
|
this.memberships.toModelArray().forEach((boardMembershipModel) => {
|
||||||
if (boardMembershipModel.userId !== exceptMemberUserId) {
|
if (boardMembershipModel.userId !== exceptMemberUserId) {
|
||||||
|
|
|
@ -1,8 +1,9 @@
|
||||||
import { Model, attr, fk } from 'redux-orm';
|
import { attr, fk } from 'redux-orm';
|
||||||
|
|
||||||
|
import BaseModel from './BaseModel';
|
||||||
import ActionTypes from '../constants/ActionTypes';
|
import ActionTypes from '../constants/ActionTypes';
|
||||||
|
|
||||||
export default class extends Model {
|
export default class extends BaseModel {
|
||||||
static modelName = 'BoardMembership';
|
static modelName = 'BoardMembership';
|
||||||
|
|
||||||
static fields = {
|
static fields = {
|
||||||
|
|
|
@ -1,10 +1,11 @@
|
||||||
import { Model, attr, fk, many, oneToOne } from 'redux-orm';
|
import { attr, fk, many, oneToOne } from 'redux-orm';
|
||||||
|
|
||||||
|
import BaseModel from './BaseModel';
|
||||||
import ActionTypes from '../constants/ActionTypes';
|
import ActionTypes from '../constants/ActionTypes';
|
||||||
import Config from '../constants/Config';
|
import Config from '../constants/Config';
|
||||||
import { ActivityTypes } from '../constants/Enums';
|
import { ActivityTypes } from '../constants/Enums';
|
||||||
|
|
||||||
export default class extends Model {
|
export default class extends BaseModel {
|
||||||
static modelName = 'Card';
|
static modelName = 'Card';
|
||||||
|
|
||||||
static fields = {
|
static fields = {
|
||||||
|
@ -74,7 +75,11 @@ export default class extends Model {
|
||||||
|
|
||||||
break;
|
break;
|
||||||
case ActionTypes.SOCKET_RECONNECT_HANDLE:
|
case ActionTypes.SOCKET_RECONNECT_HANDLE:
|
||||||
Card.all().delete();
|
Card.all()
|
||||||
|
.toModelArray()
|
||||||
|
.forEach((cardModel) => {
|
||||||
|
cardModel.deleteWithClearable();
|
||||||
|
});
|
||||||
|
|
||||||
if (payload.cards) {
|
if (payload.cards) {
|
||||||
payload.cards.forEach((card) => {
|
payload.cards.forEach((card) => {
|
||||||
|
@ -176,7 +181,7 @@ export default class extends Model {
|
||||||
|
|
||||||
break;
|
break;
|
||||||
case ActionTypes.CARD_DELETE:
|
case ActionTypes.CARD_DELETE:
|
||||||
Card.withId(payload.id).delete();
|
Card.withId(payload.id).deleteWithRelated();
|
||||||
|
|
||||||
break;
|
break;
|
||||||
case ActionTypes.CARD_DELETE__SUCCESS:
|
case ActionTypes.CARD_DELETE__SUCCESS:
|
||||||
|
@ -220,15 +225,7 @@ export default class extends Model {
|
||||||
isActivitiesDetailsFetching: false,
|
isActivitiesDetailsFetching: false,
|
||||||
});
|
});
|
||||||
|
|
||||||
cardModel.activities.toModelArray().forEach((activityModel) => {
|
cardModel.deleteActivities();
|
||||||
if (activityModel.notification) {
|
|
||||||
activityModel.update({
|
|
||||||
isInCard: false,
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
activityModel.delete();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -268,10 +265,37 @@ export default class extends Model {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
isAvailableForUser(userId) {
|
||||||
|
return this.board && this.board.isAvailableForUser(userId);
|
||||||
|
}
|
||||||
|
|
||||||
|
deleteClearable() {
|
||||||
|
this.users.clear();
|
||||||
|
this.labels.clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
deleteActivities() {
|
||||||
|
this.activities.toModelArray().forEach((activityModel) => {
|
||||||
|
if (activityModel.notification) {
|
||||||
|
activityModel.update({
|
||||||
|
isInCard: false,
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
activityModel.delete();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
deleteRelated() {
|
deleteRelated() {
|
||||||
|
this.deleteClearable();
|
||||||
this.tasks.delete();
|
this.tasks.delete();
|
||||||
this.attachments.delete();
|
this.attachments.delete();
|
||||||
this.activities.delete();
|
this.deleteActivities();
|
||||||
|
}
|
||||||
|
|
||||||
|
deleteWithClearable() {
|
||||||
|
this.deleteClearable();
|
||||||
|
this.delete();
|
||||||
}
|
}
|
||||||
|
|
||||||
deleteWithRelated() {
|
deleteWithRelated() {
|
||||||
|
|
|
@ -1,8 +1,9 @@
|
||||||
import { Model, attr, fk } from 'redux-orm';
|
import { attr, fk } from 'redux-orm';
|
||||||
|
|
||||||
|
import BaseModel from './BaseModel';
|
||||||
import ActionTypes from '../constants/ActionTypes';
|
import ActionTypes from '../constants/ActionTypes';
|
||||||
|
|
||||||
export default class extends Model {
|
export default class extends BaseModel {
|
||||||
static modelName = 'Label';
|
static modelName = 'Label';
|
||||||
|
|
||||||
static fields = {
|
static fields = {
|
||||||
|
|
|
@ -1,8 +1,9 @@
|
||||||
import { Model, attr, fk } from 'redux-orm';
|
import { attr, fk } from 'redux-orm';
|
||||||
|
|
||||||
|
import BaseModel from './BaseModel';
|
||||||
import ActionTypes from '../constants/ActionTypes';
|
import ActionTypes from '../constants/ActionTypes';
|
||||||
|
|
||||||
export default class extends Model {
|
export default class extends BaseModel {
|
||||||
static modelName = 'List';
|
static modelName = 'List';
|
||||||
|
|
||||||
static fields = {
|
static fields = {
|
||||||
|
|
|
@ -1,8 +1,9 @@
|
||||||
import { Model, attr, fk, oneToOne } from 'redux-orm';
|
import { attr, fk, oneToOne } from 'redux-orm';
|
||||||
|
|
||||||
|
import BaseModel from './BaseModel';
|
||||||
import ActionTypes from '../constants/ActionTypes';
|
import ActionTypes from '../constants/ActionTypes';
|
||||||
|
|
||||||
export default class extends Model {
|
export default class extends BaseModel {
|
||||||
static modelName = 'Notification';
|
static modelName = 'Notification';
|
||||||
|
|
||||||
static fields = {
|
static fields = {
|
||||||
|
|
|
@ -1,9 +1,10 @@
|
||||||
import { Model, attr, many } from 'redux-orm';
|
import { attr, many } from 'redux-orm';
|
||||||
|
|
||||||
|
import BaseModel from './BaseModel';
|
||||||
import ActionTypes from '../constants/ActionTypes';
|
import ActionTypes from '../constants/ActionTypes';
|
||||||
import { ProjectBackgroundTypes } from '../constants/Enums';
|
import { ProjectBackgroundTypes } from '../constants/Enums';
|
||||||
|
|
||||||
export default class extends Model {
|
export default class extends BaseModel {
|
||||||
static modelName = 'Project';
|
static modelName = 'Project';
|
||||||
|
|
||||||
static fields = {
|
static fields = {
|
||||||
|
@ -143,21 +144,21 @@ export default class extends Model {
|
||||||
return this.boards.orderBy('position');
|
return this.boards.orderBy('position');
|
||||||
}
|
}
|
||||||
|
|
||||||
getOrderedMemberBoardsModelArray(userId) {
|
getOrderedBoardsModelArrayForUser(userId) {
|
||||||
return this.getOrderedBoardsQuerySet()
|
return this.getOrderedBoardsQuerySet()
|
||||||
.toModelArray()
|
.toModelArray()
|
||||||
.filter((boardModel) => boardModel.hasMemberUser(userId));
|
.filter((boardModel) => boardModel.hasMembershipForUser(userId));
|
||||||
}
|
}
|
||||||
|
|
||||||
getOrderedAvailableBoardsModelArray(userId) {
|
getOrderedBoardsModelArrayAvailableForUser(userId) {
|
||||||
if (this.hasManagerUser(userId)) {
|
if (this.hasManagerForUser(userId)) {
|
||||||
return this.getOrderedBoardsQuerySet().toModelArray();
|
return this.getOrderedBoardsQuerySet().toModelArray();
|
||||||
}
|
}
|
||||||
|
|
||||||
return this.getOrderedMemberBoardsModelArray(userId);
|
return this.getOrderedBoardsModelArrayForUser(userId);
|
||||||
}
|
}
|
||||||
|
|
||||||
hasManagerUser(userId) {
|
hasManagerForUser(userId) {
|
||||||
return this.managers
|
return this.managers
|
||||||
.filter({
|
.filter({
|
||||||
userId,
|
userId,
|
||||||
|
@ -165,8 +166,12 @@ export default class extends Model {
|
||||||
.exists();
|
.exists();
|
||||||
}
|
}
|
||||||
|
|
||||||
hasMemberUserForAnyBoard(userId) {
|
hasMembershipInAnyBoardForUser(userId) {
|
||||||
return this.boards.toModelArray().some((boardModel) => boardModel.hasMemberUser(userId));
|
return this.boards.toModelArray().some((boardModel) => boardModel.hasMembershipForUser(userId));
|
||||||
|
}
|
||||||
|
|
||||||
|
isAvailableForUser(userId) {
|
||||||
|
return this.hasManagerForUser(userId) || this.hasMembershipInAnyBoardForUser(userId);
|
||||||
}
|
}
|
||||||
|
|
||||||
deleteRelated() {
|
deleteRelated() {
|
||||||
|
|
|
@ -1,8 +1,9 @@
|
||||||
import { Model, attr, fk } from 'redux-orm';
|
import { attr, fk } from 'redux-orm';
|
||||||
|
|
||||||
|
import BaseModel from './BaseModel';
|
||||||
import ActionTypes from '../constants/ActionTypes';
|
import ActionTypes from '../constants/ActionTypes';
|
||||||
|
|
||||||
export default class extends Model {
|
export default class extends BaseModel {
|
||||||
static modelName = 'ProjectManager';
|
static modelName = 'ProjectManager';
|
||||||
|
|
||||||
static fields = {
|
static fields = {
|
||||||
|
|
|
@ -1,8 +1,9 @@
|
||||||
import { Model, attr, fk } from 'redux-orm';
|
import { attr, fk } from 'redux-orm';
|
||||||
|
|
||||||
|
import BaseModel from './BaseModel';
|
||||||
import ActionTypes from '../constants/ActionTypes';
|
import ActionTypes from '../constants/ActionTypes';
|
||||||
|
|
||||||
export default class extends Model {
|
export default class extends BaseModel {
|
||||||
static modelName = 'Task';
|
static modelName = 'Task';
|
||||||
|
|
||||||
static fields = {
|
static fields = {
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
import { Model, attr } from 'redux-orm';
|
import { attr } from 'redux-orm';
|
||||||
|
|
||||||
|
import BaseModel from './BaseModel';
|
||||||
import ActionTypes from '../constants/ActionTypes';
|
import ActionTypes from '../constants/ActionTypes';
|
||||||
|
|
||||||
const DEFAULT_EMAIL_UPDATE_FORM = {
|
const DEFAULT_EMAIL_UPDATE_FORM = {
|
||||||
|
@ -29,7 +30,7 @@ const DEFAULT_USERNAME_UPDATE_FORM = {
|
||||||
error: null,
|
error: null,
|
||||||
};
|
};
|
||||||
|
|
||||||
export default class extends Model {
|
export default class extends BaseModel {
|
||||||
static modelName = 'User';
|
static modelName = 'User';
|
||||||
|
|
||||||
static fields = {
|
static fields = {
|
||||||
|
|
|
@ -12,7 +12,6 @@ export function* createCard(listId, data) {
|
||||||
|
|
||||||
const nextData = {
|
const nextData = {
|
||||||
...data,
|
...data,
|
||||||
listId,
|
|
||||||
position: yield select(selectors.selectNextCardPosition, listId),
|
position: yield select(selectors.selectNextCardPosition, listId),
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -22,13 +21,14 @@ export function* createCard(listId, data) {
|
||||||
actions.createCard({
|
actions.createCard({
|
||||||
...nextData,
|
...nextData,
|
||||||
boardId,
|
boardId,
|
||||||
|
listId,
|
||||||
id: localId,
|
id: localId,
|
||||||
}),
|
}),
|
||||||
);
|
);
|
||||||
|
|
||||||
let card;
|
let card;
|
||||||
try {
|
try {
|
||||||
({ item: card } = yield call(request, api.createCard, boardId, nextData));
|
({ item: card } = yield call(request, api.createCard, listId, nextData));
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
yield put(actions.createCard.failure(localId, error));
|
yield put(actions.createCard.failure(localId, error));
|
||||||
return;
|
return;
|
||||||
|
@ -61,8 +61,9 @@ export function* updateCurrentCard(data) {
|
||||||
yield call(updateCard, cardId, data);
|
yield call(updateCard, cardId, data);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO: handle card transfer
|
||||||
export function* handleCardUpdate(card) {
|
export function* handleCardUpdate(card) {
|
||||||
yield put(actions.handleCardUpdate(card)); // TODO: handle card transfer
|
yield put(actions.handleCardUpdate(card));
|
||||||
}
|
}
|
||||||
|
|
||||||
export function* moveCard(id, listId, index) {
|
export function* moveCard(id, listId, index) {
|
||||||
|
|
|
@ -39,8 +39,8 @@ export function* createListInCurrentBoard(data) {
|
||||||
yield call(createList, boardId, data);
|
yield call(createList, boardId, data);
|
||||||
}
|
}
|
||||||
|
|
||||||
export function* handleListCreate(label) {
|
export function* handleListCreate(list) {
|
||||||
yield put(actions.handleListCreate(label));
|
yield put(actions.handleListCreate(list));
|
||||||
}
|
}
|
||||||
|
|
||||||
export function* updateList(id, data) {
|
export function* updateList(id, data) {
|
||||||
|
@ -57,8 +57,8 @@ export function* updateList(id, data) {
|
||||||
yield put(actions.updateList.success(list));
|
yield put(actions.updateList.success(list));
|
||||||
}
|
}
|
||||||
|
|
||||||
export function* handleListUpdate(label) {
|
export function* handleListUpdate(list) {
|
||||||
yield put(actions.handleListUpdate(label));
|
yield put(actions.handleListUpdate(list));
|
||||||
}
|
}
|
||||||
|
|
||||||
export function* moveList(id, index) {
|
export function* moveList(id, index) {
|
||||||
|
@ -84,8 +84,8 @@ export function* deleteList(id) {
|
||||||
yield put(actions.deleteList.success(list));
|
yield put(actions.deleteList.success(list));
|
||||||
}
|
}
|
||||||
|
|
||||||
export function* handleListDelete(label) {
|
export function* handleListDelete(list) {
|
||||||
yield put(actions.handleListDelete(label));
|
yield put(actions.handleListDelete(list));
|
||||||
}
|
}
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
|
|
|
@ -69,6 +69,31 @@ export const selectMembershipsForCurrentBoard = createSelector(
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
|
||||||
|
export const selectCurrentUserMembershipForCurrentBoard = createSelector(
|
||||||
|
orm,
|
||||||
|
(state) => selectPath(state).boardId,
|
||||||
|
(state) => selectCurrentUserId(state),
|
||||||
|
({ Board }, id, currentUserId) => {
|
||||||
|
if (!id) {
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
|
||||||
|
const boardModel = Board.withId(id);
|
||||||
|
|
||||||
|
if (!boardModel) {
|
||||||
|
return boardModel;
|
||||||
|
}
|
||||||
|
|
||||||
|
const boardMembershipModel = boardModel.getMembershipModelForUser(currentUserId);
|
||||||
|
|
||||||
|
if (!boardMembershipModel) {
|
||||||
|
return boardMembershipModel;
|
||||||
|
}
|
||||||
|
|
||||||
|
return boardMembershipModel.ref;
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
export const selectLabelsForCurrentBoard = createSelector(
|
export const selectLabelsForCurrentBoard = createSelector(
|
||||||
orm,
|
orm,
|
||||||
(state) => selectPath(state).boardId,
|
(state) => selectPath(state).boardId,
|
||||||
|
@ -147,31 +172,6 @@ export const selectFilterLabelsForCurrentBoard = createSelector(
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
|
||||||
export const selectCurrentUserMembershipForCurrentBoard = createSelector(
|
|
||||||
orm,
|
|
||||||
(state) => selectPath(state).boardId,
|
|
||||||
(state) => selectCurrentUserId(state),
|
|
||||||
({ Board }, id, currentUserId) => {
|
|
||||||
if (!id) {
|
|
||||||
return id;
|
|
||||||
}
|
|
||||||
|
|
||||||
const boardModel = Board.withId(id);
|
|
||||||
|
|
||||||
if (!boardModel) {
|
|
||||||
return boardModel;
|
|
||||||
}
|
|
||||||
|
|
||||||
const boardMembershipModel = boardModel.getMembershipModel(currentUserId);
|
|
||||||
|
|
||||||
if (!boardMembershipModel) {
|
|
||||||
return boardMembershipModel;
|
|
||||||
}
|
|
||||||
|
|
||||||
return boardMembershipModel.ref;
|
|
||||||
},
|
|
||||||
);
|
|
||||||
|
|
||||||
export const selectIsBoardWithIdExists = createSelector(
|
export const selectIsBoardWithIdExists = createSelector(
|
||||||
orm,
|
orm,
|
||||||
(_, id) => id,
|
(_, id) => id,
|
||||||
|
@ -183,10 +183,10 @@ export default {
|
||||||
selectBoardById,
|
selectBoardById,
|
||||||
selectCurrentBoard,
|
selectCurrentBoard,
|
||||||
selectMembershipsForCurrentBoard,
|
selectMembershipsForCurrentBoard,
|
||||||
|
selectCurrentUserMembershipForCurrentBoard,
|
||||||
selectLabelsForCurrentBoard,
|
selectLabelsForCurrentBoard,
|
||||||
selectListIdsForCurrentBoard,
|
selectListIdsForCurrentBoard,
|
||||||
selectFilterUsersForCurrentBoard,
|
selectFilterUsersForCurrentBoard,
|
||||||
selectFilterLabelsForCurrentBoard,
|
selectFilterLabelsForCurrentBoard,
|
||||||
selectCurrentUserMembershipForCurrentBoard,
|
|
||||||
selectIsBoardWithIdExists,
|
selectIsBoardWithIdExists,
|
||||||
};
|
};
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
import router from './router';
|
import router from './router';
|
||||||
|
import socket from './socket';
|
||||||
import core from './core';
|
import core from './core';
|
||||||
import modals from './modals';
|
import modals from './modals';
|
||||||
import users from './users';
|
import users from './users';
|
||||||
|
@ -13,6 +14,7 @@ import attachments from './attachments';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
...router,
|
...router,
|
||||||
|
...socket,
|
||||||
...core,
|
...core,
|
||||||
...modals,
|
...modals,
|
||||||
...users,
|
...users,
|
||||||
|
|
|
@ -67,10 +67,12 @@ export const selectBoardsForCurrentProject = createSelector(
|
||||||
return projectModel;
|
return projectModel;
|
||||||
}
|
}
|
||||||
|
|
||||||
return projectModel.getOrderedAvailableBoardsModelArray(currentUserId).map((boardModel) => ({
|
return projectModel
|
||||||
...boardModel.ref,
|
.getOrderedBoardsModelArrayAvailableForUser(currentUserId)
|
||||||
isPersisted: !isLocalId(boardModel.id),
|
.map((boardModel) => ({
|
||||||
}));
|
...boardModel.ref,
|
||||||
|
isPersisted: !isLocalId(boardModel.id),
|
||||||
|
}));
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -89,7 +91,7 @@ export const selectIsCurrentUserManagerForCurrentProject = createSelector(
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
return projectModel.hasManagerUser(currentUserId);
|
return projectModel.hasManagerForUser(currentUserId);
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|
|
@ -26,55 +26,35 @@ export const selectPath = createReduxOrmSelector(
|
||||||
case Paths.PROJECTS: {
|
case Paths.PROJECTS: {
|
||||||
const projectModel = Project.withId(pathsMatch.params.id);
|
const projectModel = Project.withId(pathsMatch.params.id);
|
||||||
|
|
||||||
if (!projectModel) {
|
if (!projectModel || !projectModel.isAvailableForUser(currentUserId)) {
|
||||||
return {
|
return {
|
||||||
projectId: null,
|
projectId: null,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!projectModel.hasManagerUser(currentUserId)) {
|
|
||||||
if (!projectModel.hasMemberUserForAnyBoard(currentUserId)) {
|
|
||||||
return {
|
|
||||||
projectId: null,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
return {
|
||||||
projectId: projectModel.id,
|
projectId: projectModel.id,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
case Paths.BOARDS: {
|
case Paths.BOARDS: {
|
||||||
const boardModel = Board.withId(pathsMatch.params.id);
|
const boardModel = Board.withId(pathsMatch.params.id);
|
||||||
const projectModel = boardModel && boardModel.project;
|
|
||||||
|
|
||||||
if (!projectModel) {
|
if (!boardModel || !boardModel.isAvailableForUser(currentUserId)) {
|
||||||
return {
|
return {
|
||||||
boardId: null,
|
boardId: null,
|
||||||
projectId: null,
|
projectId: null,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!projectModel.hasManagerUser(currentUserId)) {
|
|
||||||
if (!boardModel.hasMemberUser(currentUserId)) {
|
|
||||||
return {
|
|
||||||
boardId: null,
|
|
||||||
projectId: null,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
return {
|
||||||
boardId: boardModel.id,
|
boardId: boardModel.id,
|
||||||
projectId: projectModel.id,
|
projectId: boardModel.projectId,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
case Paths.CARDS: {
|
case Paths.CARDS: {
|
||||||
const cardModel = Card.withId(pathsMatch.params.id);
|
const cardModel = Card.withId(pathsMatch.params.id);
|
||||||
const boardModel = cardModel && cardModel.board;
|
|
||||||
const projectModel = boardModel && boardModel.project;
|
|
||||||
|
|
||||||
if (!projectModel) {
|
if (!cardModel || !cardModel.isAvailableForUser(currentUserId)) {
|
||||||
return {
|
return {
|
||||||
cardId: null,
|
cardId: null,
|
||||||
boardId: null,
|
boardId: null,
|
||||||
|
@ -82,20 +62,10 @@ export const selectPath = createReduxOrmSelector(
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!projectModel.hasManagerUser(currentUserId)) {
|
|
||||||
if (!boardModel.hasMemberUser(currentUserId)) {
|
|
||||||
return {
|
|
||||||
cardId: null,
|
|
||||||
boardId: null,
|
|
||||||
projectId: null,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
return {
|
||||||
cardId: cardModel.id,
|
cardId: cardModel.id,
|
||||||
boardId: boardModel.id,
|
boardId: cardModel.boardId,
|
||||||
projectId: projectModel.id,
|
projectId: cardModel.board.projectId,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
default:
|
default:
|
||||||
|
|
5
client/src/selectors/socket.js
Normal file
5
client/src/selectors/socket.js
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
export const selectIsSocketDisconnected = ({ socket: { isDisconnected } }) => isDisconnected;
|
||||||
|
|
||||||
|
export default {
|
||||||
|
selectIsSocketDisconnected,
|
||||||
|
};
|
|
@ -52,7 +52,7 @@ export const selectProjectsForCurrentUser = createSelector(
|
||||||
}
|
}
|
||||||
|
|
||||||
return userModel.getOrderedAvailableProjectsModelArray().map((projectModel) => {
|
return userModel.getOrderedAvailableProjectsModelArray().map((projectModel) => {
|
||||||
const boardsModels = projectModel.getOrderedAvailableBoardsModelArray(userModel.id);
|
const boardsModels = projectModel.getOrderedBoardsModelArrayAvailableForUser(userModel.id);
|
||||||
|
|
||||||
let notificationsTotal = 0;
|
let notificationsTotal = 0;
|
||||||
boardsModels.forEach((boardModel) => {
|
boardsModels.forEach((boardModel) => {
|
||||||
|
@ -86,7 +86,7 @@ export const selectProjectsToListsForCurrentUser = createSelector(
|
||||||
|
|
||||||
return userModel.getOrderedAvailableProjectsModelArray().map((projectModel) => ({
|
return userModel.getOrderedAvailableProjectsModelArray().map((projectModel) => ({
|
||||||
...projectModel.ref,
|
...projectModel.ref,
|
||||||
boards: projectModel.getOrderedMemberBoardsModelArray(id).map((boardModel) => ({
|
boards: projectModel.getOrderedBoardsModelArrayForUser(id).map((boardModel) => ({
|
||||||
...boardModel.ref,
|
...boardModel.ref,
|
||||||
lists: boardModel.getOrderedListsQuerySet().toRefArray(),
|
lists: boardModel.getOrderedListsQuerySet().toRefArray(),
|
||||||
})),
|
})),
|
||||||
|
|
|
@ -12,16 +12,16 @@ const Errors = {
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const emailOrUsernameValidator = (value) =>
|
||||||
|
value.includes('@')
|
||||||
|
? validator.isEmail(value)
|
||||||
|
: value.length >= 3 && value.length <= 16 && /^[a-zA-Z0-9]+((_|\.)?[a-zA-Z0-9])*$/.test(value);
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
inputs: {
|
inputs: {
|
||||||
emailOrUsername: {
|
emailOrUsername: {
|
||||||
type: 'string',
|
type: 'string',
|
||||||
custom: (value) =>
|
custom: emailOrUsernameValidator,
|
||||||
value.includes('@')
|
|
||||||
? validator.isEmail(value)
|
|
||||||
: value.length >= 3 &&
|
|
||||||
value.length <= 16 &&
|
|
||||||
/^[a-zA-Z0-9]+((_|\.)?[a-zA-Z0-9])*$/.test(value),
|
|
||||||
required: true,
|
required: true,
|
||||||
},
|
},
|
||||||
password: {
|
password: {
|
||||||
|
|
|
@ -82,13 +82,15 @@ module.exports = {
|
||||||
const file = _.last(files);
|
const file = _.last(files);
|
||||||
const fileData = await sails.helpers.attachments.processUploadedFile(file);
|
const fileData = await sails.helpers.attachments.processUploadedFile(file);
|
||||||
|
|
||||||
const attachment = await sails.helpers.attachments.createOne(
|
const attachment = await sails.helpers.attachments.createOne.with({
|
||||||
fileData,
|
values: {
|
||||||
currentUser,
|
...fileData,
|
||||||
card,
|
card,
|
||||||
inputs.requestId,
|
creatorUser: currentUser,
|
||||||
this.req,
|
},
|
||||||
);
|
requestId: inputs.requestId,
|
||||||
|
request: this.req,
|
||||||
|
});
|
||||||
|
|
||||||
return exits.success({
|
return exits.success({
|
||||||
item: attachment,
|
item: attachment,
|
||||||
|
|
|
@ -48,7 +48,12 @@ module.exports = {
|
||||||
throw Errors.NOT_ENOUGH_RIGHTS;
|
throw Errors.NOT_ENOUGH_RIGHTS;
|
||||||
}
|
}
|
||||||
|
|
||||||
attachment = await sails.helpers.attachments.deleteOne(attachment, board, card, this.req);
|
attachment = await sails.helpers.attachments.deleteOne.with({
|
||||||
|
board,
|
||||||
|
card,
|
||||||
|
record: attachment,
|
||||||
|
request: this.req,
|
||||||
|
});
|
||||||
|
|
||||||
if (!attachment) {
|
if (!attachment) {
|
||||||
throw Errors.ATTACHMENT_NOT_FOUND;
|
throw Errors.ATTACHMENT_NOT_FOUND;
|
||||||
|
|
|
@ -53,7 +53,13 @@ module.exports = {
|
||||||
}
|
}
|
||||||
|
|
||||||
const values = _.pick(inputs, ['name']);
|
const values = _.pick(inputs, ['name']);
|
||||||
attachment = await sails.helpers.attachments.updateOne(attachment, values, board, this.req);
|
|
||||||
|
attachment = await sails.helpers.attachments.updateOne.with({
|
||||||
|
values,
|
||||||
|
board,
|
||||||
|
record: attachment,
|
||||||
|
request: this.req,
|
||||||
|
});
|
||||||
|
|
||||||
if (!attachment) {
|
if (!attachment) {
|
||||||
throw Errors.ATTACHMENT_NOT_FOUND;
|
throw Errors.ATTACHMENT_NOT_FOUND;
|
||||||
|
|
|
@ -68,8 +68,15 @@ module.exports = {
|
||||||
|
|
||||||
const values = _.pick(inputs, ['role', 'canComment']);
|
const values = _.pick(inputs, ['role', 'canComment']);
|
||||||
|
|
||||||
const boardMembership = await sails.helpers.boardMemberships
|
const boardMembership = await sails.helpers.boardMemberships.createOne
|
||||||
.createOne(values, user, board, this.req)
|
.with({
|
||||||
|
values: {
|
||||||
|
...values,
|
||||||
|
board,
|
||||||
|
user,
|
||||||
|
},
|
||||||
|
request: this.req,
|
||||||
|
})
|
||||||
.intercept('userAlreadyBoardMember', () => Errors.USER_ALREADY_BOARD_MEMBER);
|
.intercept('userAlreadyBoardMember', () => Errors.USER_ALREADY_BOARD_MEMBER);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
|
|
|
@ -36,15 +36,15 @@ module.exports = {
|
||||||
);
|
);
|
||||||
|
|
||||||
if (!isProjectManager) {
|
if (!isProjectManager) {
|
||||||
throw Errors.BOARD_MEMBERSHIP_NOT_FOUND;
|
throw Errors.BOARD_MEMBERSHIP_NOT_FOUND; // Forbidden
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
boardMembership = await sails.helpers.boardMemberships.deleteOne(
|
boardMembership = await sails.helpers.boardMemberships.deleteOne.with({
|
||||||
boardMembership,
|
|
||||||
project,
|
project,
|
||||||
this.req,
|
record: boardMembership,
|
||||||
);
|
request: this.req,
|
||||||
|
});
|
||||||
|
|
||||||
if (!boardMembership) {
|
if (!boardMembership) {
|
||||||
throw Errors.BOARD_MEMBERSHIP_NOT_FOUND;
|
throw Errors.BOARD_MEMBERSHIP_NOT_FOUND;
|
||||||
|
|
|
@ -44,11 +44,11 @@ module.exports = {
|
||||||
|
|
||||||
const values = _.pick(inputs, ['role', 'canComment']);
|
const values = _.pick(inputs, ['role', 'canComment']);
|
||||||
|
|
||||||
boardMembership = await sails.helpers.boardMemberships.updateOne(
|
boardMembership = await sails.helpers.boardMemberships.updateOne.with({
|
||||||
boardMembership,
|
|
||||||
values,
|
values,
|
||||||
this.req,
|
record: boardMembership,
|
||||||
);
|
request: this.req,
|
||||||
|
});
|
||||||
|
|
||||||
return {
|
return {
|
||||||
item: boardMembership,
|
item: boardMembership,
|
||||||
|
|
|
@ -20,11 +20,6 @@ module.exports = {
|
||||||
regex: /^[0-9]+$/,
|
regex: /^[0-9]+$/,
|
||||||
required: true,
|
required: true,
|
||||||
},
|
},
|
||||||
type: {
|
|
||||||
type: 'string',
|
|
||||||
isIn: Object.values(Board.Types),
|
|
||||||
required: true,
|
|
||||||
},
|
|
||||||
position: {
|
position: {
|
||||||
type: 'number',
|
type: 'number',
|
||||||
required: true,
|
required: true,
|
||||||
|
@ -70,7 +65,7 @@ module.exports = {
|
||||||
throw Errors.PROJECT_NOT_FOUND; // Forbidden
|
throw Errors.PROJECT_NOT_FOUND; // Forbidden
|
||||||
}
|
}
|
||||||
|
|
||||||
const values = _.pick(inputs, ['type', 'position', 'name']);
|
const values = _.pick(inputs, ['position', 'name']);
|
||||||
|
|
||||||
let boardImport;
|
let boardImport;
|
||||||
if (inputs.importType && Object.values(Board.ImportTypes).includes(inputs.importType)) {
|
if (inputs.importType && Object.values(Board.ImportTypes).includes(inputs.importType)) {
|
||||||
|
@ -102,14 +97,16 @@ module.exports = {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const { board, boardMembership } = await sails.helpers.boards.createOne(
|
const { board, boardMembership } = await sails.helpers.boards.createOne.with({
|
||||||
values,
|
values: {
|
||||||
boardImport,
|
...values,
|
||||||
currentUser,
|
project,
|
||||||
project,
|
},
|
||||||
inputs.requestId,
|
import: boardImport,
|
||||||
this.req,
|
user: currentUser,
|
||||||
);
|
requestId: inputs.requestId,
|
||||||
|
request: this.req,
|
||||||
|
});
|
||||||
|
|
||||||
if (this.req.isSocket) {
|
if (this.req.isSocket) {
|
||||||
sails.sockets.join(this.req, `board:${board.id}`); // TODO: only when subscription needed
|
sails.sockets.join(this.req, `board:${board.id}`); // TODO: only when subscription needed
|
||||||
|
|
|
@ -39,7 +39,10 @@ module.exports = {
|
||||||
throw Errors.BOARD_NOT_FOUND; // Forbidden
|
throw Errors.BOARD_NOT_FOUND; // Forbidden
|
||||||
}
|
}
|
||||||
|
|
||||||
board = await sails.helpers.boards.deleteOne(board, this.req);
|
board = await sails.helpers.boards.deleteOne.with({
|
||||||
|
record: board,
|
||||||
|
request: this.req,
|
||||||
|
});
|
||||||
|
|
||||||
if (!board) {
|
if (!board) {
|
||||||
throw Errors.BOARD_NOT_FOUND;
|
throw Errors.BOARD_NOT_FOUND;
|
||||||
|
|
|
@ -47,7 +47,7 @@ module.exports = {
|
||||||
const labels = await sails.helpers.boards.getLabels(board.id);
|
const labels = await sails.helpers.boards.getLabels(board.id);
|
||||||
const lists = await sails.helpers.boards.getLists(board.id);
|
const lists = await sails.helpers.boards.getLists(board.id);
|
||||||
|
|
||||||
const cards = await sails.helpers.boards.getCards(board);
|
const cards = await sails.helpers.boards.getCards(board.id);
|
||||||
const cardIds = sails.helpers.utils.mapRecords(cards);
|
const cardIds = sails.helpers.utils.mapRecords(cards);
|
||||||
|
|
||||||
const cardSubscriptions = await sails.helpers.cardSubscriptions.getMany({
|
const cardSubscriptions = await sails.helpers.cardSubscriptions.getMany({
|
||||||
|
@ -69,7 +69,8 @@ module.exports = {
|
||||||
);
|
);
|
||||||
|
|
||||||
cards.forEach((card) => {
|
cards.forEach((card) => {
|
||||||
card.isSubscribed = isSubscribedByCardId[card.id] || false; // eslint-disable-line no-param-reassign
|
// eslint-disable-next-line no-param-reassign
|
||||||
|
card.isSubscribed = isSubscribedByCardId[card.id] || false;
|
||||||
});
|
});
|
||||||
|
|
||||||
if (this.req.isSocket) {
|
if (this.req.isSocket) {
|
||||||
|
|
|
@ -47,7 +47,12 @@ module.exports = {
|
||||||
}
|
}
|
||||||
|
|
||||||
const values = _.pick(inputs, ['position', 'name']);
|
const values = _.pick(inputs, ['position', 'name']);
|
||||||
board = await sails.helpers.boards.updateOne(board, values, this.req);
|
|
||||||
|
board = await sails.helpers.boards.updateOne.with({
|
||||||
|
values,
|
||||||
|
record: board,
|
||||||
|
request: this.req,
|
||||||
|
});
|
||||||
|
|
||||||
if (!board) {
|
if (!board) {
|
||||||
throw Errors.BOARD_NOT_FOUND;
|
throw Errors.BOARD_NOT_FOUND;
|
||||||
|
|
|
@ -71,8 +71,14 @@ module.exports = {
|
||||||
throw Errors.LABEL_NOT_FOUND;
|
throw Errors.LABEL_NOT_FOUND;
|
||||||
}
|
}
|
||||||
|
|
||||||
const cardLabel = await sails.helpers.cardLabels
|
const cardLabel = await sails.helpers.cardLabels.createOne
|
||||||
.createOne(label, card, this.req)
|
.with({
|
||||||
|
values: {
|
||||||
|
card,
|
||||||
|
label,
|
||||||
|
},
|
||||||
|
request: this.req,
|
||||||
|
})
|
||||||
.intercept('labelAlreadyInCard', () => Errors.LABEL_ALREADY_IN_CARD);
|
.intercept('labelAlreadyInCard', () => Errors.LABEL_ALREADY_IN_CARD);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
|
|
|
@ -65,7 +65,11 @@ module.exports = {
|
||||||
throw Errors.LABEL_NOT_IN_CARD;
|
throw Errors.LABEL_NOT_IN_CARD;
|
||||||
}
|
}
|
||||||
|
|
||||||
cardLabel = await sails.helpers.cardLabels.deleteOne(cardLabel, board, this.req);
|
cardLabel = await sails.helpers.cardLabels.deleteOne.with({
|
||||||
|
board,
|
||||||
|
record: cardLabel,
|
||||||
|
request: this.req,
|
||||||
|
});
|
||||||
|
|
||||||
if (!cardLabel) {
|
if (!cardLabel) {
|
||||||
throw Errors.LABEL_NOT_IN_CARD;
|
throw Errors.LABEL_NOT_IN_CARD;
|
||||||
|
|
|
@ -68,8 +68,14 @@ module.exports = {
|
||||||
throw Errors.USER_NOT_FOUND;
|
throw Errors.USER_NOT_FOUND;
|
||||||
}
|
}
|
||||||
|
|
||||||
const cardMembership = await sails.helpers.cardMemberships
|
const cardMembership = await sails.helpers.cardMemberships.createOne
|
||||||
.createOne(inputs.userId, card, this.req)
|
.with({
|
||||||
|
values: {
|
||||||
|
card,
|
||||||
|
userId: inputs.userId,
|
||||||
|
},
|
||||||
|
request: this.req,
|
||||||
|
})
|
||||||
.intercept('userAlreadyCardMember', () => Errors.USER_ALREADY_CARD_MEMBER);
|
.intercept('userAlreadyCardMember', () => Errors.USER_ALREADY_CARD_MEMBER);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
|
|
|
@ -65,7 +65,11 @@ module.exports = {
|
||||||
throw Errors.USER_NOT_CARD_MEMBER;
|
throw Errors.USER_NOT_CARD_MEMBER;
|
||||||
}
|
}
|
||||||
|
|
||||||
cardMembership = await sails.helpers.cardMemberships.deleteOne(cardMembership, board, this.req);
|
cardMembership = await sails.helpers.cardMemberships.deleteOne.with({
|
||||||
|
board,
|
||||||
|
record: cardMembership,
|
||||||
|
request: this.req,
|
||||||
|
});
|
||||||
|
|
||||||
if (!cardMembership) {
|
if (!cardMembership) {
|
||||||
throw Errors.USER_NOT_CARD_MEMBER;
|
throw Errors.USER_NOT_CARD_MEMBER;
|
||||||
|
|
|
@ -4,30 +4,42 @@ const Errors = {
|
||||||
NOT_ENOUGH_RIGHTS: {
|
NOT_ENOUGH_RIGHTS: {
|
||||||
notEnoughRights: 'Not enough rights',
|
notEnoughRights: 'Not enough rights',
|
||||||
},
|
},
|
||||||
BOARD_NOT_FOUND: {
|
|
||||||
boardNotFound: 'Board not found',
|
|
||||||
},
|
|
||||||
LIST_NOT_FOUND: {
|
LIST_NOT_FOUND: {
|
||||||
listNotFound: 'List not found',
|
listNotFound: 'List not found',
|
||||||
},
|
},
|
||||||
LIST_MUST_BE_PRESENT: {
|
|
||||||
listMustBePresent: 'List must be present',
|
|
||||||
},
|
|
||||||
POSITION_MUST_BE_PRESENT: {
|
POSITION_MUST_BE_PRESENT: {
|
||||||
positionMustBePresent: 'Position must be present',
|
positionMustBePresent: 'Position must be present',
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const dueDateValidator = (value) => moment(value, moment.ISO_8601, true).isValid();
|
||||||
|
|
||||||
|
const timerValidator = (value) => {
|
||||||
|
if (!_.isPlainObject(value) || _.size(value) !== 2) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
!_.isNull(value.startedAt) &&
|
||||||
|
_.isString(value.startedAt) &&
|
||||||
|
!moment(value.startedAt, moment.ISO_8601, true).isValid()
|
||||||
|
) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!_.isFinite(value.total)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
};
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
inputs: {
|
inputs: {
|
||||||
boardId: {
|
|
||||||
type: 'string',
|
|
||||||
regex: /^[0-9]+$/,
|
|
||||||
required: true,
|
|
||||||
},
|
|
||||||
listId: {
|
listId: {
|
||||||
type: 'string',
|
type: 'string',
|
||||||
regex: /^[0-9]+$/,
|
regex: /^[0-9]+$/,
|
||||||
|
required: true,
|
||||||
},
|
},
|
||||||
position: {
|
position: {
|
||||||
type: 'number',
|
type: 'number',
|
||||||
|
@ -43,29 +55,11 @@ module.exports = {
|
||||||
},
|
},
|
||||||
dueDate: {
|
dueDate: {
|
||||||
type: 'string',
|
type: 'string',
|
||||||
custom: (value) => moment(value, moment.ISO_8601, true).isValid(),
|
custom: dueDateValidator,
|
||||||
},
|
},
|
||||||
timer: {
|
timer: {
|
||||||
type: 'json',
|
type: 'json',
|
||||||
custom: (value) => {
|
custom: timerValidator,
|
||||||
if (!_.isPlainObject(value) || _.size(value) !== 2) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (
|
|
||||||
!_.isNull(value.startedAt) &&
|
|
||||||
_.isString(value.startedAt) &&
|
|
||||||
!moment(value.startedAt, moment.ISO_8601, true).isValid()
|
|
||||||
) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!_.isFinite(value.total)) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
|
@ -73,15 +67,9 @@ module.exports = {
|
||||||
notEnoughRights: {
|
notEnoughRights: {
|
||||||
responseType: 'forbidden',
|
responseType: 'forbidden',
|
||||||
},
|
},
|
||||||
boardNotFound: {
|
|
||||||
responseType: 'notFound',
|
|
||||||
},
|
|
||||||
listNotFound: {
|
listNotFound: {
|
||||||
responseType: 'notFound',
|
responseType: 'notFound',
|
||||||
},
|
},
|
||||||
listMustBePresent: {
|
|
||||||
responseType: 'unprocessableEntity',
|
|
||||||
},
|
|
||||||
positionMustBePresent: {
|
positionMustBePresent: {
|
||||||
responseType: 'unprocessableEntity',
|
responseType: 'unprocessableEntity',
|
||||||
},
|
},
|
||||||
|
@ -90,40 +78,34 @@ module.exports = {
|
||||||
async fn(inputs) {
|
async fn(inputs) {
|
||||||
const { currentUser } = this.req;
|
const { currentUser } = this.req;
|
||||||
|
|
||||||
const { board } = await sails.helpers.boards
|
const { list } = await sails.helpers.lists
|
||||||
.getProjectPath(inputs.boardId)
|
.getProjectPath(inputs.listId)
|
||||||
.intercept('pathNotFound', () => Errors.BOARD_NOT_FOUND);
|
.intercept('pathNotFound', () => Errors.LIST_NOT_FOUND);
|
||||||
|
|
||||||
const boardMembership = await BoardMembership.findOne({
|
const boardMembership = await BoardMembership.findOne({
|
||||||
boardId: board.id,
|
boardId: list.boardId,
|
||||||
userId: currentUser.id,
|
userId: currentUser.id,
|
||||||
});
|
});
|
||||||
|
|
||||||
if (!boardMembership) {
|
if (!boardMembership) {
|
||||||
throw Errors.BOARD_NOT_FOUND; // Forbidden
|
throw Errors.LIST_NOT_FOUND; // Forbidden
|
||||||
}
|
}
|
||||||
|
|
||||||
if (boardMembership.role !== BoardMembership.Roles.EDITOR) {
|
if (boardMembership.role !== BoardMembership.Roles.EDITOR) {
|
||||||
throw Errors.NOT_ENOUGH_RIGHTS;
|
throw Errors.NOT_ENOUGH_RIGHTS;
|
||||||
}
|
}
|
||||||
|
|
||||||
let list;
|
|
||||||
if (!_.isUndefined(inputs.listId)) {
|
|
||||||
list = await List.findOne({
|
|
||||||
id: inputs.listId,
|
|
||||||
boardId: board.id,
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!list) {
|
|
||||||
throw Errors.LIST_NOT_FOUND;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const values = _.pick(inputs, ['position', 'name', 'description', 'dueDate', 'timer']);
|
const values = _.pick(inputs, ['position', 'name', 'description', 'dueDate', 'timer']);
|
||||||
|
|
||||||
const card = await sails.helpers.cards
|
const card = await sails.helpers.cards.createOne
|
||||||
.createOne(values, currentUser, board, list, this.req)
|
.with({
|
||||||
.intercept('listMustBePresent', () => Errors.LIST_MUST_BE_PRESENT)
|
values: {
|
||||||
|
...values,
|
||||||
|
list,
|
||||||
|
creatorUser: currentUser,
|
||||||
|
},
|
||||||
|
request: this.req,
|
||||||
|
})
|
||||||
.intercept('positionMustBeInValues', () => Errors.POSITION_MUST_BE_PRESENT);
|
.intercept('positionMustBeInValues', () => Errors.POSITION_MUST_BE_PRESENT);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
|
|
|
@ -45,7 +45,10 @@ module.exports = {
|
||||||
throw Errors.NOT_ENOUGH_RIGHTS;
|
throw Errors.NOT_ENOUGH_RIGHTS;
|
||||||
}
|
}
|
||||||
|
|
||||||
card = await sails.helpers.cards.deleteOne(card, this.req);
|
card = await sails.helpers.cards.deleteOne.with({
|
||||||
|
record: card,
|
||||||
|
request: this.req,
|
||||||
|
});
|
||||||
|
|
||||||
if (!card) {
|
if (!card) {
|
||||||
throw Errors.CARD_NOT_FOUND;
|
throw Errors.CARD_NOT_FOUND;
|
||||||
|
|
|
@ -1,74 +0,0 @@
|
||||||
const Errors = {
|
|
||||||
BOARD_NOT_FOUND: {
|
|
||||||
boardNotFound: 'Board not found',
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
module.exports = {
|
|
||||||
inputs: {
|
|
||||||
boardId: {
|
|
||||||
type: 'string',
|
|
||||||
regex: /^[0-9]+$/,
|
|
||||||
required: true,
|
|
||||||
},
|
|
||||||
beforeId: {
|
|
||||||
type: 'string',
|
|
||||||
regex: /^[0-9]+$/,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
exits: {
|
|
||||||
boardNotFound: {
|
|
||||||
responseType: 'notFound',
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
async fn(inputs) {
|
|
||||||
const { currentUser } = this.req;
|
|
||||||
|
|
||||||
const { board } = await sails.helpers.boards
|
|
||||||
.getProjectPath(inputs.boardId)
|
|
||||||
.intercept('pathNotFound', () => Errors.BOARD_NOT_FOUND);
|
|
||||||
|
|
||||||
const isBoardMember = await sails.helpers.users.isBoardMember(currentUser.id, board.id);
|
|
||||||
|
|
||||||
if (!isBoardMember) {
|
|
||||||
throw Errors.BOARD_NOT_FOUND; // Forbidden
|
|
||||||
}
|
|
||||||
|
|
||||||
const cards = await sails.helpers.boards.getCards(board, inputs.beforeId);
|
|
||||||
const cardIds = sails.helpers.utils.mapRecords(cards);
|
|
||||||
|
|
||||||
const cardSubscriptions = await sails.helpers.cardSubscriptions.getMany({
|
|
||||||
cardId: cardIds,
|
|
||||||
userId: currentUser.id,
|
|
||||||
});
|
|
||||||
|
|
||||||
const isSubscribedByCardId = cardSubscriptions.reduce(
|
|
||||||
(result, cardSubscription) => ({
|
|
||||||
...result,
|
|
||||||
[cardSubscription.cardId]: true,
|
|
||||||
}),
|
|
||||||
{},
|
|
||||||
);
|
|
||||||
|
|
||||||
cards.forEach((card) => {
|
|
||||||
card.isSubscribed = isSubscribedByCardId[card.id] || false; // eslint-disable-line no-param-reassign
|
|
||||||
});
|
|
||||||
|
|
||||||
const cardMemberships = await sails.helpers.cards.getCardMemberships(cardIds);
|
|
||||||
const cardLabels = await sails.helpers.cards.getCardLabels(cardIds);
|
|
||||||
const tasks = await sails.helpers.cards.getTasks(cardIds);
|
|
||||||
const attachments = await sails.helpers.cards.getAttachments(cardIds);
|
|
||||||
|
|
||||||
return {
|
|
||||||
items: cards,
|
|
||||||
included: {
|
|
||||||
cardMemberships,
|
|
||||||
cardLabels,
|
|
||||||
tasks,
|
|
||||||
attachments,
|
|
||||||
},
|
|
||||||
};
|
|
||||||
},
|
|
||||||
};
|
|
|
@ -21,6 +21,28 @@ const Errors = {
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const dueDateValidator = (value) => moment(value, moment.ISO_8601, true).isValid();
|
||||||
|
|
||||||
|
const timerValidator = (value) => {
|
||||||
|
if (!_.isPlainObject(value) || _.size(value) !== 2) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
!_.isNull(value.startedAt) &&
|
||||||
|
_.isString(value.startedAt) &&
|
||||||
|
!moment(value.startedAt, moment.ISO_8601, true).isValid()
|
||||||
|
) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!_.isFinite(value.total)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
};
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
inputs: {
|
inputs: {
|
||||||
id: {
|
id: {
|
||||||
|
@ -55,30 +77,12 @@ module.exports = {
|
||||||
},
|
},
|
||||||
dueDate: {
|
dueDate: {
|
||||||
type: 'string',
|
type: 'string',
|
||||||
custom: (value) => moment(value, moment.ISO_8601, true).isValid(),
|
custom: dueDateValidator,
|
||||||
allowNull: true,
|
allowNull: true,
|
||||||
},
|
},
|
||||||
timer: {
|
timer: {
|
||||||
type: 'json',
|
type: 'json',
|
||||||
custom: (value) => {
|
custom: timerValidator,
|
||||||
if (!_.isPlainObject(value) || _.size(value) !== 2) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (
|
|
||||||
!_.isNull(value.startedAt) &&
|
|
||||||
_.isString(value.startedAt) &&
|
|
||||||
!moment(value.startedAt, moment.ISO_8601, true).isValid()
|
|
||||||
) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!_.isFinite(value.total)) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
isSubscribed: {
|
isSubscribed: {
|
||||||
type: 'boolean',
|
type: 'boolean',
|
||||||
|
@ -171,10 +175,21 @@ module.exports = {
|
||||||
'isSubscribed',
|
'isSubscribed',
|
||||||
]);
|
]);
|
||||||
|
|
||||||
card = await sails.helpers.cards
|
card = await sails.helpers.cards.updateOne
|
||||||
.updateOne(card, values, nextBoard, nextList, currentUser, board, list, this.req)
|
.with({
|
||||||
.intercept('nextListMustBePresent', () => Errors.LIST_MUST_BE_PRESENT)
|
board,
|
||||||
.intercept('positionMustBeInValues', () => Errors.POSITION_MUST_BE_PRESENT);
|
list,
|
||||||
|
record: card,
|
||||||
|
values: {
|
||||||
|
...values,
|
||||||
|
board: nextBoard,
|
||||||
|
list: nextList,
|
||||||
|
},
|
||||||
|
user: currentUser,
|
||||||
|
request: this.req,
|
||||||
|
})
|
||||||
|
.intercept('positionMustBeInValues', () => Errors.POSITION_MUST_BE_PRESENT)
|
||||||
|
.intercept('listMustBeInValues', () => Errors.LIST_MUST_BE_PRESENT);
|
||||||
|
|
||||||
if (!card) {
|
if (!card) {
|
||||||
throw Errors.CARD_NOT_FOUND;
|
throw Errors.CARD_NOT_FOUND;
|
||||||
|
|
|
@ -54,7 +54,14 @@ module.exports = {
|
||||||
data: _.pick(inputs, ['text']),
|
data: _.pick(inputs, ['text']),
|
||||||
};
|
};
|
||||||
|
|
||||||
const action = await sails.helpers.actions.createOne(values, currentUser, card, this.req);
|
const action = await sails.helpers.actions.createOne.with({
|
||||||
|
values: {
|
||||||
|
...values,
|
||||||
|
card,
|
||||||
|
user: currentUser,
|
||||||
|
},
|
||||||
|
request: this.req,
|
||||||
|
});
|
||||||
|
|
||||||
return {
|
return {
|
||||||
item: action,
|
item: action,
|
||||||
|
|
|
@ -59,7 +59,11 @@ module.exports = {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
action = await sails.helpers.actions.deleteOne(action, board, this.req);
|
action = await sails.helpers.actions.deleteOne.with({
|
||||||
|
board,
|
||||||
|
record: action,
|
||||||
|
request: this.req,
|
||||||
|
});
|
||||||
|
|
||||||
if (!action) {
|
if (!action) {
|
||||||
throw Errors.COMMENT_ACTION_NOT_FOUND;
|
throw Errors.COMMENT_ACTION_NOT_FOUND;
|
||||||
|
|
|
@ -67,7 +67,12 @@ module.exports = {
|
||||||
data: _.pick(inputs, ['text']),
|
data: _.pick(inputs, ['text']),
|
||||||
};
|
};
|
||||||
|
|
||||||
action = await sails.helpers.actions.updateOne(action, values, board, this.req);
|
action = await sails.helpers.actions.updateOne.with({
|
||||||
|
values,
|
||||||
|
board,
|
||||||
|
record: action,
|
||||||
|
request: this.req,
|
||||||
|
});
|
||||||
|
|
||||||
if (!action) {
|
if (!action) {
|
||||||
throw Errors.COMMENT_ACTION_NOT_FOUND;
|
throw Errors.COMMENT_ACTION_NOT_FOUND;
|
||||||
|
|
|
@ -56,7 +56,14 @@ module.exports = {
|
||||||
}
|
}
|
||||||
|
|
||||||
const values = _.pick(inputs, ['name', 'color']);
|
const values = _.pick(inputs, ['name', 'color']);
|
||||||
const label = await sails.helpers.labels.createOne(values, board, this.req);
|
|
||||||
|
const label = await sails.helpers.labels.createOne.with({
|
||||||
|
values: {
|
||||||
|
...values,
|
||||||
|
board,
|
||||||
|
},
|
||||||
|
request: this.req,
|
||||||
|
});
|
||||||
|
|
||||||
return {
|
return {
|
||||||
item: label,
|
item: label,
|
||||||
|
|
|
@ -45,7 +45,10 @@ module.exports = {
|
||||||
throw Errors.NOT_ENOUGH_RIGHTS;
|
throw Errors.NOT_ENOUGH_RIGHTS;
|
||||||
}
|
}
|
||||||
|
|
||||||
label = await sails.helpers.labels.deleteOne(label, this.req);
|
label = await sails.helpers.labels.deleteOne.with({
|
||||||
|
record: label,
|
||||||
|
request: this.req,
|
||||||
|
});
|
||||||
|
|
||||||
if (!label) {
|
if (!label) {
|
||||||
throw Errors.LABEL_NOT_FOUND;
|
throw Errors.LABEL_NOT_FOUND;
|
||||||
|
|
|
@ -56,7 +56,12 @@ module.exports = {
|
||||||
}
|
}
|
||||||
|
|
||||||
const values = _.pick(inputs, ['name', 'color']);
|
const values = _.pick(inputs, ['name', 'color']);
|
||||||
label = await sails.helpers.labels.updateOne(label, values, this.req);
|
|
||||||
|
label = await sails.helpers.labels.updateOne.with({
|
||||||
|
values,
|
||||||
|
record: label,
|
||||||
|
request: this.req,
|
||||||
|
});
|
||||||
|
|
||||||
return {
|
return {
|
||||||
item: label,
|
item: label,
|
||||||
|
|
|
@ -54,7 +54,14 @@ module.exports = {
|
||||||
}
|
}
|
||||||
|
|
||||||
const values = _.pick(inputs, ['position', 'name']);
|
const values = _.pick(inputs, ['position', 'name']);
|
||||||
const list = await sails.helpers.lists.createOne(values, board, this.req);
|
|
||||||
|
const list = await sails.helpers.lists.createOne.with({
|
||||||
|
values: {
|
||||||
|
...values,
|
||||||
|
board,
|
||||||
|
},
|
||||||
|
request: this.req,
|
||||||
|
});
|
||||||
|
|
||||||
return {
|
return {
|
||||||
item: list,
|
item: list,
|
||||||
|
|
|
@ -45,7 +45,10 @@ module.exports = {
|
||||||
throw Errors.NOT_ENOUGH_RIGHTS;
|
throw Errors.NOT_ENOUGH_RIGHTS;
|
||||||
}
|
}
|
||||||
|
|
||||||
list = await sails.helpers.lists.deleteOne(list, this.req);
|
list = await sails.helpers.lists.deleteOne.with({
|
||||||
|
record: list,
|
||||||
|
request: this.req,
|
||||||
|
});
|
||||||
|
|
||||||
if (!list) {
|
if (!list) {
|
||||||
throw Errors.LIST_NOT_FOUND;
|
throw Errors.LIST_NOT_FOUND;
|
||||||
|
|
|
@ -53,7 +53,12 @@ module.exports = {
|
||||||
}
|
}
|
||||||
|
|
||||||
const values = _.pick(inputs, ['position', 'name']);
|
const values = _.pick(inputs, ['position', 'name']);
|
||||||
list = await sails.helpers.lists.updateOne(list, values, this.req);
|
|
||||||
|
list = await sails.helpers.lists.updateOne.with({
|
||||||
|
values,
|
||||||
|
record: list,
|
||||||
|
request: this.req,
|
||||||
|
});
|
||||||
|
|
||||||
if (!list) {
|
if (!list) {
|
||||||
throw Errors.LIST_NOT_FOUND;
|
throw Errors.LIST_NOT_FOUND;
|
||||||
|
|
|
@ -15,12 +15,12 @@ module.exports = {
|
||||||
|
|
||||||
const values = _.pick(inputs, ['isRead']);
|
const values = _.pick(inputs, ['isRead']);
|
||||||
|
|
||||||
const notifications = await sails.helpers.notifications.updateMany(
|
const notifications = await sails.helpers.notifications.updateMany.with({
|
||||||
inputs.ids.split(','),
|
|
||||||
values,
|
values,
|
||||||
currentUser,
|
recordsOrIds: inputs.ids.split(','),
|
||||||
this.req,
|
user: currentUser,
|
||||||
);
|
request: this.req,
|
||||||
|
});
|
||||||
|
|
||||||
return {
|
return {
|
||||||
items: notifications,
|
items: notifications,
|
||||||
|
|
|
@ -57,8 +57,14 @@ module.exports = {
|
||||||
throw Error.USER_NOT_FOUND;
|
throw Error.USER_NOT_FOUND;
|
||||||
}
|
}
|
||||||
|
|
||||||
const projectManager = await sails.helpers.projectManagers
|
const projectManager = await sails.helpers.projectManagers.createOne
|
||||||
.createOne(user, project, this.req)
|
.with({
|
||||||
|
values: {
|
||||||
|
project,
|
||||||
|
user,
|
||||||
|
},
|
||||||
|
request: this.req,
|
||||||
|
})
|
||||||
.intercept('userAlreadyProjectManager', () => Errors.USER_ALREADY_PROJECT_MANAGER);
|
.intercept('userAlreadyProjectManager', () => Errors.USER_ALREADY_PROJECT_MANAGER);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
|
|
|
@ -38,7 +38,10 @@ module.exports = {
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: check if the last one
|
// TODO: check if the last one
|
||||||
projectManager = await sails.helpers.projectManagers.deleteOne(projectManager, this.req);
|
projectManager = await sails.helpers.projectManagers.deleteOne.with({
|
||||||
|
record: projectManager,
|
||||||
|
request: this.req,
|
||||||
|
});
|
||||||
|
|
||||||
if (!projectManager) {
|
if (!projectManager) {
|
||||||
throw Errors.PROJECT_MANAGER_NOT_FOUND;
|
throw Errors.PROJECT_MANAGER_NOT_FOUND;
|
||||||
|
|
|
@ -11,11 +11,11 @@ module.exports = {
|
||||||
|
|
||||||
const values = _.pick(inputs, ['name']);
|
const values = _.pick(inputs, ['name']);
|
||||||
|
|
||||||
const { project, projectManager } = await sails.helpers.projects.createOne(
|
const { project, projectManager } = await sails.helpers.projects.createOne.with({
|
||||||
values,
|
values,
|
||||||
currentUser,
|
user: currentUser,
|
||||||
this.req,
|
request: this.req,
|
||||||
);
|
});
|
||||||
|
|
||||||
return {
|
return {
|
||||||
item: project,
|
item: project,
|
||||||
|
|
|
@ -34,7 +34,10 @@ module.exports = {
|
||||||
throw Errors.PROJECT_NOT_FOUND; // Forbidden
|
throw Errors.PROJECT_NOT_FOUND; // Forbidden
|
||||||
}
|
}
|
||||||
|
|
||||||
project = await sails.helpers.projects.deleteOne(project, this.req);
|
project = await sails.helpers.projects.deleteOne.with({
|
||||||
|
record: project,
|
||||||
|
request: this.req,
|
||||||
|
});
|
||||||
|
|
||||||
if (!project) {
|
if (!project) {
|
||||||
throw Errors.PROJECT_NOT_FOUND;
|
throw Errors.PROJECT_NOT_FOUND;
|
||||||
|
|
|
@ -6,7 +6,6 @@ module.exports = {
|
||||||
const managerProjects = await sails.helpers.projects.getMany(managerProjectIds);
|
const managerProjects = await sails.helpers.projects.getMany(managerProjectIds);
|
||||||
|
|
||||||
let boardMemberships = await sails.helpers.users.getBoardMemberships(currentUser.id);
|
let boardMemberships = await sails.helpers.users.getBoardMemberships(currentUser.id);
|
||||||
|
|
||||||
const membershipBoardIds = sails.helpers.utils.mapRecords(boardMemberships, 'boardId');
|
const membershipBoardIds = sails.helpers.utils.mapRecords(boardMemberships, 'boardId');
|
||||||
|
|
||||||
let membershipBoards = await sails.helpers.boards.getMany({
|
let membershipBoards = await sails.helpers.boards.getMany({
|
||||||
|
|
|
@ -85,13 +85,13 @@ module.exports = {
|
||||||
return Errors.FILE_IS_NOT_IMAGE;
|
return Errors.FILE_IS_NOT_IMAGE;
|
||||||
});
|
});
|
||||||
|
|
||||||
project = await sails.helpers.projects.updateOne(
|
project = await sails.helpers.projects.updateOne.with({
|
||||||
project,
|
record: project,
|
||||||
{
|
values: {
|
||||||
backgroundImage: fileData,
|
backgroundImage: fileData,
|
||||||
},
|
},
|
||||||
this.req,
|
request: this.req,
|
||||||
);
|
});
|
||||||
|
|
||||||
if (!project) {
|
if (!project) {
|
||||||
throw Errors.PROJECT_NOT_FOUND;
|
throw Errors.PROJECT_NOT_FOUND;
|
||||||
|
|
|
@ -4,6 +4,36 @@ const Errors = {
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const backgroundValidator = (value) => {
|
||||||
|
if (_.isNull(value)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!_.isPlainObject(value)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!Object.values(Project.BackgroundTypes).includes(value.type)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
value.type === Project.BackgroundTypes.GRADIENT &&
|
||||||
|
_.size(value) === 2 &&
|
||||||
|
Project.BACKGROUND_GRADIENTS.includes(value.name)
|
||||||
|
) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (value.type === Project.BackgroundTypes.IMAGE && _.size(value) === 1) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
};
|
||||||
|
|
||||||
|
const backgroundImageValidator = (value) => _.isNull(value);
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
inputs: {
|
inputs: {
|
||||||
id: {
|
id: {
|
||||||
|
@ -17,37 +47,11 @@ module.exports = {
|
||||||
},
|
},
|
||||||
background: {
|
background: {
|
||||||
type: 'json',
|
type: 'json',
|
||||||
custom: (value) => {
|
custom: backgroundValidator,
|
||||||
if (_.isNull(value)) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!_.isPlainObject(value)) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!Object.values(Project.BackgroundTypes).includes(value.type)) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (
|
|
||||||
value.type === Project.BackgroundTypes.GRADIENT &&
|
|
||||||
_.size(value) === 2 &&
|
|
||||||
Project.BACKGROUND_GRADIENTS.includes(value.name)
|
|
||||||
) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (value.type === Project.BackgroundTypes.IMAGE && _.size(value) === 1) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
backgroundImage: {
|
backgroundImage: {
|
||||||
type: 'json',
|
type: 'json',
|
||||||
custom: (value) => _.isNull(value),
|
custom: backgroundImageValidator,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
|
@ -73,7 +77,12 @@ module.exports = {
|
||||||
}
|
}
|
||||||
|
|
||||||
const values = _.pick(inputs, ['name', 'background', 'backgroundImage']);
|
const values = _.pick(inputs, ['name', 'background', 'backgroundImage']);
|
||||||
project = await sails.helpers.projects.updateOne(project, values, this.req);
|
|
||||||
|
project = await sails.helpers.projects.updateOne.with({
|
||||||
|
values,
|
||||||
|
record: project,
|
||||||
|
request: this.req,
|
||||||
|
});
|
||||||
|
|
||||||
if (!project) {
|
if (!project) {
|
||||||
throw Errors.PROJECT_NOT_FOUND;
|
throw Errors.PROJECT_NOT_FOUND;
|
||||||
|
|
|
@ -57,7 +57,14 @@ module.exports = {
|
||||||
}
|
}
|
||||||
|
|
||||||
const values = _.pick(inputs, ['position', 'name', 'isCompleted']);
|
const values = _.pick(inputs, ['position', 'name', 'isCompleted']);
|
||||||
const task = await sails.helpers.tasks.createOne(values, card, this.req);
|
|
||||||
|
const task = await sails.helpers.tasks.createOne.with({
|
||||||
|
values: {
|
||||||
|
...values,
|
||||||
|
card,
|
||||||
|
},
|
||||||
|
request: this.req,
|
||||||
|
});
|
||||||
|
|
||||||
return {
|
return {
|
||||||
item: task,
|
item: task,
|
||||||
|
|
|
@ -48,7 +48,11 @@ module.exports = {
|
||||||
throw Errors.NOT_ENOUGH_RIGHTS;
|
throw Errors.NOT_ENOUGH_RIGHTS;
|
||||||
}
|
}
|
||||||
|
|
||||||
task = await sails.helpers.tasks.deleteOne(task, board, this.req);
|
task = await sails.helpers.tasks.deleteOne.with({
|
||||||
|
board,
|
||||||
|
record: task,
|
||||||
|
request: this.req,
|
||||||
|
});
|
||||||
|
|
||||||
if (!task) {
|
if (!task) {
|
||||||
throw Errors.TASK_NOT_FOUND;
|
throw Errors.TASK_NOT_FOUND;
|
||||||
|
|
|
@ -59,7 +59,13 @@ module.exports = {
|
||||||
}
|
}
|
||||||
|
|
||||||
const values = _.pick(inputs, ['position', 'name', 'isCompleted']);
|
const values = _.pick(inputs, ['position', 'name', 'isCompleted']);
|
||||||
task = await sails.helpers.tasks.updateOne(task, values, board, this.req);
|
|
||||||
|
task = await sails.helpers.tasks.updateOne.with({
|
||||||
|
values,
|
||||||
|
board,
|
||||||
|
record: task,
|
||||||
|
request: this.req,
|
||||||
|
});
|
||||||
|
|
||||||
if (!task) {
|
if (!task) {
|
||||||
throw Errors.TASK_NOT_FOUND;
|
throw Errors.TASK_NOT_FOUND;
|
||||||
|
|
|
@ -9,6 +9,8 @@ const Errors = {
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const passwordValidator = (value) => zxcvbn(value).score >= 2; // TODO: move to config
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
inputs: {
|
inputs: {
|
||||||
email: {
|
email: {
|
||||||
|
@ -18,7 +20,7 @@ module.exports = {
|
||||||
},
|
},
|
||||||
password: {
|
password: {
|
||||||
type: 'string',
|
type: 'string',
|
||||||
custom: (value) => zxcvbn(value).score >= 2, // TODO: move to config
|
custom: passwordValidator,
|
||||||
required: true,
|
required: true,
|
||||||
},
|
},
|
||||||
name: {
|
name: {
|
||||||
|
@ -74,8 +76,11 @@ module.exports = {
|
||||||
'subscribeToOwnCards',
|
'subscribeToOwnCards',
|
||||||
]);
|
]);
|
||||||
|
|
||||||
const user = await sails.helpers.users
|
const user = await sails.helpers.users.createOne
|
||||||
.createOne(values, this.req)
|
.with({
|
||||||
|
values,
|
||||||
|
request: this.req,
|
||||||
|
})
|
||||||
.intercept('emailAlreadyInUse', () => Errors.EMAIL_ALREADY_IN_USE)
|
.intercept('emailAlreadyInUse', () => Errors.EMAIL_ALREADY_IN_USE)
|
||||||
.intercept('usernameAlreadyInUse', () => Errors.USERNAME_ALREADY_IN_USE);
|
.intercept('usernameAlreadyInUse', () => Errors.USERNAME_ALREADY_IN_USE);
|
||||||
|
|
||||||
|
|
|
@ -26,7 +26,10 @@ module.exports = {
|
||||||
throw Errors.USER_NOT_FOUND;
|
throw Errors.USER_NOT_FOUND;
|
||||||
}
|
}
|
||||||
|
|
||||||
user = await sails.helpers.users.deleteOne(user, this.req);
|
user = await sails.helpers.users.deleteOne.with({
|
||||||
|
record: user,
|
||||||
|
request: this.req,
|
||||||
|
});
|
||||||
|
|
||||||
if (!user) {
|
if (!user) {
|
||||||
throw Errors.USER_NOT_FOUND;
|
throw Errors.USER_NOT_FOUND;
|
||||||
|
|
Some files were not shown because too many files have changed in this diff Show more
Loading…
Add table
Add a link
Reference in a new issue