2022-01-10 14:01:24 -08:00
import {
Map as ImmutableMap ,
List as ImmutableList ,
} from 'immutable' ;
import { unescape } from 'lodash' ;
2022-01-10 14:17:52 -08:00
import PropTypes from 'prop-types' ;
import React from 'react' ;
import ImmutablePropTypes from 'react-immutable-proptypes' ;
import ImmutablePureComponent from 'react-immutable-pure-component' ;
import { defineMessages , injectIntl , FormattedMessage } from 'react-intl' ;
import { connect } from 'react-redux' ;
2022-01-10 14:25:06 -08:00
2022-03-21 11:09:01 -07:00
// import { updateNotificationSettings } from 'soapbox/actions/accounts';
2022-01-10 14:17:52 -08:00
import { patchMe } from 'soapbox/actions/me' ;
import snackbar from 'soapbox/actions/snackbar' ;
import { getSoapboxConfig } from 'soapbox/actions/soapbox' ;
2022-03-21 11:09:01 -07:00
// import Icon from 'soapbox/components/icon';
2020-04-21 16:00:05 -07:00
import {
2020-04-21 17:22:00 -07:00
Checkbox ,
2020-05-28 15:52:07 -07:00
} from 'soapbox/features/forms' ;
2022-01-10 14:17:52 -08:00
import { makeGetAccount } from 'soapbox/selectors' ;
2021-06-15 13:02:36 -07:00
import { getFeatures } from 'soapbox/utils/features' ;
2021-10-19 10:03:04 -07:00
import resizeImage from 'soapbox/utils/resize_image' ;
2022-01-10 14:25:06 -08:00
2022-03-21 12:23:57 -07:00
import { Button , Column , Form , FormActions , FormGroup , Input , Textarea } from '../../components/ui' ;
2022-01-10 14:25:06 -08:00
2022-01-10 14:01:24 -08:00
import ProfilePreview from './components/profile_preview' ;
2020-04-21 16:00:05 -07:00
2021-06-16 12:20:57 -07:00
const hidesNetwork = account => {
const pleroma = account . get ( 'pleroma' ) ;
if ( ! pleroma ) return false ;
const { hide _followers , hide _follows , hide _followers _count , hide _follows _count } = pleroma . toJS ( ) ;
return hide _followers && hide _follows && hide _followers _count && hide _follows _count ;
} ;
2020-04-21 16:00:05 -07:00
const messages = defineMessages ( {
heading : { id : 'column.edit_profile' , defaultMessage : 'Edit profile' } ,
2020-06-06 13:52:33 -07:00
metaFieldLabel : { id : 'edit_profile.fields.meta_fields.label_placeholder' , defaultMessage : 'Label' } ,
metaFieldContent : { id : 'edit_profile.fields.meta_fields.content_placeholder' , defaultMessage : 'Content' } ,
2020-08-05 10:47:45 -07:00
verified : { id : 'edit_profile.fields.verified_display_name' , defaultMessage : 'Verified users may not update their display name' } ,
2021-06-26 15:04:27 -07:00
success : { id : 'edit_profile.success' , defaultMessage : 'Profile saved!' } ,
2021-10-19 10:03:04 -07:00
error : { id : 'edit_profile.error' , defaultMessage : 'Profile update failed' } ,
2021-08-23 14:03:35 -07:00
bioPlaceholder : { id : 'edit_profile.fields.bio_placeholder' , defaultMessage : 'Tell us about yourself.' } ,
displayNamePlaceholder : { id : 'edit_profile.fields.display_name_placeholder' , defaultMessage : 'Name' } ,
2022-03-21 11:09:01 -07:00
websitePlaceholder : { id : 'edit_profile.fields.website_placeholder' , defaultMessage : 'Display a Link' } ,
locationPlaceholder : { id : 'edit_profile.fields.location_placeholder' , defaultMessage : 'Location' } ,
cancel : { id : 'common.cancel' , defaultMessage : 'Cancel' } ,
2020-04-21 16:00:05 -07:00
} ) ;
2021-08-23 15:51:32 -07:00
const makeMapStateToProps = ( ) => {
const getAccount = makeGetAccount ( ) ;
2021-08-23 14:03:35 -07:00
2021-08-23 15:51:32 -07:00
const mapStateToProps = state => {
const me = state . get ( 'me' ) ;
const account = getAccount ( state , me ) ;
const soapbox = getSoapboxConfig ( state ) ;
2021-08-23 14:03:35 -07:00
2021-08-23 15:51:32 -07:00
return {
account ,
maxFields : state . getIn ( [ 'instance' , 'pleroma' , 'metadata' , 'fields_limits' , 'max_fields' ] , 4 ) ,
verifiedCanEditName : soapbox . get ( 'verifiedCanEditName' ) ,
2022-03-21 11:09:01 -07:00
supportsEmailList : getFeatures ( state . get ( 'instance' ) ) . emailList ,
2021-08-23 15:51:32 -07:00
} ;
2020-04-21 16:00:05 -07:00
} ;
2021-08-23 15:51:32 -07:00
return mapStateToProps ;
2020-04-21 16:00:05 -07:00
} ;
2020-08-08 18:32:39 -07:00
// Forces fields to be maxFields size, filling empty values
const normalizeFields = ( fields , maxFields ) => (
2021-06-30 07:15:44 -07:00
ImmutableList ( fields ) . setSize ( Math . max ( fields . size , maxFields ) ) . map ( field =>
2020-10-07 11:08:36 -07:00
field ? field : ImmutableMap ( { name : '' , value : '' } ) ,
2020-05-18 10:20:39 -07:00
)
) ;
2020-05-28 13:04:56 -07:00
// HTML unescape for special chars, eg <br>
const unescapeParams = ( map , params ) => (
params . reduce ( ( map , param ) => (
map . set ( param , unescape ( map . get ( param ) ) )
) , map )
) ;
2021-08-23 15:51:32 -07:00
export default @ connect ( makeMapStateToProps )
2020-04-21 16:00:05 -07:00
@ injectIntl
class EditProfile extends ImmutablePureComponent {
static propTypes = {
dispatch : PropTypes . func . isRequired ,
intl : PropTypes . object . isRequired ,
2022-03-23 10:14:42 -07:00
account : ImmutablePropTypes . record ,
2020-08-08 18:32:39 -07:00
maxFields : PropTypes . number ,
2021-03-15 20:23:33 -07:00
verifiedCanEditName : PropTypes . bool ,
2020-04-21 16:00:05 -07:00
} ;
2020-04-21 17:22:00 -07:00
state = {
isLoading : false ,
2020-04-21 16:00:05 -07:00
}
2020-07-04 17:32:07 -07:00
constructor ( props ) {
super ( props ) ;
2021-08-23 14:03:35 -07:00
const { account , maxFields } = this . props ;
2021-06-06 20:43:18 -07:00
const strangerNotifications = account . getIn ( [ 'pleroma' , 'notification_settings' , 'block_from_strangers' ] ) ;
2021-06-14 20:07:48 -07:00
const acceptsEmailList = account . getIn ( [ 'pleroma' , 'accepts_email_list' ] ) ;
2021-12-26 13:44:43 -08:00
const discoverable = account . getIn ( [ 'source' , 'pleroma' , 'discoverable' ] ) ;
2021-08-23 14:03:35 -07:00
2022-03-08 22:12:27 -08:00
const initialState = ImmutableMap ( account ) . withMutations ( map => {
2020-07-04 17:32:07 -07:00
map . merge ( map . get ( 'source' ) ) ;
map . delete ( 'source' ) ;
2021-08-23 14:03:35 -07:00
map . set ( 'fields' , normalizeFields ( map . get ( 'fields' ) , Math . min ( maxFields , 4 ) ) ) ;
2021-06-06 20:43:18 -07:00
map . set ( 'stranger_notifications' , strangerNotifications ) ;
2021-06-14 20:07:48 -07:00
map . set ( 'accepts_email_list' , acceptsEmailList ) ;
2021-06-16 12:20:57 -07:00
map . set ( 'hide_network' , hidesNetwork ( account ) ) ;
2021-12-26 13:44:43 -08:00
map . set ( 'discoverable' , discoverable ) ;
2020-08-09 13:13:36 -07:00
unescapeParams ( map , [ 'display_name' , 'bio' ] ) ;
2020-07-04 17:32:07 -07:00
} ) ;
2021-08-23 14:03:35 -07:00
2020-07-04 17:32:07 -07:00
this . state = initialState . toObject ( ) ;
}
2020-04-22 15:04:08 -07:00
makePreviewAccount = ( ) => {
const { account } = this . props ;
return account . merge ( ImmutableMap ( {
header : this . state . header ,
avatar : this . state . avatar ,
2021-08-23 14:03:35 -07:00
display _name : this . state . display _name || account . get ( 'username' ) ,
2022-03-21 11:09:01 -07:00
website : this . state . website || account . get ( 'website' ) ,
location : this . state . location || account . get ( 'location' ) ,
2020-04-22 15:04:08 -07:00
} ) ) ;
}
2020-05-18 10:20:39 -07:00
getFieldParams = ( ) => {
let params = ImmutableMap ( ) ;
this . state . fields . forEach ( ( f , i ) =>
params = params
. set ( ` fields_attributes[ ${ i } ][name] ` , f . get ( 'name' ) )
2020-10-07 11:08:36 -07:00
. set ( ` fields_attributes[ ${ i } ][value] ` , f . get ( 'value' ) ) ,
2020-05-18 10:20:39 -07:00
) ;
return params ;
}
2020-04-21 17:22:00 -07:00
getParams = ( ) => {
const { state } = this ;
2020-05-18 10:20:39 -07:00
return Object . assign ( {
2020-04-21 17:22:00 -07:00
discoverable : state . discoverable ,
bot : state . bot ,
display _name : state . display _name ,
2022-03-21 11:09:01 -07:00
website : state . website ,
location : state . location ,
2020-04-21 17:22:00 -07:00
note : state . note ,
2020-04-22 15:04:08 -07:00
avatar : state . avatar _file ,
header : state . header _file ,
2020-04-21 17:22:00 -07:00
locked : state . locked ,
2021-06-14 20:07:48 -07:00
accepts _email _list : state . accepts _email _list ,
2021-06-16 12:20:57 -07:00
hide _followers : state . hide _network ,
hide _follows : state . hide _network ,
hide _followers _count : state . hide _network ,
hide _follows _count : state . hide _network ,
2020-05-18 10:20:39 -07:00
} , this . getFieldParams ( ) . toJS ( ) ) ;
2020-04-21 16:00:05 -07:00
}
2020-04-22 15:04:08 -07:00
getFormdata = ( ) => {
const data = this . getParams ( ) ;
2021-08-03 10:10:42 -07:00
const formData = new FormData ( ) ;
for ( const key in data ) {
2022-03-21 11:09:01 -07:00
const hasValue = data [ key ] !== null && data [ key ] !== undefined ;
2020-08-09 13:13:36 -07:00
// Compact the submission. This should probably be done better.
2022-03-21 11:09:01 -07:00
const shouldAppend = Boolean ( hasValue || key . startsWith ( 'fields_attributes' ) ) ;
if ( shouldAppend ) formData . append ( key , hasValue ? data [ key ] : '' ) ;
2020-04-22 15:04:08 -07:00
}
return formData ;
}
2020-04-21 16:00:05 -07:00
handleSubmit = ( event ) => {
2022-03-21 11:09:01 -07:00
const { dispatch , intl } = this . props ;
2021-06-06 20:43:18 -07:00
const credentials = dispatch ( patchMe ( this . getFormdata ( ) ) ) ;
2022-03-21 11:09:01 -07:00
/* Bad API url, was causing errors in the promise call below blocking the success message after making edits. */
/ * c o n s t n o t i f i c a t i o n s = d i s p a t c h ( u p d a t e N o t i f i c a t i o n S e t t i n g s ( {
2021-06-06 20:43:18 -07:00
block _from _strangers : this . state . stranger _notifications || false ,
2022-03-21 11:09:01 -07:00
} ) ) ; * /
2021-06-06 20:43:18 -07:00
this . setState ( { isLoading : true } ) ;
2022-03-21 11:09:01 -07:00
Promise . all ( [ credentials /*notifications*/ ] ) . then ( ( ) => {
2020-04-21 16:00:05 -07:00
this . setState ( { isLoading : false } ) ;
2022-03-21 11:09:01 -07:00
dispatch ( snackbar . success ( intl . formatMessage ( messages . success ) ) ) ;
2020-04-21 16:00:05 -07:00
} ) . catch ( ( error ) => {
this . setState ( { isLoading : false } ) ;
2021-10-19 10:03:04 -07:00
dispatch ( snackbar . error ( intl . formatMessage ( messages . error ) ) ) ;
2020-04-21 16:00:05 -07:00
} ) ;
2021-06-06 20:43:18 -07:00
2020-04-21 16:00:05 -07:00
event . preventDefault ( ) ;
}
2020-04-21 17:22:00 -07:00
handleCheckboxChange = e => {
this . setState ( { [ e . target . name ] : e . target . checked } ) ;
}
handleTextChange = e => {
this . setState ( { [ e . target . name ] : e . target . value } ) ;
}
2020-05-18 10:20:39 -07:00
handleFieldChange = ( i , key ) => {
return ( e ) => {
this . setState ( {
fields : this . state . fields . setIn ( [ i , key ] , e . target . value ) ,
} ) ;
} ;
}
2021-10-19 10:03:04 -07:00
handleFileChange = maxPixels => {
return e => {
const { name } = e . target ;
const [ f ] = e . target . files || [ ] ;
2020-04-22 15:04:08 -07:00
2021-10-19 10:03:04 -07:00
resizeImage ( f , maxPixels ) . then ( file => {
const url = file ? URL . createObjectURL ( file ) : this . state [ name ] ;
this . setState ( {
[ name ] : url ,
[ ` ${ name } _file ` ] : file ,
} ) ;
} ) . catch ( console . error ) ;
} ;
2020-04-22 15:04:08 -07:00
}
2021-06-30 07:15:44 -07:00
handleAddField = ( ) => {
this . setState ( {
fields : this . state . fields . push ( ImmutableMap ( { name : '' , value : '' } ) ) ,
} ) ;
}
handleDeleteField = i => {
return ( ) => {
this . setState ( {
fields : normalizeFields ( this . state . fields . delete ( i ) , Math . min ( this . props . maxFields , 4 ) ) ,
} ) ;
} ;
}
2020-04-21 16:00:05 -07:00
render ( ) {
2022-03-21 11:09:01 -07:00
const { intl , account , verifiedCanEditName , supportsEmailList /* maxFields */ } = this . props ;
2022-02-27 20:25:23 -08:00
const verified = account . get ( 'verified' ) ;
2021-03-15 20:23:33 -07:00
const canEditName = verifiedCanEditName || ! verified ;
2020-04-21 16:00:05 -07:00
return (
2022-03-21 12:23:57 -07:00
< Column label = 'Edit Profile' >
< Form onSubmit = { this . handleSubmit } >
< FormGroup
labelText = { < FormattedMessage id = 'edit_profile.fields.display_name_label' defaultMessage = 'Display name' / > }
hintText = { ! canEditName && intl . formatMessage ( messages . verified ) }
>
< Input
name = 'display_name'
value = { this . state . display _name }
onChange = { this . handleTextChange }
placeholder = { intl . formatMessage ( messages . displayNamePlaceholder ) }
disabled = { ! canEditName }
/ >
< / F o r m G r o u p >
< FormGroup
labelText = { < FormattedMessage id = 'edit_profile.fields.location_label' defaultMessage = 'Location' / > }
>
< Input
name = 'location'
value = { this . state . location }
onChange = { this . handleTextChange }
placeholder = { intl . formatMessage ( messages . locationPlaceholder ) }
/ >
< / F o r m G r o u p >
< FormGroup
labelText = { < FormattedMessage id = 'edit_profile.fields.website_label' defaultMessage = 'Website' / > }
>
< Input
name = 'website'
value = { this . state . website }
onChange = { this . handleTextChange }
placeholder = { intl . formatMessage ( messages . websitePlaceholder ) }
/ >
< / F o r m G r o u p >
< FormGroup
labelText = { < FormattedMessage id = 'edit_profile.fields.bio_label' defaultMessage = 'Bio' / > }
>
< Textarea
name = 'note'
value = { this . state . note }
onChange = { this . handleTextChange }
autoComplete = 'off'
placeholder = { intl . formatMessage ( messages . bioPlaceholder ) }
/ >
< / F o r m G r o u p >
< div className = 'grid grid-cols-2 gap-4' >
< ProfilePreview account = { this . makePreviewAccount ( ) } / >
< div className = 'space-y-4' >
2022-03-21 11:09:01 -07:00
< FormGroup
2022-03-21 12:23:57 -07:00
labelText = { < FormattedMessage id = 'edit_profile.fields.header_label' defaultMessage = 'Choose Background Picture' / > }
hintText = { < FormattedMessage id = 'edit_profile.hints.header' defaultMessage = 'PNG, GIF or JPG. Will be downscaled to {size}' values = { { size : '1920x1080px' } } / > }
2022-03-21 11:09:01 -07:00
>
2022-03-21 12:23:57 -07:00
< input type = 'file' name = 'header' onChange = { this . handleFileChange ( 1920 * 1080 ) } className = 'text-sm' / >
2022-03-21 11:09:01 -07:00
< / F o r m G r o u p >
< FormGroup
2022-03-21 12:23:57 -07:00
labelText = { < FormattedMessage id = 'edit_profile.fields.avatar_label' defaultMessage = 'Choose Profile Picture' / > }
hintText = { < FormattedMessage id = 'edit_profile.hints.avatar' defaultMessage = 'PNG, GIF or JPG. Will be downscaled to {size}' values = { { size : '400x400px' } } / > }
2022-03-21 11:09:01 -07:00
>
2022-03-21 12:23:57 -07:00
< input type = 'file' name = 'avatar' onChange = { this . handleFileChange ( 400 * 400 ) } className = 'text-sm' / >
2022-03-21 11:09:01 -07:00
< / F o r m G r o u p >
2022-03-21 12:23:57 -07:00
< / d i v >
< / d i v >
2022-03-21 11:09:01 -07:00
2022-03-21 12:23:57 -07:00
{ / * < C h e c k b o x
2020-06-06 13:52:33 -07:00
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' / > }
2020-04-21 17:22:00 -07:00
name = 'locked'
checked = { this . state . locked }
onChange = { this . handleCheckboxChange }
/ >
2021-06-16 12:20:57 -07:00
< Checkbox
label = { < FormattedMessage id = 'edit_profile.fields.hide_network_label' defaultMessage = 'Hide network' / > }
hint = { < FormattedMessage id = 'edit_profile.hints.hide_network' defaultMessage = 'Who you follow and who follows you will not be shown on your profile' / > }
name = 'hide_network'
checked = { this . state . hide _network }
onChange = { this . handleCheckboxChange }
/ >
2020-04-21 17:22:00 -07:00
< Checkbox
2020-06-06 13:52:33 -07:00
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' / > }
2020-04-21 17:22:00 -07:00
name = 'bot'
checked = { this . state . bot }
onChange = { this . handleCheckboxChange }
2020-04-21 16:00:05 -07:00
/ >
2021-06-06 20:43:18 -07:00
< Checkbox
label = { < FormattedMessage id = 'edit_profile.fields.stranger_notifications_label' defaultMessage = 'Block notifications from strangers' / > }
hint = { < FormattedMessage id = 'edit_profile.hints.stranger_notifications' defaultMessage = 'Only show notifications from people you follow' / > }
name = 'stranger_notifications'
checked = { this . state . stranger _notifications }
onChange = { this . handleCheckboxChange }
2021-12-26 13:44:43 -08:00
/ >
< Checkbox
label = { < FormattedMessage id = 'edit_profile.fields.discoverable_label' defaultMessage = 'Allow account discovery' / > }
hint = { < FormattedMessage id = 'edit_profile.hints.discoverable' defaultMessage = 'Display account in profile directory and allow indexing by external services' / > }
name = 'discoverable'
checked = { this . state . discoverable }
onChange = { this . handleCheckboxChange }
2022-03-21 11:09:01 -07:00
/>*/ }
2022-03-21 12:23:57 -07:00
{ supportsEmailList && < Checkbox
label = { < FormattedMessage id = 'edit_profile.fields.accepts_email_list_label' defaultMessage = 'Subscribe to newsletter' / > }
hint = { < FormattedMessage id = 'edit_profile.hints.accepts_email_list' defaultMessage = 'Opt-in to news and marketing updates.' / > }
name = 'accepts_email_list'
checked = { this . state . accepts _email _list }
onChange = { this . handleCheckboxChange }
/ > }
{ /* </FieldsGroup> */ }
{ / * < F i e l d s G r o u p >
2020-05-18 10:20:39 -07:00
< div className = 'fields-row__column fields-group' >
< div className = 'input with_block_label' >
2020-06-06 13:52:33 -07:00
< label > < FormattedMessage id = 'edit_profile.fields.meta_fields_label' defaultMessage = 'Profile metadata' / > < / l a b e l >
< span className = 'hint' >
2020-08-08 18:32:39 -07:00
< 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 : maxFields } } / >
2020-06-06 13:52:33 -07:00
< / s p a n >
2020-05-18 10:20:39 -07:00
{
this . state . fields . map ( ( field , i ) => (
< div className = 'row' key = { i } >
< TextInput
2020-06-06 13:52:33 -07:00
placeholder = { intl . formatMessage ( messages . metaFieldLabel ) }
2020-05-18 10:20:39 -07:00
value = { field . get ( 'name' ) }
onChange = { this . handleFieldChange ( i , 'name' ) }
/ >
< TextInput
2020-06-06 13:52:33 -07:00
placeholder = { intl . formatMessage ( messages . metaFieldContent ) }
2020-05-18 10:20:39 -07:00
value = { field . get ( 'value' ) }
onChange = { this . handleFieldChange ( i , 'value' ) }
/ >
2021-06-30 07:15:44 -07:00
{
2021-12-16 14:05:33 -08:00
this . state . fields . size > 4 && < Icon className = 'delete-field' src = { require ( '@tabler/icons/icons/circle-x.svg' ) } onClick = { this . handleDeleteField ( i ) } / >
2021-06-30 07:15:44 -07:00
}
2020-05-18 10:20:39 -07:00
< / d i v >
) )
}
2021-06-30 07:15:44 -07:00
{
this . state . fields . size < maxFields && (
< div className = 'actions add-row' >
< div name = 'button' type = 'button' role = 'presentation' className = 'btn button button-secondary' onClick = { this . handleAddField } >
2021-12-16 14:05:33 -08:00
< Icon src = { require ( '@tabler/icons/icons/circle-plus.svg' ) } / >
2021-06-30 07:15:44 -07:00
< FormattedMessage id = 'edit_profile.meta_fields.add' defaultMessage = 'Add new item' / >
< / d i v >
< / d i v >
)
}
2020-05-18 10:20:39 -07:00
< / d i v >
< / d i v >
2022-03-21 11:09:01 -07:00
< /FieldsGroup>*/ }
2022-03-21 12:23:57 -07:00
{ /* </fieldset> */ }
< FormActions >
< Button to = '/settings' theme = 'ghost' >
{ intl . formatMessage ( messages . cancel ) }
< / B u t t o n >
< Button theme = 'primary' type = 'submit' disabled = { this . state . isLoading } >
< FormattedMessage id = 'edit_profile.save' defaultMessage = 'Save' / >
< / B u t t o n >
< / F o r m A c t i o n s >
< / F o r m >
2020-04-21 16:00:05 -07:00
< / C o l u m n >
) ;
}
}