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:
commit
5fda820a95
11 changed files with 68 additions and 29 deletions
|
@ -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 => {
|
||||||
|
|
|
@ -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));
|
||||||
});
|
});
|
||||||
|
|
|
@ -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 {
|
||||||
|
|
|
@ -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));
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|
|
@ -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)));
|
||||||
|
|
|
@ -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));
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|
|
@ -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));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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,
|
||||||
};
|
};
|
||||||
|
|
|
@ -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);
|
||||||
|
|
|
@ -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',
|
||||||
|
|
|
@ -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;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue