Merge branch 'snackbar-action-link' into 'develop'

Add 'View' action link to some notifications

Closes #791

See merge request soapbox-pub/soapbox-fe!1008
This commit is contained in:
marcin mikołajczak 2022-02-01 08:42:11 +00:00
commit 5fda820a95
11 changed files with 68 additions and 29 deletions

View file

@ -81,6 +81,7 @@ const messages = defineMessages({
uploadErrorPoll: { id: 'upload_error.poll', defaultMessage: 'File upload not allowed with polls.' }, uploadErrorPoll: { id: 'upload_error.poll', defaultMessage: 'File upload not allowed with polls.' },
scheduleError: { id: 'compose.invalid_schedule', defaultMessage: 'You must schedule a post at least 5 minutes out.' }, scheduleError: { id: 'compose.invalid_schedule', defaultMessage: 'You must schedule a post at least 5 minutes out.' },
success: { id: 'compose.submit_success', defaultMessage: 'Your post was sent' }, success: { id: 'compose.submit_success', defaultMessage: 'Your post was sent' },
view: { id: 'snackbar.view', defaultMessage: 'View' },
}); });
const COMPOSE_PANEL_BREAKPOINT = 600 + (285 * 1) + (10 * 1); const COMPOSE_PANEL_BREAKPOINT = 600 + (285 * 1) + (10 * 1);
@ -190,7 +191,7 @@ export function handleComposeSubmit(dispatch, getState, data, status) {
dispatch(insertIntoTagHistory(data.tags || [], status)); dispatch(insertIntoTagHistory(data.tags || [], status));
dispatch(submitComposeSuccess({ ...data })); dispatch(submitComposeSuccess({ ...data }));
dispatch(snackbar.success(messages.success)); dispatch(snackbar.success(messages.success, messages.view, `/@${data.account.acct}/posts/${data.id}`));
} }
const needsDescriptions = state => { const needsDescriptions = state => {

View file

@ -58,6 +58,7 @@ export const REMOTE_INTERACTION_FAIL = 'REMOTE_INTERACTION_FAIL';
const messages = defineMessages({ const messages = defineMessages({
bookmarkAdded: { id: 'status.bookmarked', defaultMessage: 'Bookmark added.' }, bookmarkAdded: { id: 'status.bookmarked', defaultMessage: 'Bookmark added.' },
bookmarkRemoved: { id: 'status.unbookmarked', defaultMessage: 'Bookmark removed.' }, bookmarkRemoved: { id: 'status.unbookmarked', defaultMessage: 'Bookmark removed.' },
view: { id: 'snackbar.view', defaultMessage: 'View' },
}); });
export function reblog(status) { export function reblog(status) {
@ -220,28 +221,28 @@ export function unfavouriteFail(status, error) {
}; };
} }
export function bookmark(intl, status) { export function bookmark(status) {
return function(dispatch, getState) { return function(dispatch, getState) {
dispatch(bookmarkRequest(status)); dispatch(bookmarkRequest(status));
api(getState).post(`/api/v1/statuses/${status.get('id')}/bookmark`).then(function(response) { api(getState).post(`/api/v1/statuses/${status.get('id')}/bookmark`).then(function(response) {
dispatch(importFetchedStatus(response.data)); dispatch(importFetchedStatus(response.data));
dispatch(bookmarkSuccess(status, response.data)); dispatch(bookmarkSuccess(status, response.data));
dispatch(snackbar.success(intl.formatMessage(messages.bookmarkAdded))); dispatch(snackbar.success(messages.bookmarkAdded, messages.view, '/bookmarks'));
}).catch(function(error) { }).catch(function(error) {
dispatch(bookmarkFail(status, error)); dispatch(bookmarkFail(status, error));
}); });
}; };
} }
export function unbookmark(intl, status) { export function unbookmark(status) {
return (dispatch, getState) => { return (dispatch, getState) => {
dispatch(unbookmarkRequest(status)); dispatch(unbookmarkRequest(status));
api(getState).post(`/api/v1/statuses/${status.get('id')}/unbookmark`).then(response => { api(getState).post(`/api/v1/statuses/${status.get('id')}/unbookmark`).then(response => {
dispatch(importFetchedStatus(response.data)); dispatch(importFetchedStatus(response.data));
dispatch(unbookmarkSuccess(status, response.data)); dispatch(unbookmarkSuccess(status, response.data));
dispatch(snackbar.success(intl.formatMessage(messages.bookmarkRemoved))); dispatch(snackbar.success(messages.bookmarkRemoved));
}).catch(error => { }).catch(error => {
dispatch(unbookmarkFail(status, error)); dispatch(unbookmarkFail(status, error));
}); });

View file

@ -1,21 +1,23 @@
import { ALERT_SHOW } from './alerts'; import { ALERT_SHOW } from './alerts';
export const show = (severity, message) => ({ export const show = (severity, message, actionLabel, actionLink) => ({
type: ALERT_SHOW, type: ALERT_SHOW,
message, message,
actionLabel,
actionLink,
severity, severity,
}); });
export function info(message) { export function info(message, actionLabel, actionLink) {
return show('info', message); return show('info', message, actionLabel, actionLink);
} }
export function success(message) { export function success(message, actionLabel, actionLink) {
return show('success', message); return show('success', message, actionLabel, actionLink);
} }
export function error(message) { export function error(message, actionLabel, actionLink) {
return show('error', message); return show('error', message, actionLabel, actionLink);
} }
export default { export default {

View file

@ -134,9 +134,9 @@ const mapDispatchToProps = (dispatch, { intl }) => {
onBookmark(status) { onBookmark(status) {
if (status.get('bookmarked')) { if (status.get('bookmarked')) {
dispatch(unbookmark(intl, status)); dispatch(unbookmark(status));
} else { } else {
dispatch(bookmark(intl, status)); dispatch(bookmark(status));
} }
}, },

View file

@ -50,6 +50,7 @@ const messages = defineMessages({
error: { id: 'edit_profile.error', defaultMessage: 'Profile update failed' }, error: { id: 'edit_profile.error', defaultMessage: 'Profile update failed' },
bioPlaceholder: { id: 'edit_profile.fields.bio_placeholder', defaultMessage: 'Tell us about yourself.' }, bioPlaceholder: { id: 'edit_profile.fields.bio_placeholder', defaultMessage: 'Tell us about yourself.' },
displayNamePlaceholder: { id: 'edit_profile.fields.display_name_placeholder', defaultMessage: 'Name' }, displayNamePlaceholder: { id: 'edit_profile.fields.display_name_placeholder', defaultMessage: 'Name' },
view: { id: 'snackbar.view', defaultMessage: 'View' },
birthdayPlaceholder: { id: 'edit_profile.fields.birthday_placeholder', defaultMessage: 'Your birthday' }, birthdayPlaceholder: { id: 'edit_profile.fields.birthday_placeholder', defaultMessage: 'Your birthday' },
}); });
@ -188,7 +189,7 @@ class EditProfile extends ImmutablePureComponent {
} }
handleSubmit = (event) => { handleSubmit = (event) => {
const { dispatch, intl } = this.props; const { dispatch, intl, account } = this.props;
const credentials = dispatch(patchMe(this.getFormdata())); const credentials = dispatch(patchMe(this.getFormdata()));
const notifications = dispatch(updateNotificationSettings({ const notifications = dispatch(updateNotificationSettings({
@ -199,7 +200,7 @@ class EditProfile extends ImmutablePureComponent {
Promise.all([credentials, notifications]).then(() => { Promise.all([credentials, notifications]).then(() => {
this.setState({ isLoading: false }); this.setState({ isLoading: false });
dispatch(snackbar.success(intl.formatMessage(messages.success))); dispatch(snackbar.success(intl.formatMessage(messages.success), intl.formatMessage(messages.view), `/@${account.get('acct')}`));
}).catch((error) => { }).catch((error) => {
this.setState({ isLoading: false }); this.setState({ isLoading: false });
dispatch(snackbar.error(intl.formatMessage(messages.error))); dispatch(snackbar.error(intl.formatMessage(messages.error)));

View file

@ -98,9 +98,9 @@ const mapDispatchToProps = (dispatch, { intl }) => ({
onBookmark(status) { onBookmark(status) {
if (status.get('bookmarked')) { if (status.get('bookmarked')) {
dispatch(unbookmark(intl, status)); dispatch(unbookmark(status));
} else { } else {
dispatch(bookmark(intl, status)); dispatch(bookmark(status));
} }
}, },

View file

@ -221,9 +221,9 @@ class Status extends ImmutablePureComponent {
handleBookmark = (status) => { handleBookmark = (status) => {
if (status.get('bookmarked')) { if (status.get('bookmarked')) {
this.props.dispatch(unbookmark(this.props.intl, status)); this.props.dispatch(unbookmark(status));
} else { } else {
this.props.dispatch(bookmark(this.props.intl, status)); this.props.dispatch(bookmark(status));
} }
} }

View file

@ -1,6 +1,8 @@
import React from 'react';
import { injectIntl } from 'react-intl'; import { injectIntl } from 'react-intl';
import { NotificationStack } from 'react-notification'; import { NotificationStack } from 'react-notification';
import { connect } from 'react-redux'; import { connect } from 'react-redux';
import { Link } from 'react-router-dom';
import { dismissAlert } from '../../../actions/alerts'; import { dismissAlert } from '../../../actions/alerts';
import { getAlerts } from '../../../selectors'; import { getAlerts } from '../../../selectors';
@ -16,22 +18,35 @@ const defaultBarStyleFactory = (index, style, notification) => {
const mapStateToProps = (state, { intl }) => { const mapStateToProps = (state, { intl }) => {
const notifications = getAlerts(state); const notifications = getAlerts(state);
notifications.forEach(notification => ['title', 'message'].forEach(key => { notifications.forEach(notification => {
['title', 'message', 'actionLabel'].forEach(key => {
const value = notification[key]; const value = notification[key];
if (typeof value === 'object') { if (typeof value === 'object') {
notification[key] = intl.formatMessage(value); notification[key] = intl.formatMessage(value);
} }
})); });
return { notifications }; if (notification.actionLabel) {
notification.action = (
<Link to={notification.actionLink}>
{notification.actionLabel}
</Link>
);
}
});
return { notifications, linkComponent: Link };
}; };
const mapDispatchToProps = (dispatch) => { const mapDispatchToProps = (dispatch) => {
return { const onDismiss = alert => {
onDismiss: alert => {
dispatch(dismissAlert(alert)); dispatch(dismissAlert(alert));
}, };
return {
onDismiss,
onClick: onDismiss,
barStyleFactory: defaultBarStyleFactory, barStyleFactory: defaultBarStyleFactory,
activeBarStyleFactory: defaultBarStyleFactory, activeBarStyleFactory: defaultBarStyleFactory,
}; };

View file

@ -16,6 +16,8 @@ export default function alerts(state = initialState, action) {
title: action.title, title: action.title,
message: action.message, message: action.message,
severity: action.severity || 'info', severity: action.severity || 'info',
actionLabel: action.actionLabel,
actionLink: action.actionLink,
})); }));
case ALERT_DISMISS: case ALERT_DISMISS:
return state.filterNot(item => item.get('key') === action.alert.key); return state.filterNot(item => item.get('key') === action.alert.key);

View file

@ -171,6 +171,8 @@ export const getAlerts = createSelector([getAlertsBase], (base) => {
arr.push({ arr.push({
message: item.get('message'), message: item.get('message'),
title: item.get('title'), title: item.get('title'),
actionLabel: item.get('actionLabel'),
actionLink: item.get('actionLink'),
key: item.get('key'), key: item.get('key'),
className: `snackbar snackbar--${item.get('severity', 'info')}`, className: `snackbar snackbar--${item.get('severity', 'info')}`,
activeClassName: 'snackbar--active', activeClassName: 'snackbar--active',

View file

@ -55,4 +55,19 @@
.notification-bar-wrapper { .notification-bar-wrapper {
transform: translateY(1px); transform: translateY(1px);
} }
.notification-bar-action a {
@include font-roboto;
font-size: 16px;
color: white;
font-weight: 700;
text-decoration: none;
text-transform: none;
&:hover,
&:active,
&:focus {
text-decoration: underline;
}
}
} }