import { Map as ImmutableMap, List as ImmutableList, } from 'immutable'; import { unescape } from 'lodash'; 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'; // import { updateNotificationSettings } from 'soapbox/actions/accounts'; import { patchMe } from 'soapbox/actions/me'; import snackbar from 'soapbox/actions/snackbar'; import { getSoapboxConfig } from 'soapbox/actions/soapbox'; // import Icon from 'soapbox/components/icon'; import { Checkbox, } from 'soapbox/features/forms'; import { makeGetAccount } from 'soapbox/selectors'; import { getFeatures } from 'soapbox/utils/features'; import resizeImage from 'soapbox/utils/resize_image'; import { Button, Column, Form, FormActions, FormGroup, Input, Textarea } from '../../components/ui'; import ProfilePreview from './components/profile_preview'; 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; }; const messages = defineMessages({ 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' }, verified: { id: 'edit_profile.fields.verified_display_name', defaultMessage: 'Verified users may not update their display name' }, success: { id: 'edit_profile.success', defaultMessage: 'Profile saved!' }, error: { id: 'edit_profile.error', defaultMessage: 'Profile update failed' }, bioPlaceholder: { id: 'edit_profile.fields.bio_placeholder', defaultMessage: 'Tell us about yourself.' }, displayNamePlaceholder: { id: 'edit_profile.fields.display_name_placeholder', defaultMessage: 'Name' }, 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' }, }); const makeMapStateToProps = () => { const getAccount = makeGetAccount(); const mapStateToProps = state => { const me = state.get('me'); const account = getAccount(state, me); const soapbox = getSoapboxConfig(state); return { account, maxFields: state.getIn(['instance', 'pleroma', 'metadata', 'fields_limits', 'max_fields'], 4), verifiedCanEditName: soapbox.get('verifiedCanEditName'), supportsEmailList: getFeatures(state.get('instance')).emailList, }; }; return mapStateToProps; }; // Forces fields to be maxFields size, filling empty values const normalizeFields = (fields, maxFields) => ( ImmutableList(fields).setSize(Math.max(fields.size, maxFields)).map(field => field ? field : ImmutableMap({ name: '', value: '' }), ) ); // HTML unescape for special chars, eg const unescapeParams = (map, params) => ( params.reduce((map, param) => ( map.set(param, unescape(map.get(param))) ), map) ); export default @connect(makeMapStateToProps) @injectIntl class EditProfile extends ImmutablePureComponent { static propTypes = { dispatch: PropTypes.func.isRequired, intl: PropTypes.object.isRequired, account: ImmutablePropTypes.map, maxFields: PropTypes.number, verifiedCanEditName: PropTypes.bool, }; state = { isLoading: false, } constructor(props) { super(props); const { account, maxFields } = this.props; const strangerNotifications = account.getIn(['pleroma', 'notification_settings', 'block_from_strangers']); const acceptsEmailList = account.getIn(['pleroma', 'accepts_email_list']); const discoverable = account.getIn(['source', 'pleroma', 'discoverable']); const initialState = ImmutableMap(account).withMutations(map => { map.merge(map.get('source')); map.delete('source'); map.set('fields', normalizeFields(map.get('fields'), Math.min(maxFields, 4))); map.set('stranger_notifications', strangerNotifications); map.set('accepts_email_list', acceptsEmailList); map.set('hide_network', hidesNetwork(account)); map.set('discoverable', discoverable); unescapeParams(map, ['display_name', 'bio']); }); 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 || account.get('username'), website: this.state.website || account.get('website'), location: this.state.location || account.get('location'), })); } 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, website: state.website, location: state.location, note: state.note, avatar: state.avatar_file, header: state.header_file, locked: state.locked, accepts_email_list: state.accepts_email_list, hide_followers: state.hide_network, hide_follows: state.hide_network, hide_followers_count: state.hide_network, hide_follows_count: state.hide_network, }, this.getFieldParams().toJS()); } getFormdata = () => { const data = this.getParams(); const formData = new FormData(); for (const key in data) { const hasValue = data[key] !== null && data[key] !== undefined; // Compact the submission. This should probably be done better. const shouldAppend = Boolean(hasValue || key.startsWith('fields_attributes')); if (shouldAppend) formData.append(key, hasValue ? data[key] : ''); } return formData; } handleSubmit = (event) => { const { dispatch, intl } = this.props; const credentials = dispatch(patchMe(this.getFormdata())); /* Bad API url, was causing errors in the promise call below blocking the success message after making edits. */ /* const notifications = dispatch(updateNotificationSettings({ block_from_strangers: this.state.stranger_notifications || false, })); */ this.setState({ isLoading: true }); Promise.all([credentials /*notifications*/]).then(() => { this.setState({ isLoading: false }); dispatch(snackbar.success(intl.formatMessage(messages.success))); }).catch((error) => { this.setState({ isLoading: false }); dispatch(snackbar.error(intl.formatMessage(messages.error))); }); 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 = maxPixels => { return e => { const { name } = e.target; const [f] = e.target.files || []; resizeImage(f, maxPixels).then(file => { const url = file ? URL.createObjectURL(file) : this.state[name]; this.setState({ [name]: url, [`${name}_file`]: file, }); }).catch(console.error); }; } 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)), }); }; } render() { const { intl, account, verifiedCanEditName, supportsEmailList /* maxFields */ } = this.props; const verified = account.get('verified'); const canEditName = verifiedCanEditName || !verified; return ( } hintText={!canEditName && intl.formatMessage(messages.verified)} > } > } > } > } hintText={} > } hintText={} > {/*} hint={} name='locked' checked={this.state.locked} onChange={this.handleCheckboxChange} /> } hint={} name='hide_network' checked={this.state.hide_network} onChange={this.handleCheckboxChange} /> } hint={} name='bot' checked={this.state.bot} onChange={this.handleCheckboxChange} /> } hint={} name='stranger_notifications' checked={this.state.stranger_notifications} onChange={this.handleCheckboxChange} /> } hint={} name='discoverable' checked={this.state.discoverable} onChange={this.handleCheckboxChange} />*/} {supportsEmailList && } hint={} name='accepts_email_list' checked={this.state.accepts_email_list} onChange={this.handleCheckboxChange} />} {/* */} {/* { this.state.fields.map((field, i) => ( { this.state.fields.size > 4 && } )) } { this.state.fields.size < maxFields && ( ) } */} {/* */} {intl.formatMessage(messages.cancel)} ); } }