Merge branch 'email-list' into 'develop'
Email list integration See merge request soapbox-pub/soapbox-fe!527
This commit is contained in:
commit
845ac6ab08
6 changed files with 93 additions and 3 deletions
19
app/soapbox/actions/email_list.js
Normal file
19
app/soapbox/actions/email_list.js
Normal 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');
|
||||||
|
};
|
||||||
|
}
|
|
@ -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>
|
||||||
);
|
);
|
||||||
|
|
|
@ -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'>
|
||||||
|
|
|
@ -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'>
|
||||||
|
|
|
@ -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'),
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -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 {
|
||||||
|
|
Loading…
Reference in a new issue