@ -4,7 +4,7 @@ import PropTypes from 'prop-types';
import ImmutablePureComponent from 'react-immutable-pure-component' ;
import ImmutablePureComponent from 'react-immutable-pure-component' ;
import { connect } from 'react-redux' ;
import { connect } from 'react-redux' ;
import classNames from 'classnames' ;
import classNames from 'classnames' ;
import { changeUploadCompose } from '../../../actions/compose' ;
import { changeUploadCompose , uploadThumbnail } from '../../../actions/compose' ;
import { getPointerPosition } from '../../video' ;
import { getPointerPosition } from '../../video' ;
import { FormattedMessage , defineMessages , injectIntl } from 'react-intl' ;
import { FormattedMessage , defineMessages , injectIntl } from 'react-intl' ;
import IconButton from 'mastodon/components/icon_button' ;
import IconButton from 'mastodon/components/icon_button' ;
@ -23,11 +23,13 @@ const messages = defineMessages({
close : { id : 'lightbox.close' , defaultMessage : 'Close' } ,
close : { id : 'lightbox.close' , defaultMessage : 'Close' } ,
apply : { id : 'upload_modal.apply' , defaultMessage : 'Apply' } ,
apply : { id : 'upload_modal.apply' , defaultMessage : 'Apply' } ,
placeholder : { id : 'upload_modal.description_placeholder' , defaultMessage : 'A quick brown fox jumps over the lazy dog' } ,
placeholder : { id : 'upload_modal.description_placeholder' , defaultMessage : 'A quick brown fox jumps over the lazy dog' } ,
chooseImage : { id : 'upload_modal.choose_image' , defaultMessage : 'Choose image' } ,
} ) ;
} ) ;
const mapStateToProps = ( state , { id } ) => ( {
const mapStateToProps = ( state , { id } ) => ( {
media : state . getIn ( [ 'compose' , 'media_attachments' ] ) . find ( item => item . get ( 'id' ) === id ) ,
media : state . getIn ( [ 'compose' , 'media_attachments' ] ) . find ( item => item . get ( 'id' ) === id ) ,
account : state . getIn ( [ 'accounts' , me ] ) ,
account : state . getIn ( [ 'accounts' , me ] ) ,
isUploadingThumbnail : state . getIn ( [ 'compose' , 'isUploadingThumbnail' ] ) ,
} ) ;
} ) ;
const mapDispatchToProps = ( dispatch , { id } ) => ( {
const mapDispatchToProps = ( dispatch , { id } ) => ( {
@ -36,6 +38,10 @@ const mapDispatchToProps = (dispatch, { id }) => ({
dispatch ( changeUploadCompose ( id , { description , focus : ` ${ x . toFixed ( 2 ) } , ${ y . toFixed ( 2 ) } ` } ) ) ;
dispatch ( changeUploadCompose ( id , { description , focus : ` ${ x . toFixed ( 2 ) } , ${ y . toFixed ( 2 ) } ` } ) ) ;
} ,
} ,
onSelectThumbnail : files => {
dispatch ( uploadThumbnail ( id , files [ 0 ] ) ) ;
} ,
} ) ;
} ) ;
const removeExtraLineBreaks = str => str . replace ( /\n\n/g , '******' )
const removeExtraLineBreaks = str => str . replace ( /\n\n/g , '******' )
@ -81,6 +87,9 @@ class FocalPointModal extends ImmutablePureComponent {
static propTypes = {
static propTypes = {
media : ImmutablePropTypes . map . isRequired ,
media : ImmutablePropTypes . map . isRequired ,
account : ImmutablePropTypes . map . isRequired ,
account : ImmutablePropTypes . map . isRequired ,
isUploadingThumbnail : PropTypes . bool ,
onSave : PropTypes . func . isRequired ,
onSelectThumbnail : PropTypes . func . isRequired ,
onClose : PropTypes . func . isRequired ,
onClose : PropTypes . func . isRequired ,
intl : PropTypes . object . isRequired ,
intl : PropTypes . object . isRequired ,
} ;
} ;
@ -235,13 +244,29 @@ class FocalPointModal extends ImmutablePureComponent {
} ) . catch ( ( ) => this . setState ( { detecting : false } ) ) ;
} ) . catch ( ( ) => this . setState ( { detecting : false } ) ) ;
}
}
handleThumbnailChange = e => {
if ( e . target . files . length > 0 ) {
this . setState ( { dirty : true } ) ;
this . props . onSelectThumbnail ( e . target . files ) ;
}
}
setFileInputRef = c => {
this . fileInput = c ;
}
handleFileInputClick = ( ) => {
this . fileInput . click ( ) ;
}
render ( ) {
render ( ) {
const { media , intl , account , onClose } = this . props ;
const { media , intl , account , onClose , isUploadingThumbnail } = this . props ;
const { x , y , dragging , description , dirty , detecting , progress } = this . state ;
const { x , y , dragging , description , dirty , detecting , progress } = this . state ;
const width = media . getIn ( [ 'meta' , 'original' , 'width' ] ) || null ;
const width = media . getIn ( [ 'meta' , 'original' , 'width' ] ) || null ;
const height = media . getIn ( [ 'meta' , 'original' , 'height' ] ) || null ;
const height = media . getIn ( [ 'meta' , 'original' , 'height' ] ) || null ;
const focals = [ 'image' , 'gifv' ] . includes ( media . get ( 'type' ) ) ;
const focals = [ 'image' , 'gifv' ] . includes ( media . get ( 'type' ) ) ;
const thumbnailable = [ 'audio' , 'video' ] . includes ( media . get ( 'type' ) ) ;
const previewRatio = 16 / 9 ;
const previewRatio = 16 / 9 ;
const previewWidth = 200 ;
const previewWidth = 200 ;
@ -268,6 +293,30 @@ class FocalPointModal extends ImmutablePureComponent {
< div className = 'report-modal__comment' >
< div className = 'report-modal__comment' >
{ focals && < p > < FormattedMessage id = 'upload_modal.hint' defaultMessage = 'Click or drag the circle on the preview to choose the focal point which will always be in view on all thumbnails.' / > < / p > }
{ focals && < p > < FormattedMessage id = 'upload_modal.hint' defaultMessage = 'Click or drag the circle on the preview to choose the focal point which will always be in view on all thumbnails.' / > < / p > }
{ thumbnailable && (
< React . Fragment >
< label className = 'setting-text-label' htmlFor = 'upload-modal__thumbnail' > < FormattedMessage id = 'upload_form.thumbnail' defaultMessage = 'Change thumbnail' / > < / l a b e l >
< Button disabled = { isUploadingThumbnail } text = { intl . formatMessage ( messages . chooseImage ) } onClick = { this . handleFileInputClick } / >
< label >
< span style = { { display : 'none' } } > { intl . formatMessage ( messages . chooseImage ) } < / s p a n >
< input
id = 'upload-modal__thumbnail'
ref = { this . setFileInputRef }
type = 'file'
accept = 'image/png,image/jpeg'
onChange = { this . handleThumbnailChange }
style = { { display : 'none' } }
disabled = { isUploadingThumbnail }
/ >
< / l a b e l >
< hr className = 'setting-divider' / >
< / R e a c t . F r a g m e n t >
) }
< label className = 'setting-text-label' htmlFor = 'upload-modal__description' >
< label className = 'setting-text-label' htmlFor = 'upload-modal__description' >
{ descriptionLabel }
{ descriptionLabel }
< / l a b e l >
< / l a b e l >
@ -293,7 +342,7 @@ class FocalPointModal extends ImmutablePureComponent {
< CharacterCounter max = { 1500 } text = { detecting ? '' : description } / >
< CharacterCounter max = { 1500 } text = { detecting ? '' : description } / >
< / d i v >
< / d i v >
< Button disabled = { ! dirty || detecting || length ( description ) > 1500 } text = { intl . formatMessage ( messages . apply ) } onClick = { this . handleSubmit } / >
< Button disabled = { ! dirty || detecting || isUploadingThumbnail || length ( description ) > 1500 } text = { intl . formatMessage ( messages . apply ) } onClick = { this . handleSubmit } / >
< / d i v >
< / d i v >
< div className = 'focal-point-modal__content' >
< div className = 'focal-point-modal__content' >