2020-07-20 16:18:54 -07:00
import React from 'react' ;
import { connect } from 'react-redux' ;
import { defineMessages , injectIntl , FormattedMessage } from 'react-intl' ;
import ImmutablePureComponent from 'react-immutable-pure-component' ;
import PropTypes from 'prop-types' ;
import ImmutablePropTypes from 'react-immutable-proptypes' ;
import Column from '../ui/components/column' ;
import {
SimpleForm ,
FieldsGroup ,
2020-07-21 06:30:51 -07:00
TextInput ,
Checkbox ,
FileChooser ,
2020-07-20 16:18:54 -07:00
} from 'soapbox/features/forms' ;
2020-07-21 06:30:51 -07:00
import {
Map as ImmutableMap ,
List as ImmutableList ,
} from 'immutable' ;
import { patchMe } from 'soapbox/actions/me' ;
import { unescape } from 'lodash' ;
2020-07-20 16:18:54 -07:00
const messages = defineMessages ( {
2020-07-21 06:30:51 -07:00
heading : { id : 'column.edit_profile' , defaultMessage : 'Edit profile' } ,
metaFieldLabel : { id : 'edit_profile.fields.meta_fields.label_placeholder' , defaultMessage : 'Label' } ,
metaFieldContent : { id : 'edit_profile.fields.meta_fields.content_placeholder' , defaultMessage : 'Content' } ,
2020-07-20 16:18:54 -07:00
} ) ;
2020-07-21 06:30:51 -07:00
const mapStateToProps = state => {
const soapbox = state . get ( 'soapbox' ) ;
return {
themeCss : generateThemeCss ( state . getIn ( [ 'soapbox' , 'brandColor' ] ) ) ,
customCss : state . getIn ( [ 'soapbox' , 'customCss' ] ) ,
logo : state . getIn ( [ 'soapbox' , 'logo' ] ) ,
logo : state . getIn ( [ 'soapbox' , 'logo' ] ) ,
promoPanel : state . getIn ( [ 'soapbox' , 'promoPanel' , 'items' ] ) ,
patronEnabled : state . getIn ( [ 'soapbox' , 'extensions' , 'patron' , 'enabled' ] ) ,
displayMode : state . getIn ( [ 'soapbox' , 'defaultSettings' , 'mode' ] ) ,
copyright : state . getIn ( [ 'soapbox' , 'copyright' ] ) ,
homeFooter : state . getIn ( [ 'soapbox' , 'navLinks' , 'homeFooter' ] ) ,
// {
// soapbox: {
// brandColor: '#1ca82b',
// logo: '/favicon.png',
// promoPanel: {
// items: [
// {
// icon: 'area-chart',
// text: 'Gleasonator stats',
// url: 'https://fediverse.network/gleasonator.com'
// },
// {
// icon: 'comment-o',
// text: 'Gleasonator theme song',
// url: 'https://media.gleasonator.com/custom/261905_gleasonator_song.mp3'
// }
// ]
// },
// extensions: {
// patron: {
// enabled: true
// }
// },
// defaultSettings: {
// mode: 'light'
// },
// copyright: '♡2020. Copying is an act of love. Please copy and share.',
// navlinks: {
// homeFooter: [
// {
// title: 'About',
// url: '/about'
// },
// {
// title: 'Terms of Service',
// url: '/about/tos'
// },
// {
// title: 'Privacy Policy',
// url: '/about/privacy'
// },
// {
// title: 'DMCA',
// url: '/about/dmca'
// },
// {
// title: 'Source Code',
// url: '/about#opensource'
// }
// ]
// }
// }
// }
} ;
} ;
// HTML unescape for special chars, eg <br>
const unescapeParams = ( map , params ) => (
params . reduce ( ( map , param ) => (
map . set ( param , unescape ( map . get ( param ) ) )
) , map )
) ;
2020-07-20 16:18:54 -07:00
export default @ connect ( mapStateToProps )
@ injectIntl
2020-07-21 06:30:51 -07:00
class ConfigSoapbox extends ImmutablePureComponent {
2020-07-20 16:18:54 -07:00
static propTypes = {
dispatch : PropTypes . func . isRequired ,
intl : PropTypes . object . isRequired ,
2020-07-21 06:30:51 -07:00
account : ImmutablePropTypes . map ,
2020-07-20 16:18:54 -07:00
} ;
2020-07-21 06:30:51 -07:00
state = {
isLoading : false ,
fields : normalizeFields ( Array . from ( { length : MAX _FIELDS } ) ) ,
}
constructor ( props ) {
super ( props ) ;
const initialState = props . account . withMutations ( map => {
map . merge ( map . get ( 'source' ) ) ;
map . delete ( 'source' ) ;
map . set ( 'fields' , normalizeFields ( map . get ( 'fields' ) ) ) ;
unescapeParams ( map , [ 'display_name' , 'note' ] ) ;
} ) ;
this . state = initialState . toObject ( ) ;
}
makePreviewAccount = ( ) => {
const { account } = this . props ;
return account . merge ( ImmutableMap ( {
header : this . state . header ,
avatar : this . state . avatar ,
display _name : this . state . display _name ,
} ) ) ;
}
getFieldParams = ( ) => {
let params = ImmutableMap ( ) ;
this . state . fields . forEach ( ( f , i ) =>
params = params
. set ( ` fields_attributes[ ${ i } ][name] ` , f . get ( 'name' ) )
. set ( ` fields_attributes[ ${ i } ][value] ` , f . get ( 'value' ) )
) ;
return params ;
}
getParams = ( ) => {
const { state } = this ;
return Object . assign ( {
discoverable : state . discoverable ,
bot : state . bot ,
display _name : state . display _name ,
note : state . note ,
avatar : state . avatar _file ,
header : state . header _file ,
locked : state . locked ,
} , this . getFieldParams ( ) . toJS ( ) ) ;
}
2020-07-20 16:18:54 -07:00
2020-07-21 06:30:51 -07:00
getFormdata = ( ) => {
const data = this . getParams ( ) ;
let formData = new FormData ( ) ;
for ( let key in data ) {
const shouldAppend = Boolean ( data [ key ] || key . startsWith ( 'fields_attributes' ) ) ;
if ( shouldAppend ) formData . append ( key , data [ key ] || '' ) ;
}
return formData ;
}
handleSubmit = ( event ) => {
2020-07-20 16:18:54 -07:00
const { dispatch } = this . props ;
2020-07-21 06:30:51 -07:00
dispatch ( patchMe ( this . getFormdata ( ) ) ) . then ( ( ) => {
this . setState ( { isLoading : false } ) ;
} ) . catch ( ( error ) => {
this . setState ( { isLoading : false } ) ;
} ) ;
this . setState ( { isLoading : true } ) ;
event . preventDefault ( ) ;
}
handleCheckboxChange = e => {
this . setState ( { [ e . target . name ] : e . target . checked } ) ;
}
handleTextChange = e => {
this . setState ( { [ e . target . name ] : e . target . value } ) ;
}
handleFieldChange = ( i , key ) => {
return ( e ) => {
this . setState ( {
fields : this . state . fields . setIn ( [ i , key ] , e . target . value ) ,
} ) ;
} ;
}
handleFileChange = e => {
const { name } = e . target ;
const [ file ] = e . target . files || [ ] ;
const url = file ? URL . createObjectURL ( file ) : this . state [ name ] ;
this . setState ( {
[ name ] : url ,
[ ` ${ name } _file ` ] : file ,
} ) ;
2020-07-20 16:18:54 -07:00
}
render ( ) {
2020-07-21 06:30:51 -07:00
const { intl } = this . props ;
2020-07-20 16:18:54 -07:00
return (
2020-07-21 06:30:51 -07:00
< Column icon = 'user' heading = { intl . formatMessage ( messages . heading ) } backBtnSlim >
< SimpleForm onSubmit = { this . handleSubmit } >
< fieldset disabled = { this . state . isLoading } >
< FieldsGroup >
< TextInput
label = { < FormattedMessage id = 'edit_profile.fields.display_name_label' defaultMessage = 'Display name' / > }
name = 'display_name'
value = { this . state . display _name }
onChange = { this . handleTextChange }
2020-07-20 16:18:54 -07:00
/ >
2020-07-21 06:30:51 -07:00
< TextInput
label = { < FormattedMessage id = 'edit_profile.fields.bio_label' defaultMessage = 'Bio' / > }
name = 'note'
value = { this . state . note }
onChange = { this . handleTextChange }
2020-07-20 16:18:54 -07:00
/ >
2020-07-21 06:30:51 -07:00
< div className = 'fields-row' >
< div className = 'fields-row__column fields-row__column-6' >
< ProfilePreview account = { this . makePreviewAccount ( ) } / >
< / d i v >
< div className = 'fields-row__column fields-group fields-row__column-6' >
< FileChooser
label = { < FormattedMessage id = 'edit_profile.fields.header_label' defaultMessage = 'Header' / > }
name = 'header'
hint = { < FormattedMessage id = 'edit_profile.hints.header' defaultMessage = 'PNG, GIF or JPG. At most 2 MB. Will be downscaled to 1500x500px' / > }
onChange = { this . handleFileChange }
/ >
< FileChooser
label = { < FormattedMessage id = 'edit_profile.fields.avatar_label' defaultMessage = 'Avatar' / > }
name = 'avatar'
hint = { < FormattedMessage id = 'edit_profile.hints.avatar' defaultMessage = 'PNG, GIF or JPG. At most 2 MB. Will be downscaled to 400x400px' / > }
onChange = { this . handleFileChange }
/ >
< / d i v >
< / d i v >
< Checkbox
label = { < FormattedMessage id = 'edit_profile.fields.locked_label' defaultMessage = 'Lock account' / > }
hint = { < FormattedMessage id = 'edit_profile.hints.locked' defaultMessage = 'Requires you to manually approve followers' / > }
name = 'locked'
checked = { this . state . locked }
onChange = { this . handleCheckboxChange }
2020-07-20 16:18:54 -07:00
/ >
2020-07-21 06:30:51 -07:00
< Checkbox
label = { < FormattedMessage id = 'edit_profile.fields.bot_label' defaultMessage = 'This is a bot account' / > }
hint = { < FormattedMessage id = 'edit_profile.hints.bot' defaultMessage = 'This account mainly performs automated actions and might not be monitored' / > }
name = 'bot'
checked = { this . state . bot }
onChange = { this . handleCheckboxChange }
2020-07-20 16:18:54 -07:00
/ >
2020-07-21 06:30:51 -07:00
< / F i e l d s G r o u p >
< FieldsGroup >
< div className = 'fields-row__column fields-group' >
< div className = 'input with_block_label' >
< label > < FormattedMessage id = 'edit_profile.fields.meta_fields_label' defaultMessage = 'Profile metadata' / > < / l a b e l >
< span className = 'hint' >
< FormattedMessage id = 'edit_profile.hints.meta_fields' defaultMessage = 'You can have up to {count, plural, one {# item} other {# items}} displayed as a table on your profile' values = { { count : MAX _FIELDS } } / >
< / s p a n >
{
this . state . fields . map ( ( field , i ) => (
< div className = 'row' key = { i } >
< TextInput
placeholder = { intl . formatMessage ( messages . metaFieldLabel ) }
value = { field . get ( 'name' ) }
onChange = { this . handleFieldChange ( i , 'name' ) }
/ >
< TextInput
placeholder = { intl . formatMessage ( messages . metaFieldContent ) }
value = { field . get ( 'value' ) }
onChange = { this . handleFieldChange ( i , 'value' ) }
/ >
< / d i v >
) )
}
< / d i v >
< / d i v >
< / F i e l d s G r o u p >
< / f i e l d s e t >
< div className = 'actions' >
< button name = 'button' type = 'submit' className = 'btn button button-primary' >
< FormattedMessage id = 'edit_profile.save' defaultMessage = 'Save' / >
< / b u t t o n >
< / d i v >
2020-07-20 16:18:54 -07:00
< / S i m p l e F o r m >
< / C o l u m n >
) ;
}
}