2015-07-08 22:54:07 +02:00
|
|
|
'use strict';
|
|
|
|
|
|
|
|
import React from 'react';
|
2015-10-01 11:16:38 +02:00
|
|
|
import { History } from 'react-router';
|
2015-11-03 10:57:41 +01:00
|
|
|
import Moment from 'moment';
|
2015-07-08 22:54:07 +02:00
|
|
|
|
2015-11-30 18:23:03 +01:00
|
|
|
import ReactError from '../../mixins/react_error';
|
|
|
|
import { ResourceNotFoundError } from '../../models/errors';
|
|
|
|
|
2015-07-08 22:54:07 +02:00
|
|
|
import PieceActions from '../../actions/piece_actions';
|
|
|
|
import PieceStore from '../../stores/piece_store';
|
|
|
|
|
2015-08-11 17:12:12 +02:00
|
|
|
import PieceListActions from '../../actions/piece_list_actions';
|
|
|
|
import PieceListStore from '../../stores/piece_list_store';
|
|
|
|
|
|
|
|
import UserActions from '../../actions/user_actions';
|
|
|
|
import UserStore from '../../stores/user_store';
|
|
|
|
|
|
|
|
import EditionListActions from '../../actions/edition_list_actions';
|
|
|
|
|
2015-07-08 22:54:07 +02:00
|
|
|
import Piece from './piece';
|
2015-07-15 11:39:08 +02:00
|
|
|
import CollapsibleParagraph from './../ascribe_collapsible/collapsible_paragraph';
|
|
|
|
import FurtherDetails from './further_details';
|
2015-07-08 22:54:07 +02:00
|
|
|
|
2015-08-11 17:12:12 +02:00
|
|
|
import DetailProperty from './detail_property';
|
2015-09-29 15:58:24 +02:00
|
|
|
import LicenseDetail from './license_detail';
|
2015-08-20 15:50:30 +02:00
|
|
|
import HistoryIterator from './history_iterator';
|
2015-08-11 17:12:12 +02:00
|
|
|
|
|
|
|
import AclButtonList from './../ascribe_buttons/acl_button_list';
|
|
|
|
import CreateEditionsForm from '../ascribe_forms/create_editions_form';
|
|
|
|
import CreateEditionsButton from '../ascribe_buttons/create_editions_button';
|
|
|
|
import DeleteButton from '../ascribe_buttons/delete_button';
|
|
|
|
|
2015-11-03 10:39:01 +01:00
|
|
|
import AclInformation from '../ascribe_buttons/acl_information';
|
|
|
|
import AclProxy from '../acl_proxy';
|
|
|
|
|
2015-08-27 13:43:26 +02:00
|
|
|
import ListRequestActions from '../ascribe_forms/list_form_request_actions';
|
2015-08-21 16:49:04 +02:00
|
|
|
|
2015-08-11 17:12:12 +02:00
|
|
|
import GlobalNotificationModel from '../../models/global_notification_model';
|
|
|
|
import GlobalNotificationActions from '../../actions/global_notification_actions';
|
|
|
|
|
2015-08-21 15:04:38 +02:00
|
|
|
import Note from './note';
|
2015-08-20 14:01:02 +02:00
|
|
|
|
|
|
|
import ApiUrls from '../../constants/api_urls';
|
2015-10-12 15:25:21 +02:00
|
|
|
import AscribeSpinner from '../ascribe_spinner';
|
|
|
|
|
2015-08-11 17:12:12 +02:00
|
|
|
import { mergeOptions } from '../../utils/general_utils';
|
|
|
|
import { getLangText } from '../../utils/lang_utils';
|
2015-10-13 17:52:45 +02:00
|
|
|
import { setDocumentTitle } from '../../utils/dom_utils';
|
2015-07-13 15:21:09 +02:00
|
|
|
|
2015-07-08 22:54:07 +02:00
|
|
|
/**
|
|
|
|
* This is the component that implements resource/data specific functionality
|
|
|
|
*/
|
|
|
|
let PieceContainer = React.createClass({
|
2015-09-30 18:30:50 +02:00
|
|
|
propTypes: {
|
2015-10-28 11:26:54 +01:00
|
|
|
furtherDetailsType: React.PropTypes.func,
|
2015-11-23 10:46:20 +01:00
|
|
|
params: React.PropTypes.object
|
2015-09-30 18:30:50 +02:00
|
|
|
},
|
2015-07-08 22:54:07 +02:00
|
|
|
|
2015-11-30 18:23:03 +01:00
|
|
|
mixins: [History, ReactError],
|
2015-08-11 17:12:12 +02:00
|
|
|
|
2015-10-28 11:26:54 +01:00
|
|
|
getDefaultProps() {
|
|
|
|
return {
|
|
|
|
furtherDetailsType: FurtherDetails
|
|
|
|
};
|
|
|
|
},
|
|
|
|
|
2015-08-11 17:12:12 +02:00
|
|
|
getInitialState() {
|
|
|
|
return mergeOptions(
|
|
|
|
UserStore.getState(),
|
|
|
|
PieceListStore.getState(),
|
2016-01-14 13:20:45 +01:00
|
|
|
PieceStore.getInitialState(),
|
2015-08-11 17:12:12 +02:00
|
|
|
{
|
|
|
|
showCreateEditionsDialog: false
|
|
|
|
}
|
|
|
|
);
|
2015-07-08 22:54:07 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
componentDidMount() {
|
2015-08-11 17:12:12 +02:00
|
|
|
UserStore.listen(this.onChange);
|
|
|
|
PieceListStore.listen(this.onChange);
|
2015-07-08 22:54:07 +02:00
|
|
|
PieceStore.listen(this.onChange);
|
|
|
|
|
2015-11-04 13:56:43 +01:00
|
|
|
this.loadPiece();
|
2015-10-19 15:58:05 +02:00
|
|
|
UserActions.fetchCurrentUser();
|
2015-07-08 22:54:07 +02:00
|
|
|
},
|
|
|
|
|
2016-01-14 13:20:45 +01:00
|
|
|
// This is done to update the container when the user clicks on the prev or next
|
|
|
|
// button to update the URL parameter (and therefore to switch pieces) or
|
|
|
|
// when the user clicks on a notification while being in another piece view
|
|
|
|
componentWillReceiveProps(nextProps) {
|
|
|
|
if (this.props.params.pieceId !== nextProps.params.pieceId) {
|
|
|
|
PieceActions.flushPiece();
|
|
|
|
this.loadPiece(nextProps.params.pieceId);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2015-10-19 15:58:05 +02:00
|
|
|
componentDidUpdate() {
|
2016-01-15 13:00:09 +01:00
|
|
|
const { err: pieceErr } = this.state.pieceMeta;
|
2015-10-19 15:58:05 +02:00
|
|
|
|
2016-01-15 13:00:09 +01:00
|
|
|
if (pieceErr && pieceErr.json && pieceErr.json.status === 404) {
|
2015-12-07 11:48:28 +01:00
|
|
|
this.throws(new ResourceNotFoundError(getLangText("Oops, the piece you're looking for doesn't exist.")));
|
2015-10-19 15:58:05 +02:00
|
|
|
}
|
2015-11-04 13:56:43 +01:00
|
|
|
},
|
|
|
|
|
|
|
|
componentWillUnmount() {
|
2015-07-08 22:54:07 +02:00
|
|
|
PieceStore.unlisten(this.onChange);
|
2015-08-11 17:12:12 +02:00
|
|
|
UserStore.unlisten(this.onChange);
|
|
|
|
PieceListStore.unlisten(this.onChange);
|
2015-07-08 22:54:07 +02:00
|
|
|
},
|
|
|
|
|
2015-08-11 17:12:12 +02:00
|
|
|
onChange(state) {
|
2015-08-26 16:44:24 +02:00
|
|
|
/*
|
2015-10-28 11:26:54 +01:00
|
|
|
|
2015-08-26 16:44:24 +02:00
|
|
|
ATTENTION:
|
|
|
|
THIS IS JUST A TEMPORARY USABILITY FIX THAT ESSENTIALLY REMOVES THE LOAN BUTTON
|
|
|
|
FROM THE PIECE DETAIL PAGE SO THAT USERS DO NOT CONFUSE A PIECE WITH AN EDITION.
|
|
|
|
|
|
|
|
IT SHOULD BE REMOVED AND REPLACED WITH A BETTER SOLUTION ASAP!
|
|
|
|
|
2015-08-28 14:27:02 +02:00
|
|
|
ALSO, WE ENABLED THE LOAN BUTTON FOR IKONOTV TO LET THEM LOAN ON A PIECE LEVEL
|
|
|
|
|
2015-08-26 16:44:24 +02:00
|
|
|
*/
|
2016-01-14 13:20:45 +01:00
|
|
|
if (state && state.piece && state.piece.acl && typeof state.piece.acl.acl_loan !== 'undefined') {
|
2015-08-26 16:44:24 +02:00
|
|
|
let pieceState = mergeOptions({}, state.piece);
|
|
|
|
pieceState.acl.acl_loan = false;
|
|
|
|
this.setState({
|
|
|
|
piece: pieceState
|
|
|
|
});
|
|
|
|
|
|
|
|
} else {
|
|
|
|
this.setState(state);
|
|
|
|
}
|
2015-08-11 17:12:12 +02:00
|
|
|
},
|
2015-07-08 22:54:07 +02:00
|
|
|
|
2016-01-14 13:20:45 +01:00
|
|
|
loadPiece(pieceId = this.props.params.pieceId) {
|
|
|
|
PieceActions.fetchPiece(pieceId);
|
2015-07-08 22:54:07 +02:00
|
|
|
},
|
|
|
|
|
2015-08-11 17:12:12 +02:00
|
|
|
toggleCreateEditionsDialog() {
|
|
|
|
this.setState({
|
|
|
|
showCreateEditionsDialog: !this.state.showCreateEditionsDialog
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
|
|
|
handleEditionCreationSuccess() {
|
2015-12-08 15:09:03 +01:00
|
|
|
const { filterBy, orderAsc, orderBy, page, pageSize, search } = this.state;
|
|
|
|
|
2016-01-14 13:20:45 +01:00
|
|
|
PieceActions.updateProperty({ key: 'num_editions', value: 0 });
|
2016-01-18 10:53:01 +01:00
|
|
|
PieceListActions.fetchPieceList({ page, pageSize, search, orderBy, orderAsc, filterBy });
|
2016-01-18 14:15:19 +01:00
|
|
|
|
2015-08-11 17:12:12 +02:00
|
|
|
this.toggleCreateEditionsDialog();
|
|
|
|
},
|
|
|
|
|
|
|
|
handleDeleteSuccess(response) {
|
2015-12-08 15:09:03 +01:00
|
|
|
const { filterBy, orderAsc, orderBy, page, pageSize, search } = this.state;
|
|
|
|
|
2016-01-18 10:53:01 +01:00
|
|
|
PieceListActions.fetchPieceList({ page, pageSize, search, orderBy, orderAsc, filterBy });
|
2015-08-11 17:12:12 +02:00
|
|
|
|
|
|
|
// since we're deleting a piece, we just need to close
|
|
|
|
// all editions dialogs and not reload them
|
|
|
|
EditionListActions.closeAllEditionLists();
|
|
|
|
EditionListActions.clearAllEditionSelections();
|
|
|
|
|
2016-01-14 13:20:45 +01:00
|
|
|
const notification = new GlobalNotificationModel(response.notification, 'success');
|
2015-08-11 17:12:12 +02:00
|
|
|
GlobalNotificationActions.appendGlobalNotification(notification);
|
|
|
|
|
2016-01-05 17:27:25 +01:00
|
|
|
this.history.push('/collection');
|
2015-08-11 17:12:12 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
getCreateEditionsDialog() {
|
2016-01-14 13:20:45 +01:00
|
|
|
if (this.state.piece.num_editions < 1 && this.state.showCreateEditionsDialog) {
|
2015-08-11 17:12:12 +02:00
|
|
|
return (
|
|
|
|
<div style={{marginTop: '1em'}}>
|
|
|
|
<CreateEditionsForm
|
|
|
|
pieceId={this.state.piece.id}
|
|
|
|
handleSuccess={this.handleEditionCreationSuccess} />
|
2016-01-14 13:20:45 +01:00
|
|
|
<hr />
|
2015-08-11 17:12:12 +02:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
} else {
|
2016-01-14 13:20:45 +01:00
|
|
|
return (<hr />);
|
2015-08-11 17:12:12 +02:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
handlePollingSuccess(pieceId, numEditions) {
|
2015-12-08 15:09:03 +01:00
|
|
|
const { filterBy, orderAsc, orderBy, page, pageSize, search } = this.state;
|
2015-08-11 17:12:12 +02:00
|
|
|
|
|
|
|
// we need to refresh the num_editions property of the actual piece we're looking at
|
|
|
|
PieceActions.updateProperty({
|
|
|
|
key: 'num_editions',
|
|
|
|
value: numEditions
|
|
|
|
});
|
|
|
|
|
|
|
|
// as well as its representation in the collection
|
|
|
|
// btw.: It's not sufficient to just set num_editions to numEditions, since a single accordion
|
|
|
|
// list item also uses the firstEdition property which we can only get from the server in that case.
|
|
|
|
// Therefore we need to at least refetch the changed piece from the server or on our case simply all
|
2016-01-18 10:53:01 +01:00
|
|
|
PieceListActions.fetchPieceList({ page, pageSize, search, orderBy, orderAsc, filterBy });
|
2015-08-11 17:12:12 +02:00
|
|
|
|
2016-01-18 13:29:08 +01:00
|
|
|
const notification = new GlobalNotificationModel(getLangText('Editions successfully created'), 'success', 10000);
|
2015-08-11 17:12:12 +02:00
|
|
|
GlobalNotificationActions.appendGlobalNotification(notification);
|
|
|
|
},
|
|
|
|
|
2015-08-21 15:04:38 +02:00
|
|
|
getId() {
|
2016-01-14 13:20:45 +01:00
|
|
|
return { 'id': this.state.piece.id };
|
2015-08-21 15:04:38 +02:00
|
|
|
},
|
|
|
|
|
2015-09-03 15:17:12 +02:00
|
|
|
getActions() {
|
2015-11-03 10:39:01 +01:00
|
|
|
const { piece, currentUser } = this.state;
|
|
|
|
|
2016-01-14 13:20:45 +01:00
|
|
|
if (piece.notifications && piece.notifications.length > 0) {
|
2015-08-21 16:49:04 +02:00
|
|
|
return (
|
2015-08-27 13:43:26 +02:00
|
|
|
<ListRequestActions
|
2015-11-03 10:39:01 +01:00
|
|
|
pieceOrEditions={piece}
|
|
|
|
currentUser={currentUser}
|
2015-08-27 13:43:26 +02:00
|
|
|
handleSuccess={this.loadPiece}
|
2016-01-14 13:20:45 +01:00
|
|
|
notifications={piece.notifications} />);
|
2015-11-03 10:39:01 +01:00
|
|
|
} else {
|
2015-08-21 16:49:04 +02:00
|
|
|
return (
|
2015-11-03 10:39:01 +01:00
|
|
|
<AclProxy
|
2015-11-30 13:20:05 +01:00
|
|
|
show={currentUser && currentUser.email && Object.keys(piece.acl).length > 1}>
|
2015-12-07 10:48:46 +01:00
|
|
|
{/*
|
|
|
|
`acl_view` is always available in `edition.acl`, therefore if it has
|
|
|
|
no more than 1 key, we're hiding the `DetailProperty` actions as otherwise
|
|
|
|
`AclInformation` would show up
|
|
|
|
*/}
|
2015-12-23 09:39:02 +01:00
|
|
|
<DetailProperty
|
|
|
|
label={getLangText('ACTIONS')}
|
|
|
|
className="hidden-print">
|
2015-11-03 10:39:01 +01:00
|
|
|
<AclButtonList
|
|
|
|
className="ascribe-button-list"
|
|
|
|
availableAcls={piece.acl}
|
2015-11-03 11:13:32 +01:00
|
|
|
pieceOrEditions={piece}
|
2015-11-03 10:39:01 +01:00
|
|
|
handleSuccess={this.loadPiece}>
|
|
|
|
<CreateEditionsButton
|
|
|
|
label={getLangText('CREATE EDITIONS')}
|
|
|
|
className="btn-sm"
|
|
|
|
piece={piece}
|
|
|
|
toggleCreateEditionsDialog={this.toggleCreateEditionsDialog}
|
|
|
|
onPollingSuccess={this.handlePollingSuccess}/>
|
|
|
|
<DeleteButton
|
|
|
|
handleSuccess={this.handleDeleteSuccess}
|
2016-01-14 13:20:45 +01:00
|
|
|
piece={piece} />
|
2015-11-03 10:39:01 +01:00
|
|
|
<AclInformation
|
|
|
|
aim="button"
|
2015-11-19 19:47:33 +01:00
|
|
|
verbs={['acl_share', 'acl_transfer', 'acl_create_editions', 'acl_loan', 'acl_delete',
|
|
|
|
'acl_consign']}
|
2016-01-14 13:20:45 +01:00
|
|
|
aclObject={piece.acl} />
|
2015-11-03 10:39:01 +01:00
|
|
|
</AclButtonList>
|
|
|
|
</DetailProperty>
|
|
|
|
</AclProxy>
|
2015-08-21 16:49:04 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
},
|
2015-08-28 12:33:00 +02:00
|
|
|
|
2015-07-08 22:54:07 +02:00
|
|
|
render() {
|
2015-12-16 17:49:30 +01:00
|
|
|
const { furtherDetailsType: FurtherDetailsType } = this.props;
|
|
|
|
const { currentUser, piece } = this.state;
|
|
|
|
|
2016-01-14 13:20:45 +01:00
|
|
|
if (piece.id) {
|
2016-01-18 11:37:11 +01:00
|
|
|
setDocumentTitle(`${piece.artist_name}, ${piece.title}`);
|
2015-10-13 17:52:45 +02:00
|
|
|
|
2015-07-08 22:54:07 +02:00
|
|
|
return (
|
|
|
|
<Piece
|
2015-12-16 17:49:30 +01:00
|
|
|
piece={piece}
|
|
|
|
currentUser={currentUser}
|
2015-08-11 17:12:12 +02:00
|
|
|
header={
|
|
|
|
<div className="ascribe-detail-header">
|
2016-01-11 11:59:35 +01:00
|
|
|
<hr className="hidden-print" style={{marginTop: 0}} />
|
2015-12-16 17:49:30 +01:00
|
|
|
<h1 className="ascribe-detail-title">{piece.title}</h1>
|
2016-01-21 15:17:19 +01:00
|
|
|
<DetailProperty label="CREATED BY" value={piece.artist_name} />
|
2015-12-16 17:49:30 +01:00
|
|
|
<DetailProperty label="DATE" value={Moment(piece.date_created, 'YYYY-MM-DD').year() } />
|
|
|
|
{piece.num_editions > 0 ? <DetailProperty label="EDITIONS" value={ piece.num_editions } /> : null}
|
2015-08-11 17:12:12 +02:00
|
|
|
<hr/>
|
|
|
|
</div>
|
|
|
|
}
|
|
|
|
subheader={
|
|
|
|
<div className="ascribe-detail-header">
|
2016-01-21 15:17:19 +01:00
|
|
|
<DetailProperty label={getLangText('ASCRIBED BY')} value={ piece.user_registered } />
|
2015-12-16 17:49:30 +01:00
|
|
|
<DetailProperty label={getLangText('ID')} value={ piece.bitcoin_id } ellipsis={true} />
|
|
|
|
<LicenseDetail license={piece.license_type} />
|
2015-08-11 17:12:12 +02:00
|
|
|
</div>
|
|
|
|
}
|
2015-08-21 16:49:04 +02:00
|
|
|
buttons={this.getActions()}>
|
2015-08-11 17:12:12 +02:00
|
|
|
{this.getCreateEditionsDialog()}
|
2015-08-20 15:50:30 +02:00
|
|
|
<CollapsibleParagraph
|
|
|
|
title={getLangText('Loan History')}
|
2015-12-16 17:49:30 +01:00
|
|
|
show={piece.loan_history && piece.loan_history.length > 0}>
|
2015-08-20 15:50:30 +02:00
|
|
|
<HistoryIterator
|
2015-12-16 17:49:30 +01:00
|
|
|
history={piece.loan_history} />
|
2015-08-20 15:50:30 +02:00
|
|
|
</CollapsibleParagraph>
|
2015-08-21 15:04:38 +02:00
|
|
|
<CollapsibleParagraph
|
2015-08-21 16:38:18 +02:00
|
|
|
title={getLangText('Notes')}
|
2015-12-16 17:49:30 +01:00
|
|
|
show={!!(currentUser.username || piece.acl.acl_edit || piece.public_note)}>
|
2015-08-21 15:04:38 +02:00
|
|
|
<Note
|
|
|
|
id={this.getId}
|
|
|
|
label={getLangText('Personal note (private)')}
|
2015-12-16 17:49:30 +01:00
|
|
|
defaultValue={piece.private_note || null}
|
|
|
|
show = {!!currentUser.username}
|
2015-08-21 16:38:18 +02:00
|
|
|
placeholder={getLangText('Enter your comments ...')}
|
2015-08-21 15:04:38 +02:00
|
|
|
editable={true}
|
2015-08-21 16:38:18 +02:00
|
|
|
successMessage={getLangText('Private note saved')}
|
2015-08-21 15:04:38 +02:00
|
|
|
url={ApiUrls.note_private_piece}
|
2016-01-14 13:20:45 +01:00
|
|
|
currentUser={currentUser} />
|
2015-10-29 15:31:25 +01:00
|
|
|
<Note
|
|
|
|
id={this.getId}
|
2015-11-20 11:11:03 +01:00
|
|
|
label={getLangText('Personal note (public)')}
|
2015-12-16 17:49:30 +01:00
|
|
|
defaultValue={piece.public_note || null}
|
2015-10-29 15:31:25 +01:00
|
|
|
placeholder={getLangText('Enter your comments ...')}
|
2015-12-16 17:49:30 +01:00
|
|
|
editable={!!piece.acl.acl_edit}
|
|
|
|
show={!!(piece.public_note || piece.acl.acl_edit)}
|
2015-11-20 11:11:03 +01:00
|
|
|
successMessage={getLangText('Public note saved')}
|
2015-10-29 15:31:25 +01:00
|
|
|
url={ApiUrls.note_public_piece}
|
2016-01-14 13:20:45 +01:00
|
|
|
currentUser={currentUser} />
|
2015-08-21 15:04:38 +02:00
|
|
|
</CollapsibleParagraph>
|
2015-07-15 11:39:08 +02:00
|
|
|
<CollapsibleParagraph
|
2015-08-21 16:38:18 +02:00
|
|
|
title={getLangText('Further Details')}
|
2015-12-16 17:49:30 +01:00
|
|
|
show={piece.acl.acl_edit
|
|
|
|
|| Object.keys(piece.extra_data).length > 0
|
|
|
|
|| piece.other_data.length > 0}
|
2015-07-15 11:39:08 +02:00
|
|
|
defaultExpanded={true}>
|
2015-10-28 11:26:54 +01:00
|
|
|
<FurtherDetailsType
|
2015-12-16 17:49:30 +01:00
|
|
|
editable={piece.acl.acl_edit}
|
|
|
|
pieceId={piece.id}
|
|
|
|
extraData={piece.extra_data}
|
|
|
|
otherData={piece.other_data}
|
2015-11-23 10:46:20 +01:00
|
|
|
handleSuccess={this.loadPiece} />
|
2015-07-15 11:39:08 +02:00
|
|
|
</CollapsibleParagraph>
|
2015-08-21 15:04:38 +02:00
|
|
|
|
2015-07-15 11:39:08 +02:00
|
|
|
</Piece>
|
2015-07-08 22:54:07 +02:00
|
|
|
);
|
|
|
|
} else {
|
|
|
|
return (
|
2015-07-13 15:00:12 +02:00
|
|
|
<div className="fullpage-spinner">
|
2015-10-12 15:25:21 +02:00
|
|
|
<AscribeSpinner color='dark-blue' size='lg'/>
|
2015-07-13 15:00:12 +02:00
|
|
|
</div>
|
2015-07-08 22:54:07 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
export default PieceContainer;
|