diff --git a/packages/component-user-manager/src/helpers/Team.js b/packages/component-user-manager/src/helpers/Team.js index dc3990cb2d173f804fba18a0640dd1bf42f3426c..3b700c4436f1306a7b41b07ab4b9ded4aa566944 100644 --- a/packages/component-user-manager/src/helpers/Team.js +++ b/packages/component-user-manager/src/helpers/Team.js @@ -1,5 +1,5 @@ const logger = require('@pubsweet/logger') -const get = require('lodash/get') +// const get = require('lodash/get') const createNewTeam = async (collectionId, role, userId, TeamModel) => { let permissions, group, name diff --git a/packages/component-wizard/src/components/Wizard.js b/packages/component-wizard/src/components/Wizard.js index 3ef3ff5830d9294c81924d5ffbb0074460a853cf..6b34441594bfda3c5595a3712bd8fb3f3f16781a 100644 --- a/packages/component-wizard/src/components/Wizard.js +++ b/packages/component-wizard/src/components/Wizard.js @@ -13,7 +13,7 @@ export default ({ step, project, }) => ( - <Root data-test={`submission-${project.customId}`}> + <Root data-test-submission={project.customId}> {showProgress && ( <Steps currentStep={step} margin="0 20px 60px 0"> {getSteps().map((step, index) => ( diff --git a/packages/components-faraday/src/components/AuthorList/Author.js b/packages/components-faraday/src/components/AuthorList/Author.js index a0092ad7d7644d0bca0f57fd25f9ef281161bf75..b18d8ed6017b7bf1e58c3c0b3d674d2f77e33bb7 100644 --- a/packages/components-faraday/src/components/AuthorList/Author.js +++ b/packages/components-faraday/src/components/AuthorList/Author.js @@ -7,14 +7,11 @@ import { Label } from './FormItems' export default ({ id, firstName, - middleName, lastName, email, affiliation, - country, dragHandle, isOver, - countryParser, removeAuthor, isSubmitting, isCorresponding, @@ -46,13 +43,11 @@ export default ({ </Header> <Row> <Label label="First name" value={firstName} /> - <Label label="Middle name" value={middleName} /> <Label label="Last name" value={lastName} /> </Row> <Row> <Label label="Email" value={email} /> <Label label="Affiliation" value={affiliation} /> - <Label label="Country" value={countryParser(country)} /> </Row> </AuthorContainer> </Root> diff --git a/packages/components-faraday/src/components/AuthorList/AuthorAdder.js b/packages/components-faraday/src/components/AuthorList/AuthorAdder.js index 5bb4d5917744c8cca6982a6a9dea6180f3da64e3..876ec1f9d7b3b6b5141a2fb02b73cb63ed30f2d7 100644 --- a/packages/components-faraday/src/components/AuthorList/AuthorAdder.js +++ b/packages/components-faraday/src/components/AuthorList/AuthorAdder.js @@ -7,10 +7,14 @@ import styled from 'styled-components' import { compose, withProps } from 'recompose' import { selectCurrentUser } from 'xpub-selectors' -import countries from './countries' import { Spinner } from '../UIComponents/' -import { getAuthorFetching } from '../../redux/authors' -import { MenuItem, ValidatedTextField } from './FormItems' +import { + getAuthorFetching, + getAuthors, + authorSuccess, + authorFailure, +} from '../../redux/authors' +import { ValidatedTextField } from './FormItems' const emailRegex = new RegExp( /^(([^<>()\[\]\.,;:\s@\"]+(\.[^<>()\[\]\.,;:\s@\"]+)*)|(\".+\"))@(([^<>()[\]\.,;:\s@\"]+\.)+[^<>()[\]\.,;:\s@\"]{2,})$/i, //eslint-disable-line @@ -35,7 +39,6 @@ const AuthorAdder = ({ <Title>{authors.length === 0 ? 'Submitting author' : 'Author'}</Title> <Row> <ValidatedTextField isRequired label="First name*" name="firstName" /> - <ValidatedTextField label="Middle name" name="middleName" /> <ValidatedTextField isRequired label="Last name*" name="lastName" /> </Row> @@ -51,7 +54,6 @@ const AuthorAdder = ({ label="Affiliation*" name="affiliation" /> - <MenuItem label="Country*" name="country" options={countries} /> </Row> <ButtonsContainer> <Button data-test="button-cancel-author" onClick={setEditMode(false)}> @@ -75,10 +77,16 @@ const AuthorAdder = ({ ) export default compose( - connect(state => ({ - currentUser: selectCurrentUser(state), - isFetching: getAuthorFetching(state), - })), + connect( + state => ({ + currentUser: selectCurrentUser(state), + isFetching: getAuthorFetching(state), + }), + { + authorSuccess, + authorFailure, + }, + ), withProps(({ currentUser: { admin, username, email }, authors }) => { if (!admin && authors.length === 0) { return { @@ -106,14 +114,22 @@ export default compose( isCorresponding: isFirstAuthor, }, collectionId, - ).then(author => { - const newAuthors = [...authors, author] - setEditMode(false)() - setTimeout(() => { - setFormAuthors(newAuthors) - }, 1000) - reset() - }) + ).then( + () => { + setEditMode(false)() + setTimeout(() => { + getAuthors(collectionId).then( + data => { + dispatch(authorSuccess()) + setFormAuthors(data) + }, + err => dispatch(authorFailure(err)), + ) + }, 10) + reset() + }, + err => dispatch(authorFailure(err)), + ) }, }), )(AuthorAdder) diff --git a/packages/components-faraday/src/components/AuthorList/AuthorEditor.js b/packages/components-faraday/src/components/AuthorList/AuthorEditor.js index 879ff26a344b952246bc176aa5ff21700b8dc228..53f69eecc5e7fc0e87c570a54b93930133eb768f 100644 --- a/packages/components-faraday/src/components/AuthorList/AuthorEditor.js +++ b/packages/components-faraday/src/components/AuthorList/AuthorEditor.js @@ -6,10 +6,15 @@ import styled, { css } from 'styled-components' import { compose, withHandlers, withProps } from 'recompose' import { reduxForm, Field, change as changeForm } from 'redux-form' -import countries from './countries' import { Spinner } from '../UIComponents' -import { getAuthorFetching } from '../../redux/authors' -import { ValidatedTextField, MenuItem } from './FormItems' +import { + getAuthors, + editAuthor, + getAuthorFetching, + authorSuccess, + authorFailure, +} from '../../redux/authors' +import { ValidatedTextField } from './FormItems' const emailRegex = new RegExp( /^(([^<>()\[\]\.,;:\s@\"]+(\.[^<>()\[\]\.,;:\s@\"]+)*)|(\".+\"))@(([^<>()[\]\.,;:\s@\"]+\.)+[^<>()[\]\.,;:\s@\"]{2,})$/i, //eslint-disable-line @@ -32,6 +37,7 @@ const AuthorEdit = ({ isCorresponding, email, changeCorresponding, + ...rest }) => ( <Root> <Header> @@ -69,7 +75,6 @@ const AuthorEdit = ({ label="First name*" name="edit.firstName" /> - <ValidatedTextField label="Middle name" name="edit.middleName" /> <ValidatedTextField isRequired label="Last name*" name="edit.lastName" /> </Row> @@ -85,7 +90,6 @@ const AuthorEdit = ({ label="Affiliation*" name="edit.affiliation" /> - <MenuItem label="Country*" name="edit.country" options={countries} /> </Row> </Root> ) @@ -95,19 +99,18 @@ export default compose( state => ({ isFetching: getAuthorFetching(state), }), - { changeForm }, + { changeForm, authorSuccess, authorFailure }, ), withProps(props => ({ initialValues: { edit: pick(props, [ + 'id', 'isCorresponding', 'isSubmitting', 'firstName', 'lastName', - 'middleName', 'email', 'affiliation', - 'country', ]), }, })), @@ -125,15 +128,22 @@ export default compose( onSubmit: ( values, dispatch, - { setAuthorEdit, setAuthors, authors, index, changeForm }, + { setAuthorEdit, setAuthors, authors, index, changeForm, project }, ) => { - const newAuthors = [ - ...authors.slice(0, index), - values.edit, - ...authors.slice(index + 1), - ] - setAuthorEdit(-1)() - setAuthors(newAuthors) + const newAuthor = values.edit + editAuthor(project.id, newAuthor.id, newAuthor).then( + () => { + getAuthors(project.id).then( + data => { + dispatch(authorSuccess()) + setAuthorEdit(-1)() + setAuthors(data) + }, + err => dispatch(authorFailure(err)), + ) + }, + err => dispatch(authorFailure(err)), + ) }, }), )(AuthorEdit) diff --git a/packages/components-faraday/src/components/AuthorList/AuthorList.js b/packages/components-faraday/src/components/AuthorList/AuthorList.js index c3505142b7c176178f07be5ab03e5c31f03b581e..6e5f0e2a78e3cda041c505de7426a741bb8ec09d 100644 --- a/packages/components-faraday/src/components/AuthorList/AuthorList.js +++ b/packages/components-faraday/src/components/AuthorList/AuthorList.js @@ -1,5 +1,4 @@ import React from 'react' -import { get } from 'lodash' import { th } from '@pubsweet/ui' import PropTypes from 'prop-types' import { connect } from 'react-redux' @@ -15,10 +14,17 @@ import { import { change as changeForm } from 'redux-form' import { SortableList } from 'pubsweet-component-sortable-list/src/components' -import { addAuthor, deleteAuthor, getAuthors } from '../../redux/authors' +import { + addAuthor, + deleteAuthor, + getAuthors, + authorFailure, + getAuthorsTeam, + getAuthorError, + updateAuthorsTeam, +} from '../../redux/authors' import Author from './Author' -import countries from './countries' import StaticList from './StaticList' import AuthorAdder from './AuthorAdder' import AuthorEditor from './AuthorEditor' @@ -36,6 +42,7 @@ const Authors = ({ setEditMode, editedAuthor, setFormAuthors, + error, ...rest }) => ( <Root> @@ -68,6 +75,7 @@ const Authors = ({ {...rest} /> )} + {error && <ErrorMessage>{error}</ErrorMessage>} </Root> ) @@ -77,11 +85,13 @@ export default compose( connect( state => ({ currentUser: state.currentUser.user, + error: getAuthorError(state), }), { addAuthor, changeForm, deleteAuthor, + authorFailure, }, ), withState('authors', 'setAuthors', []), @@ -104,11 +114,21 @@ export default compose( changeForm('wizard', 'editMode', mode) setEditMode(v => mode) }, - dropItem: ({ authors, setFormAuthors }) => () => { + dropItem: ({ authors, setFormAuthors, project, authorFailure }) => () => { setFormAuthors(authors) + getAuthorsTeam(project.id) + .then(team => { + const members = authors.map(a => a.id) + updateAuthorsTeam(team.id, { members }).catch(err => { + authorFailure(err) + getAuthors(project.id).then(setFormAuthors) + }) + }) + .catch(err => { + authorFailure(err) + getAuthors(project.id).then(setFormAuthors) + }) }, - countryParser: () => countryCode => - get(countries.find(c => c.value === countryCode), 'label'), parseAuthorType: () => (isSubmitting, isCorresponding, index) => { if (isSubmitting) return `#${index + 1} Submitting author` if (isCorresponding) return `#${index + 1} Corresponding author` @@ -121,14 +141,20 @@ export default compose( const newAuthors = SortableList.moveItem(authors, dragIndex, hoverIndex) setFormAuthors(newAuthors) }, - removeAuthor: ({ authors, setFormAuthors, deleteAuthor, project }) => ( - id, - authorEmail, - ) => () => { - deleteAuthor(project.id, id).then(() => { - const newAuthors = authors.filter(a => a.id !== id) - setFormAuthors(newAuthors) - }) + removeAuthor: ({ + authors, + setFormAuthors, + deleteAuthor, + authorFailure, + project, + }) => (id, authorEmail) => () => { + deleteAuthor(project.id, id).then( + () => { + const newAuthors = authors.filter(a => a.id !== id) + setFormAuthors(newAuthors) + }, + err => authorFailure(err), + ) }, setAsCorresponding: ({ authors, setFormAuthors }) => authorEmail => () => { const newAuthors = authors.map( @@ -156,4 +182,7 @@ const Root = styled.div` border: ${th('borderDefault')}; padding: ${th('subGridUnit')}; ` +const ErrorMessage = styled.div` + color: ${th('colorError')}; +` // #endregion diff --git a/packages/components-faraday/src/components/AuthorList/StaticList.js b/packages/components-faraday/src/components/AuthorList/StaticList.js index b8b45d89a865d9630f649127414076bb66c8284d..6e00a7a456175175eb4953ddfa8a82671c82d540 100644 --- a/packages/components-faraday/src/components/AuthorList/StaticList.js +++ b/packages/components-faraday/src/components/AuthorList/StaticList.js @@ -7,7 +7,6 @@ export default ({ editIndex, setFormAuthors, removeAuthor, - countryParser, editComponent, setAuthorEdit, parseAuthorType, @@ -27,16 +26,15 @@ export default ({ }, setAuthors: setFormAuthors, setAuthorEdit, - countryParser, parseAuthorType, setAsCorresponding, ...author, + ...rest, }) ) : ( <Author - key={author.firstName} + key={author.id} {...author} - countryParser={countryParser} index={index} parseAuthorType={parseAuthorType} removeAuthor={removeAuthor} diff --git a/packages/components-faraday/src/redux/authors.js b/packages/components-faraday/src/redux/authors.js index 10f6cc645ab334eb872dd282e5ea0f9e46235d08..e7767063fce3ad2c52194935e28b5875a5d60d95 100644 --- a/packages/components-faraday/src/redux/authors.js +++ b/packages/components-faraday/src/redux/authors.js @@ -1,5 +1,10 @@ -import { get } from 'lodash' -import { create, get as apiGet, remove } from 'pubsweet-client/src/helpers/api' +import { get, head } from 'lodash' +import { + create, + get as apiGet, + remove, + update, +} from 'pubsweet-client/src/helpers/api' // constants const REQUEST = 'authors/REQUEST' @@ -11,7 +16,7 @@ export const authorRequest = () => ({ type: REQUEST, }) -export const authorFaiure = error => ({ +export const authorFailure = error => ({ type: FAILURE, error, }) @@ -22,19 +27,31 @@ export const authorSuccess = () => ({ export const addAuthor = (author, collectionId) => dispatch => { dispatch(authorRequest()) - return create(`/users/invite/${collectionId}`, { + return create(`/collections/${collectionId}/users`, { email: author.email, role: 'author', ...author, }) } -export const deleteAuthor = (collectionId, userId) => dispatch => - remove(`/collections/${collectionId}/users/${userId}?role=author`) -// Promise.resolve(author) +export const deleteAuthor = (collectionId, userId) => dispatch => { + dispatch(authorRequest()) + return remove(`/collections/${collectionId}/users/${userId}`) +} + +export const editAuthor = (collectionId, userId, body) => + update(`/collections/${collectionId}/users/${userId}`, body) export const getAuthors = collectionId => - apiGet(`/collections/${collectionId}/users?role=author`) + apiGet(`/collections/${collectionId}/users`) + +export const getAuthorsTeam = collectionId => + apiGet(`/teams?object.id=${collectionId}&group=author`).then(teams => + head(teams), + ) + +export const updateAuthorsTeam = (teamId, body) => + update(`/teams/${teamId}s`, body) // selectors export const getFragmentAuthors = (state, fragmentId) => @@ -57,7 +74,7 @@ export default (state = initialState, action) => { case FAILURE: return { ...initialState, - error: action.error, + error: get(JSON.parse(get(action.error, 'response') || {}), 'error'), isFetching: false, } case 'UPDATE_FRAGMENT_SUCCESS':