Merge branch 'email-list' into 'develop'

Email list integration

See merge request soapbox-pub/soapbox-fe!527
This commit is contained in:
Alex Gleason 2021-06-15 22:05:45 +00:00
commit 845ac6ab08
6 changed files with 93 additions and 3 deletions

View file

@ -0,0 +1,19 @@
import api from '../api';
export function getSubscribersCsv() {
return (dispatch, getState) => {
return api(getState).get('/api/v1/pleroma/admin/email_list/subscribers.csv');
};
}
export function getUnsubscribersCsv() {
return (dispatch, getState) => {
return api(getState).get('/api/v1/pleroma/admin/email_list/unsubscribers.csv');
};
}
export function getCombinedCsv() {
return (dispatch, getState) => {
return api(getState).get('/api/v1/pleroma/admin/email_list/combined.csv');
};
}

View file

@ -8,6 +8,19 @@ import Column from '../ui/components/column';
import RegistrationModePicker from './components/registration_mode_picker'; import RegistrationModePicker from './components/registration_mode_picker';
import { parseVersion } from 'soapbox/utils/features'; import { parseVersion } from 'soapbox/utils/features';
import sourceCode from 'soapbox/utils/code'; import sourceCode from 'soapbox/utils/code';
import { getSubscribersCsv, getUnsubscribersCsv, getCombinedCsv } from 'soapbox/actions/email_list';
import { getFeatures } from 'soapbox/utils/features';
// https://stackoverflow.com/a/53230807
const download = (response, filename) => {
const url = URL.createObjectURL(new Blob([response.data]));
const link = document.createElement('a');
link.href = url;
link.setAttribute('download', filename);
document.body.appendChild(link);
link.click();
link.remove();
};
const messages = defineMessages({ const messages = defineMessages({
heading: { id: 'column.admin.dashboard', defaultMessage: 'Dashboard' }, heading: { id: 'column.admin.dashboard', defaultMessage: 'Dashboard' },
@ -15,6 +28,7 @@ const messages = defineMessages({
const mapStateToProps = (state, props) => ({ const mapStateToProps = (state, props) => ({
instance: state.get('instance'), instance: state.get('instance'),
supportsEmailList: getFeatures(state.get('instance')).emailList,
}); });
export default @connect(mapStateToProps) export default @connect(mapStateToProps)
@ -24,10 +38,32 @@ class Dashboard extends ImmutablePureComponent {
static propTypes = { static propTypes = {
intl: PropTypes.object.isRequired, intl: PropTypes.object.isRequired,
instance: ImmutablePropTypes.map.isRequired, instance: ImmutablePropTypes.map.isRequired,
supportsEmailList: PropTypes.bool,
}; };
handleSubscribersClick = e => {
this.props.dispatch(getSubscribersCsv()).then((response) => {
download(response, 'subscribers.csv');
}).catch(() => {});
e.preventDefault();
}
handleUnsubscribersClick = e => {
this.props.dispatch(getUnsubscribersCsv()).then((response) => {
download(response, 'unsubscribers.csv');
}).catch(() => {});
e.preventDefault();
}
handleCombinedClick = e => {
this.props.dispatch(getCombinedCsv()).then((response) => {
download(response, 'combined.csv');
}).catch(() => {});
e.preventDefault();
}
render() { render() {
const { intl, instance } = this.props; const { intl, instance, supportsEmailList } = this.props;
const v = parseVersion(instance.get('version')); const v = parseVersion(instance.get('version'));
const userCount = instance.getIn(['stats', 'user_count']); const userCount = instance.getIn(['stats', 'user_count']);
const mau = instance.getIn(['pleroma', 'stats', 'mau']); const mau = instance.getIn(['pleroma', 'stats', 'mau']);
@ -96,6 +132,14 @@ class Dashboard extends ImmutablePureComponent {
<li>{v.software} <span className='pull-right'>{v.version}</span></li> <li>{v.software} <span className='pull-right'>{v.version}</span></li>
</ul> </ul>
</div> </div>
{supportsEmailList && <div className='dashwidget'>
<h4><FormattedMessage id='admin.dashwidgets.email_list_header' defaultMessage='Email list' /></h4>
<ul>
<li><a href='#' onClick={this.handleSubscribersClick} target='_blank'>subscribers.csv</a></li>
<li><a href='#' onClick={this.handleUnsubscribersClick} target='_blank'>unsubscribers.csv</a></li>
<li><a href='#' onClick={this.handleCombinedClick} target='_blank'>combined.csv</a></li>
</ul>
</div>}
</div> </div>
</Column> </Column>
); );

View file

@ -24,6 +24,7 @@ import { updateNotificationSettings } from 'soapbox/actions/accounts';
import { unescape } from 'lodash'; import { unescape } from 'lodash';
import { isVerified } from 'soapbox/utils/accounts'; import { isVerified } from 'soapbox/utils/accounts';
import { getSoapboxConfig } from 'soapbox/actions/soapbox'; import { getSoapboxConfig } from 'soapbox/actions/soapbox';
import { getFeatures } from 'soapbox/utils/features';
const messages = defineMessages({ const messages = defineMessages({
heading: { id: 'column.edit_profile', defaultMessage: 'Edit profile' }, heading: { id: 'column.edit_profile', defaultMessage: 'Edit profile' },
@ -41,6 +42,7 @@ const mapStateToProps = state => {
account, account,
maxFields: state.getIn(['instance', 'pleroma', 'metadata', 'fields_limits', 'max_fields'], 4), maxFields: state.getIn(['instance', 'pleroma', 'metadata', 'fields_limits', 'max_fields'], 4),
verifiedCanEditName: soapbox.get('verifiedCanEditName'), verifiedCanEditName: soapbox.get('verifiedCanEditName'),
supportsEmailList: getFeatures(state.get('instance')).emailList,
}; };
}; };
@ -78,11 +80,13 @@ class EditProfile extends ImmutablePureComponent {
super(props); super(props);
const { account } = this.props; const { account } = this.props;
const strangerNotifications = account.getIn(['pleroma', 'notification_settings', 'block_from_strangers']); const strangerNotifications = account.getIn(['pleroma', 'notification_settings', 'block_from_strangers']);
const acceptsEmailList = account.getIn(['pleroma', 'accepts_email_list']);
const initialState = account.withMutations(map => { const initialState = account.withMutations(map => {
map.merge(map.get('source')); map.merge(map.get('source'));
map.delete('source'); map.delete('source');
map.set('fields', normalizeFields(map.get('fields'), props.maxFields)); map.set('fields', normalizeFields(map.get('fields'), props.maxFields));
map.set('stranger_notifications', strangerNotifications); map.set('stranger_notifications', strangerNotifications);
map.set('accepts_email_list', acceptsEmailList);
unescapeParams(map, ['display_name', 'bio']); unescapeParams(map, ['display_name', 'bio']);
}); });
this.state = initialState.toObject(); this.state = initialState.toObject();
@ -117,6 +121,7 @@ class EditProfile extends ImmutablePureComponent {
avatar: state.avatar_file, avatar: state.avatar_file,
header: state.header_file, header: state.header_file,
locked: state.locked, locked: state.locked,
accepts_email_list: state.accepts_email_list,
}, this.getFieldParams().toJS()); }, this.getFieldParams().toJS());
} }
@ -179,7 +184,7 @@ class EditProfile extends ImmutablePureComponent {
} }
render() { render() {
const { intl, maxFields, account, verifiedCanEditName } = this.props; const { intl, maxFields, account, verifiedCanEditName, supportsEmailList } = this.props;
const verified = isVerified(account); const verified = isVerified(account);
const canEditName = verifiedCanEditName || !verified; const canEditName = verifiedCanEditName || !verified;
@ -246,6 +251,13 @@ class EditProfile extends ImmutablePureComponent {
checked={this.state.stranger_notifications} checked={this.state.stranger_notifications}
onChange={this.handleCheckboxChange} onChange={this.handleCheckboxChange}
/> />
{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> </FieldsGroup>
<FieldsGroup> <FieldsGroup>
<div className='fields-row__column fields-group'> <div className='fields-row__column fields-group'>

View file

@ -18,6 +18,7 @@ import { Map as ImmutableMap } from 'immutable';
import { v4 as uuidv4 } from 'uuid'; import { v4 as uuidv4 } from 'uuid';
import { getSettings } from 'soapbox/actions/settings'; import { getSettings } from 'soapbox/actions/settings';
import { openModal } from 'soapbox/actions/modal'; import { openModal } from 'soapbox/actions/modal';
import { getFeatures } from 'soapbox/utils/features';
const messages = defineMessages({ const messages = defineMessages({
username: { id: 'registration.fields.username_placeholder', defaultMessage: 'Username' }, username: { id: 'registration.fields.username_placeholder', defaultMessage: 'Username' },
@ -28,6 +29,7 @@ const messages = defineMessages({
agreement: { id: 'registration.agreement', defaultMessage: 'I agree to the {tos}.' }, agreement: { id: 'registration.agreement', defaultMessage: 'I agree to the {tos}.' },
tos: { id: 'registration.tos', defaultMessage: 'Terms of Service' }, tos: { id: 'registration.tos', defaultMessage: 'Terms of Service' },
close: { id: 'registration.confirmation_modal.close', defaultMessage: 'Close' }, close: { id: 'registration.confirmation_modal.close', defaultMessage: 'Close' },
newsletter: { id: 'registration.newsletter', defaultMessage: 'Subscribe to newsletter.' },
}); });
const mapStateToProps = (state, props) => ({ const mapStateToProps = (state, props) => ({
@ -35,6 +37,7 @@ const mapStateToProps = (state, props) => ({
locale: getSettings(state).get('locale'), locale: getSettings(state).get('locale'),
needsConfirmation: state.getIn(['instance', 'pleroma', 'metadata', 'account_activation_required']), needsConfirmation: state.getIn(['instance', 'pleroma', 'metadata', 'account_activation_required']),
needsApproval: state.getIn(['instance', 'approval_required']), needsApproval: state.getIn(['instance', 'approval_required']),
supportsEmailList: getFeatures(state.get('instance')).emailList,
}); });
export default @connect(mapStateToProps) export default @connect(mapStateToProps)
@ -135,7 +138,7 @@ class RegistrationForm extends ImmutablePureComponent {
} }
render() { render() {
const { instance, intl } = this.props; const { instance, intl, supportsEmailList } = this.props;
const { params } = this.state; const { params } = this.state;
const isOpen = instance.get('registrations'); const isOpen = instance.get('registrations');
const isLoading = this.state.captchaLoading || this.state.submissionLoading; const isLoading = this.state.captchaLoading || this.state.submissionLoading;
@ -232,6 +235,11 @@ class RegistrationForm extends ImmutablePureComponent {
onChange={this.onCheckboxChange} onChange={this.onCheckboxChange}
required required
/> />
{supportsEmailList && <Checkbox
label={intl.formatMessage(messages.newsletter)}
name='accepts_email_list'
onChange={this.onCheckboxChange}
/>}
</div> </div>
<div className='actions'> <div className='actions'>
<button name='button' type='submit' className='btn button button-primary'> <button name='button' type='submit' className='btn button button-primary'>

View file

@ -1,8 +1,10 @@
// Detect backend features to conditionally render elements // Detect backend features to conditionally render elements
import gte from 'semver/functions/gte'; import gte from 'semver/functions/gte';
import { List as ImmutableList } from 'immutable';
export const getFeatures = instance => { export const getFeatures = instance => {
const v = parseVersion(instance.get('version')); const v = parseVersion(instance.get('version'));
const f = instance.getIn(['pleroma', 'metadata', 'features'], ImmutableList());
return { return {
suggestions: v.software === 'Mastodon' && gte(v.compatVersion, '2.4.3'), suggestions: v.software === 'Mastodon' && gte(v.compatVersion, '2.4.3'),
trends: v.software === 'Mastodon' && gte(v.compatVersion, '3.0.0'), trends: v.software === 'Mastodon' && gte(v.compatVersion, '3.0.0'),
@ -11,6 +13,7 @@ export const getFeatures = instance => {
attachmentLimit: v.software === 'Pleroma' ? Infinity : 4, attachmentLimit: v.software === 'Pleroma' ? Infinity : 4,
focalPoint: v.software === 'Mastodon' && gte(v.compatVersion, '2.3.0'), focalPoint: v.software === 'Mastodon' && gte(v.compatVersion, '2.3.0'),
importMutes: v.software === 'Pleroma' && gte(v.version, '2.2.0'), importMutes: v.software === 'Pleroma' && gte(v.version, '2.2.0'),
emailList: f.includes('email_list'),
}; };
}; };

View file

@ -68,6 +68,10 @@
margin-bottom: 8px; margin-bottom: 8px;
border-bottom: 1px solid var(--accent-color--med); border-bottom: 1px solid var(--accent-color--med);
} }
a {
color: var(--brand-color);
}
} }
.unapproved-account { .unapproved-account {