2018-05-24 13:47:31 -04:00
|
|
|
const bindAll = require('lodash.bindall');
|
2018-03-08 15:57:19 -05:00
|
|
|
const FormattedDate = require('react-intl').FormattedDate;
|
|
|
|
const injectIntl = require('react-intl').injectIntl;
|
|
|
|
const intlShape = require('react-intl').intlShape;
|
|
|
|
const PropTypes = require('prop-types');
|
|
|
|
const React = require('react');
|
2018-04-24 11:00:47 -04:00
|
|
|
const Formsy = require('formsy-react').default;
|
|
|
|
const classNames = require('classnames');
|
2018-03-08 15:57:19 -05:00
|
|
|
|
2018-05-02 15:27:49 -04:00
|
|
|
const GUI = require('scratch-gui').default;
|
|
|
|
const IntlGUI = injectIntl(GUI);
|
|
|
|
|
2018-04-24 11:00:47 -04:00
|
|
|
const sessionActions = require('../../redux/session.js');
|
|
|
|
const decorateText = require('../../lib/decorate-text.jsx');
|
2018-03-08 15:57:19 -05:00
|
|
|
const FlexRow = require('../../components/flex-row/flex-row.jsx');
|
2018-05-24 09:57:06 -04:00
|
|
|
const Button = require('../../components/forms/button.jsx');
|
2018-03-08 15:57:19 -05:00
|
|
|
const Avatar = require('../../components/avatar/avatar.jsx');
|
|
|
|
const CappedNumber = require('../../components/cappednumber/cappednumber.jsx');
|
|
|
|
const ShareBanner = require('../../components/share-banner/share-banner.jsx');
|
2018-04-24 11:00:47 -04:00
|
|
|
const ThumbnailColumn = require('../../components/thumbnailcolumn/thumbnailcolumn.jsx');
|
|
|
|
const InplaceInput = require('../../components/forms/inplace-input.jsx');
|
2018-05-24 13:47:31 -04:00
|
|
|
const ReportModal = require('../../components/modal/report/modal.jsx');
|
|
|
|
|
2018-03-08 15:57:19 -05:00
|
|
|
require('./preview.scss');
|
|
|
|
|
2018-05-24 13:47:31 -04:00
|
|
|
class PreviewPresentation extends React.Component {
|
|
|
|
constructor (props) {
|
|
|
|
super(props);
|
|
|
|
bindAll(this, [
|
|
|
|
'handleReportClick',
|
|
|
|
'handleReportClose',
|
|
|
|
'handleReportSubmit'
|
|
|
|
]);
|
|
|
|
this.state = {
|
2018-05-27 18:00:56 -04:00
|
|
|
reportOpen: false
|
2018-05-24 13:47:31 -04:00
|
|
|
};
|
|
|
|
}
|
|
|
|
handleReportClick (e) {
|
|
|
|
e.preventDefault();
|
|
|
|
this.setState({reportOpen: true});
|
|
|
|
}
|
|
|
|
handleReportClose () {
|
|
|
|
this.setState({reportOpen: false});
|
|
|
|
}
|
2018-05-27 18:00:56 -04:00
|
|
|
|
|
|
|
handleReportSubmit (formData, callback) {
|
|
|
|
const data = {
|
|
|
|
...formData,
|
|
|
|
id: this.props.projectId,
|
|
|
|
username: this.props.user.username
|
|
|
|
};
|
|
|
|
console.log('submit report data', data);
|
|
|
|
// TODO: pass error to modal via callback.
|
|
|
|
callback();
|
2018-05-24 13:47:31 -04:00
|
|
|
this.setState({reportOpen: false});
|
|
|
|
}
|
|
|
|
render () {
|
|
|
|
const {
|
|
|
|
editable,
|
|
|
|
faved,
|
|
|
|
favoriteCount,
|
|
|
|
intl,
|
|
|
|
isFullScreen,
|
|
|
|
loved,
|
|
|
|
loveCount,
|
|
|
|
originalInfo,
|
|
|
|
parentInfo,
|
|
|
|
projectId,
|
|
|
|
projectInfo,
|
|
|
|
remixes,
|
|
|
|
sessionStatus,
|
|
|
|
studios,
|
|
|
|
user,
|
|
|
|
onFavoriteClicked,
|
|
|
|
onLoveClicked,
|
|
|
|
onSeeInside,
|
|
|
|
onUpdate
|
|
|
|
// ...otherProps TBD
|
|
|
|
} = this.props;
|
|
|
|
const shareDate = (projectInfo.history && projectInfo.history.shared) ? projectInfo.history.shared : '';
|
|
|
|
return (
|
|
|
|
<div className="preview">
|
|
|
|
{projectInfo.history && shareDate === '' &&
|
|
|
|
<ShareBanner>
|
2018-04-24 11:00:47 -04:00
|
|
|
<FlexRow className="preview-row">
|
2018-05-24 13:47:31 -04:00
|
|
|
<span className="share-text">
|
2018-05-27 18:00:56 -04:00
|
|
|
This project is not shared — so only you can see it. Click share to let everyone see it!
|
2018-05-24 13:47:31 -04:00
|
|
|
</span>
|
|
|
|
<Button className="button share-button">
|
|
|
|
Share
|
|
|
|
</Button>
|
|
|
|
</FlexRow>
|
|
|
|
</ShareBanner>
|
|
|
|
}
|
|
|
|
|
|
|
|
{ projectInfo && projectInfo.author && projectInfo.author.id && (
|
|
|
|
<div className="inner">
|
|
|
|
<Formsy>
|
|
|
|
<FlexRow className="preview-row">
|
|
|
|
<FlexRow className="project-header">
|
|
|
|
<Avatar
|
|
|
|
src={`https://cdn2.scratch.mit.edu/get_image/user/${projectInfo. author.id}_48x48.png`}
|
|
|
|
/>
|
|
|
|
<div className="title">
|
|
|
|
{editable ?
|
|
|
|
|
|
|
|
<InplaceInput
|
|
|
|
className="project-title"
|
|
|
|
handleUpdate={onUpdate}
|
|
|
|
name="title"
|
|
|
|
validationErrors={{
|
|
|
|
maxLength: 'Sorry title is too long'
|
|
|
|
// maxLength: props.intl.formatMessage({
|
|
|
|
// id: 'project.titleMaxLength'
|
|
|
|
// })
|
|
|
|
}}
|
|
|
|
validations={{
|
2018-05-27 18:00:56 -04:00
|
|
|
maxLength: 100
|
2018-05-24 13:47:31 -04:00
|
|
|
}}
|
|
|
|
value={projectInfo.title}
|
|
|
|
/> :
|
|
|
|
<div className="project-title">{projectInfo.title}</div>
|
|
|
|
}
|
|
|
|
{`${intl.formatMessage({id: 'thumbnail.by'})} `}
|
|
|
|
<a href={`/users/${projectInfo.author.username}`}>
|
|
|
|
{projectInfo.author.username}
|
|
|
|
</a>
|
|
|
|
</div>
|
|
|
|
</FlexRow>
|
|
|
|
<div className="project-buttons">
|
|
|
|
{sessionStatus === sessionActions.Status.FETCHED &&
|
|
|
|
Object.keys(user).length > 0 &&
|
|
|
|
user.id !== projectInfo.author.id &&
|
|
|
|
<Button className="button remix-button">
|
|
|
|
Remix
|
|
|
|
</Button>
|
2018-04-24 11:00:47 -04:00
|
|
|
}
|
2018-05-24 13:47:31 -04:00
|
|
|
<Button
|
|
|
|
className="button see-inside-button"
|
|
|
|
onClick={onSeeInside}
|
|
|
|
>
|
|
|
|
See Inside
|
|
|
|
</Button>
|
2018-03-08 15:57:19 -05:00
|
|
|
</div>
|
2018-04-24 11:00:47 -04:00
|
|
|
</FlexRow>
|
2018-05-24 13:47:31 -04:00
|
|
|
<FlexRow className="preview-row">
|
|
|
|
<div className="guiPlayer">
|
|
|
|
<IntlGUI
|
|
|
|
isPlayerOnly
|
|
|
|
basePath="/"
|
|
|
|
className="guiPlayer"
|
|
|
|
isFullScreen={isFullScreen}
|
|
|
|
previewInfoVisible="false"
|
|
|
|
projectId={projectId}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
<FlexRow className="project-notes">
|
|
|
|
{parentInfo && parentInfo.author && parentInfo.id && (
|
|
|
|
<FlexRow className="remix-credit">
|
|
|
|
<Avatar
|
|
|
|
className="remix"
|
|
|
|
src={`https://cdn2.scratch.mit.edu/get_image/user/${parentInfo.author.id}_48x48.png`}
|
|
|
|
/>
|
|
|
|
<div className="credit-text">
|
|
|
|
Thanks to <a
|
|
|
|
href={`/users/${parentInfo.author.username}`}
|
|
|
|
>
|
|
|
|
{parentInfo.author.username}
|
|
|
|
</a> for the original project <a
|
|
|
|
href={`/preview/${parentInfo.id}`}
|
|
|
|
>
|
|
|
|
{parentInfo.title}
|
|
|
|
</a>.
|
|
|
|
</div>
|
|
|
|
</FlexRow>
|
|
|
|
)}
|
|
|
|
{originalInfo && originalInfo.author && originalInfo.id && (
|
|
|
|
<FlexRow className="remix-credit">
|
|
|
|
<Avatar
|
|
|
|
className="remix"
|
|
|
|
src={`https://cdn2.scratch.mit.edu/get_image/user/${originalInfo.author.id}_48x48.png`}
|
|
|
|
/>
|
|
|
|
<div className="credit-text">
|
|
|
|
Thanks to <a
|
|
|
|
href={`/users/${originalInfo.author.username}`}
|
|
|
|
>
|
|
|
|
{originalInfo.author.username}
|
|
|
|
</a> for the original project <a
|
|
|
|
href={`/preview/${originalInfo.id}`}
|
|
|
|
>
|
|
|
|
{originalInfo.title}
|
|
|
|
</a>.
|
|
|
|
</div>
|
|
|
|
</FlexRow>
|
|
|
|
)}
|
|
|
|
|
|
|
|
<FlexRow className="description-block">
|
|
|
|
<div className="project-textlabel">
|
|
|
|
Instructions
|
2018-05-24 09:57:06 -04:00
|
|
|
</div>
|
2018-05-24 13:47:31 -04:00
|
|
|
{editable ?
|
|
|
|
<InplaceInput
|
|
|
|
className={classNames(
|
|
|
|
'project-description-edit',
|
|
|
|
{remixes: parentInfo && parentInfo.author}
|
|
|
|
)}
|
|
|
|
handleUpdate={onUpdate}
|
|
|
|
name="instructions"
|
|
|
|
type="textarea"
|
|
|
|
validationErrors={{
|
|
|
|
maxLength: 'Sorry description is too long'
|
|
|
|
// maxLength: props.intl.formatMessage({
|
|
|
|
// id: 'project.descriptionMaxLength'
|
|
|
|
// })
|
|
|
|
}}
|
|
|
|
validations={{
|
|
|
|
// TODO: actual 5000
|
|
|
|
maxLength: 1000
|
|
|
|
}}
|
|
|
|
value={projectInfo.instructions}
|
|
|
|
/> :
|
|
|
|
<div className="project-description">
|
|
|
|
{decorateText(projectInfo.instructions)}
|
|
|
|
</div>
|
|
|
|
}
|
2018-05-24 09:57:06 -04:00
|
|
|
</FlexRow>
|
2018-05-24 13:47:31 -04:00
|
|
|
<FlexRow className="description-block">
|
|
|
|
<div className="project-textlabel">
|
|
|
|
Notes and Credits
|
2018-04-24 11:00:47 -04:00
|
|
|
</div>
|
2018-05-24 13:47:31 -04:00
|
|
|
{editable ?
|
|
|
|
<InplaceInput
|
|
|
|
className={classNames(
|
|
|
|
'project-description-edit',
|
|
|
|
{remixes: parentInfo && parentInfo.author}
|
|
|
|
)}
|
|
|
|
handleUpdate={onUpdate}
|
|
|
|
name="description"
|
|
|
|
type="textarea"
|
|
|
|
validationErrors={{
|
|
|
|
maxLength: 'Sorry description is too long'
|
|
|
|
// maxLength: props.intl.formatMessage({
|
|
|
|
// id: 'project.descriptionMaxLength'
|
|
|
|
// })
|
|
|
|
}}
|
|
|
|
validations={{
|
|
|
|
// TODO: actual 5000
|
|
|
|
maxLength: 1000
|
|
|
|
}}
|
|
|
|
value={projectInfo.description}
|
|
|
|
/> :
|
|
|
|
<div className="project-description last">
|
|
|
|
{decorateText(projectInfo.description)}
|
|
|
|
</div>
|
|
|
|
}
|
2018-04-24 11:00:47 -04:00
|
|
|
</FlexRow>
|
2018-05-24 09:57:06 -04:00
|
|
|
</FlexRow>
|
2018-05-24 13:47:31 -04:00
|
|
|
</FlexRow>
|
|
|
|
<FlexRow className="preview-row">
|
|
|
|
<FlexRow className="stats">
|
|
|
|
<div
|
|
|
|
className={classNames('project-loves', {loved: loved})}
|
|
|
|
key="loves"
|
|
|
|
onClick={onLoveClicked}
|
|
|
|
>
|
|
|
|
{loveCount}
|
|
|
|
</div>
|
|
|
|
<div
|
|
|
|
className={classNames('project-favorites', {favorited: faved})}
|
|
|
|
key="favorites"
|
|
|
|
onClick={onFavoriteClicked}
|
|
|
|
>
|
|
|
|
{favoriteCount}
|
|
|
|
</div>
|
|
|
|
<div
|
|
|
|
className="project-remixes"
|
|
|
|
key="remixes"
|
|
|
|
>
|
|
|
|
{projectInfo.remix.count}
|
|
|
|
</div>
|
|
|
|
<div
|
|
|
|
className="project-views"
|
|
|
|
key="views"
|
|
|
|
>
|
|
|
|
<CappedNumber value={projectInfo.stats.views} />
|
2018-05-24 09:57:06 -04:00
|
|
|
</div>
|
|
|
|
</FlexRow>
|
2018-05-24 13:47:31 -04:00
|
|
|
<FlexRow className="subactions">
|
|
|
|
<div className="share-date">
|
|
|
|
<div className="copyleft">©</div>
|
|
|
|
{' '}
|
|
|
|
{/* eslint-disable react/jsx-sort-props */}
|
|
|
|
{shareDate === null ?
|
|
|
|
'Unshared' :
|
|
|
|
<FormattedDate
|
|
|
|
value={Date.parse(shareDate)}
|
|
|
|
day="2-digit"
|
|
|
|
month="short"
|
|
|
|
year="numeric"
|
|
|
|
/>
|
|
|
|
}
|
|
|
|
{/* eslint-enable react/jsx-sort-props */}
|
|
|
|
</div>
|
2018-05-24 09:57:06 -04:00
|
|
|
|
2018-05-24 13:47:31 -04:00
|
|
|
<FlexRow className="action-buttons">
|
|
|
|
<Button className="action-button studio-button">
|
|
|
|
Add to Studio
|
|
|
|
</Button>
|
|
|
|
<Button className="action-button copy-link-button">
|
|
|
|
Copy Link
|
|
|
|
</Button>
|
|
|
|
{
|
|
|
|
sessionStatus === sessionActions.Status.FETCHED &&
|
|
|
|
Object.keys(user).length > 0 &&
|
|
|
|
user.id !== projectInfo.author.id && [
|
|
|
|
<Button
|
|
|
|
className="action-button report-button"
|
|
|
|
key="report-button"
|
|
|
|
onClick={this.handleReportClick}
|
|
|
|
>
|
|
|
|
Report
|
|
|
|
</Button>,
|
|
|
|
<ReportModal
|
|
|
|
isOpen={this.state.reportOpen}
|
|
|
|
key="report-modal"
|
2018-05-27 18:00:56 -04:00
|
|
|
type="project"
|
|
|
|
onReport={this.handleReportSubmit}
|
2018-05-24 13:47:31 -04:00
|
|
|
onRequestClose={this.handleReportClose}
|
2018-05-27 18:00:56 -04:00
|
|
|
/>
|
2018-05-24 13:47:31 -04:00
|
|
|
]
|
|
|
|
}
|
|
|
|
</FlexRow>
|
2018-05-24 09:57:06 -04:00
|
|
|
</FlexRow>
|
2018-04-24 11:00:47 -04:00
|
|
|
</FlexRow>
|
2018-05-24 13:47:31 -04:00
|
|
|
<FlexRow className="preview-row">
|
|
|
|
<div className="comments-container">
|
2018-04-24 11:00:47 -04:00
|
|
|
<div className="project-title">
|
2018-05-24 13:47:31 -04:00
|
|
|
Comments go here
|
2018-04-24 11:00:47 -04:00
|
|
|
</div>
|
2018-05-24 13:47:31 -04:00
|
|
|
</div>
|
|
|
|
<FlexRow className="column">
|
|
|
|
{/* hide remixes if there aren't any */}
|
|
|
|
{remixes && remixes.length !== 0 && (
|
|
|
|
<FlexRow className="remix-list">
|
|
|
|
<div className="project-title">
|
|
|
|
Remixes
|
|
|
|
</div>
|
|
|
|
{remixes && remixes.length === 0 ? (
|
|
|
|
// TODO: style remix invitation
|
|
|
|
<span>Invite user to remix</span>
|
|
|
|
) : (
|
|
|
|
<ThumbnailColumn
|
|
|
|
cards
|
|
|
|
showAvatar
|
|
|
|
itemType="preview"
|
|
|
|
items={remixes.slice(0, 5)}
|
|
|
|
showFavorites={false}
|
|
|
|
showLoves={false}
|
|
|
|
showViews={false}
|
|
|
|
/>
|
|
|
|
)}
|
|
|
|
</FlexRow>
|
2018-04-24 11:00:47 -04:00
|
|
|
)}
|
2018-05-24 13:47:31 -04:00
|
|
|
{/* hide studios if there aren't any */}
|
|
|
|
{studios && studios.length !== 0 && (
|
|
|
|
<FlexRow className="studio-list">
|
|
|
|
<div className="project-title">
|
|
|
|
Studios
|
|
|
|
</div>
|
|
|
|
{studios && studios.length === 0 ? (
|
|
|
|
// TODO: invite user to add to studio?
|
|
|
|
<span>None </span>
|
|
|
|
) : (
|
|
|
|
<ThumbnailColumn
|
|
|
|
cards
|
|
|
|
showAvatar
|
|
|
|
itemType="gallery"
|
|
|
|
items={studios.slice(0, 5)}
|
|
|
|
showFavorites={false}
|
|
|
|
showLoves={false}
|
|
|
|
showViews={false}
|
|
|
|
/>
|
|
|
|
)}
|
|
|
|
</FlexRow>
|
2018-04-24 11:00:47 -04:00
|
|
|
)}
|
|
|
|
</FlexRow>
|
|
|
|
</FlexRow>
|
2018-05-24 13:47:31 -04:00
|
|
|
</Formsy>
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
2018-03-08 15:57:19 -05:00
|
|
|
|
2018-03-15 17:40:16 -04:00
|
|
|
PreviewPresentation.propTypes = {
|
2018-05-24 09:57:06 -04:00
|
|
|
editable: PropTypes.bool,
|
|
|
|
faved: PropTypes.bool,
|
|
|
|
favoriteCount: PropTypes.number,
|
|
|
|
intl: intlShape,
|
|
|
|
isFullScreen: PropTypes.bool,
|
|
|
|
loveCount: PropTypes.number,
|
|
|
|
loved: PropTypes.bool,
|
|
|
|
onFavoriteClicked: PropTypes.func,
|
|
|
|
onLoveClicked: PropTypes.func,
|
|
|
|
onSeeInside: PropTypes.func,
|
|
|
|
onUpdate: PropTypes.func,
|
|
|
|
originalInfo: PropTypes.shape({
|
|
|
|
id: PropTypes.number,
|
|
|
|
title: PropTypes.string,
|
|
|
|
description: PropTypes.string,
|
|
|
|
author: PropTypes.shape({id: PropTypes.number}),
|
|
|
|
history: PropTypes.shape({
|
|
|
|
created: PropTypes.string,
|
|
|
|
modified: PropTypes.string,
|
|
|
|
shared: PropTypes.string
|
|
|
|
}),
|
|
|
|
stats: PropTypes.shape({
|
|
|
|
views: PropTypes.number,
|
|
|
|
loves: PropTypes.number,
|
|
|
|
favorites: PropTypes.number
|
|
|
|
}),
|
|
|
|
remix: PropTypes.shape({
|
|
|
|
parent: PropTypes.number,
|
|
|
|
root: PropTypes.number
|
|
|
|
})
|
|
|
|
}),
|
|
|
|
parentInfo: PropTypes.shape({
|
2018-03-14 15:50:27 -04:00
|
|
|
id: PropTypes.number,
|
|
|
|
title: PropTypes.string,
|
|
|
|
description: PropTypes.string,
|
|
|
|
author: PropTypes.shape({id: PropTypes.number}),
|
|
|
|
history: PropTypes.shape({
|
|
|
|
created: PropTypes.string,
|
|
|
|
modified: PropTypes.string,
|
|
|
|
shared: PropTypes.string
|
|
|
|
}),
|
|
|
|
stats: PropTypes.shape({
|
|
|
|
views: PropTypes.number,
|
|
|
|
loves: PropTypes.number,
|
|
|
|
favorites: PropTypes.number
|
|
|
|
}),
|
|
|
|
remix: PropTypes.shape({
|
|
|
|
parent: PropTypes.number,
|
|
|
|
root: PropTypes.number
|
|
|
|
})
|
|
|
|
}),
|
2018-05-24 16:23:07 -04:00
|
|
|
projectId: PropTypes.string,
|
2018-03-08 15:57:19 -05:00
|
|
|
projectInfo: PropTypes.shape({
|
|
|
|
id: PropTypes.number,
|
|
|
|
title: PropTypes.string,
|
|
|
|
description: PropTypes.string,
|
|
|
|
author: PropTypes.shape({id: PropTypes.number}),
|
|
|
|
history: PropTypes.shape({
|
|
|
|
created: PropTypes.string,
|
|
|
|
modified: PropTypes.string,
|
|
|
|
shared: PropTypes.string
|
|
|
|
}),
|
|
|
|
stats: PropTypes.shape({
|
|
|
|
views: PropTypes.number,
|
|
|
|
loves: PropTypes.number,
|
|
|
|
favorites: PropTypes.number
|
|
|
|
}),
|
|
|
|
remix: PropTypes.shape({
|
|
|
|
parent: PropTypes.number,
|
|
|
|
root: PropTypes.number
|
|
|
|
})
|
|
|
|
}),
|
2018-04-24 11:00:47 -04:00
|
|
|
remixes: PropTypes.arrayOf(PropTypes.object),
|
|
|
|
sessionStatus: PropTypes.string.isRequired,
|
|
|
|
studios: PropTypes.arrayOf(PropTypes.object),
|
|
|
|
user: PropTypes.shape({
|
|
|
|
id: PropTypes.number,
|
|
|
|
banned: PropTypes.bool,
|
|
|
|
username: PropTypes.string,
|
|
|
|
token: PropTypes.string,
|
|
|
|
thumbnailUrl: PropTypes.string,
|
|
|
|
dateJoined: PropTypes.string,
|
|
|
|
email: PropTypes.string,
|
|
|
|
classroomId: PropTypes.string
|
|
|
|
}).isRequired
|
2018-03-15 17:40:16 -04:00
|
|
|
};
|
2018-03-08 15:57:19 -05:00
|
|
|
|
2018-03-15 17:40:16 -04:00
|
|
|
module.exports = injectIntl(PreviewPresentation);
|