Add notification quick-filter bar in the frontend app (#9399)
* create FilterBar componer and its container, unstyled * introduce basic styling for FilterBar * add selection css * allow FilterBar to display active CSS with js * connect the FilterBar to the Redux state * change getNotifications to use filter * remove temporary comments * add an option to turn the FilterBar off in settings * fix showFilterBar data type to boolean * fix eslint errors * add English and Polish translations * allowed filter bar overflow to accomodate for longer languages * fix mispelled translation key * add unified CSS look * replace text in FilterBar with icons * add tooltips * replace text @ with an icon * introduce simple and advanced filtering view * add ability to toggle the advanced view * add Polish translations * change Advanced View description to be more clear * make each filter flush notifications and load new ones, fixing pagination * simplify getNotifications once frontend filtering is not needed for FilterBar * add a semicolon * Revert "simplify getNotifications once frontend filtering is not needed for FilterBar" This reverts commit 9f4be7857135b0327814bd22a3e8a4e7b546f7cc. * reset filter to 'all' when turning off FilterBar
This commit is contained in:
		
							parent
							
								
									5f0d3e8bad
								
							
						
					
					
						commit
						13dce12665
					
				
					 11 changed files with 244 additions and 7 deletions
				
			
		| 
						 | 
				
			
			@ -8,6 +8,7 @@ import {
 | 
			
		|||
  importFetchedStatuses,
 | 
			
		||||
} from './importer';
 | 
			
		||||
import { defineMessages } from 'react-intl';
 | 
			
		||||
import { List as ImmutableList } from 'immutable';
 | 
			
		||||
import { unescapeHTML } from '../utils/html';
 | 
			
		||||
import { getFilters, regexFromFilters } from '../selectors';
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -18,6 +19,8 @@ export const NOTIFICATIONS_EXPAND_REQUEST = 'NOTIFICATIONS_EXPAND_REQUEST';
 | 
			
		|||
export const NOTIFICATIONS_EXPAND_SUCCESS = 'NOTIFICATIONS_EXPAND_SUCCESS';
 | 
			
		||||
export const NOTIFICATIONS_EXPAND_FAIL    = 'NOTIFICATIONS_EXPAND_FAIL';
 | 
			
		||||
 | 
			
		||||
export const NOTIFICATIONS_FILTER_SET = 'NOTIFICATIONS_FILTER_SET';
 | 
			
		||||
 | 
			
		||||
export const NOTIFICATIONS_CLEAR      = 'NOTIFICATIONS_CLEAR';
 | 
			
		||||
export const NOTIFICATIONS_SCROLL_TOP = 'NOTIFICATIONS_SCROLL_TOP';
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -88,10 +91,16 @@ export function updateNotifications(notification, intlMessages, intlLocale) {
 | 
			
		|||
 | 
			
		||||
const excludeTypesFromSettings = state => state.getIn(['settings', 'notifications', 'shows']).filter(enabled => !enabled).keySeq().toJS();
 | 
			
		||||
 | 
			
		||||
const excludeTypesFromFilter = filter => {
 | 
			
		||||
  const allTypes = ImmutableList(['follow', 'favourite', 'reblog', 'mention']);
 | 
			
		||||
  return allTypes.filterNot(item => item === filter).toJS();
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
const noOp = () => {};
 | 
			
		||||
 | 
			
		||||
export function expandNotifications({ maxId } = {}, done = noOp) {
 | 
			
		||||
  return (dispatch, getState) => {
 | 
			
		||||
    const activeFilter = getState().getIn(['settings', 'notifications', 'quickFilter', 'active']);
 | 
			
		||||
    const notifications = getState().get('notifications');
 | 
			
		||||
    const isLoadingMore = !!maxId;
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -102,7 +111,9 @@ export function expandNotifications({ maxId } = {}, done = noOp) {
 | 
			
		|||
 | 
			
		||||
    const params = {
 | 
			
		||||
      max_id: maxId,
 | 
			
		||||
      exclude_types: excludeTypesFromSettings(getState()),
 | 
			
		||||
      exclude_types: activeFilter === 'all'
 | 
			
		||||
        ? excludeTypesFromSettings(getState())
 | 
			
		||||
        : excludeTypesFromFilter(activeFilter),
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    if (!maxId && notifications.get('items').size > 0) {
 | 
			
		||||
| 
						 | 
				
			
			@ -167,3 +178,14 @@ export function scrollTopNotifications(top) {
 | 
			
		|||
    top,
 | 
			
		||||
  };
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
export function setFilter (filterType) {
 | 
			
		||||
  return dispatch => {
 | 
			
		||||
    dispatch({
 | 
			
		||||
      type: NOTIFICATIONS_FILTER_SET,
 | 
			
		||||
      path: ['notifications', 'quickFilter', 'active'],
 | 
			
		||||
      value: filterType,
 | 
			
		||||
    });
 | 
			
		||||
    dispatch(expandNotifications());
 | 
			
		||||
  };
 | 
			
		||||
};
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -21,9 +21,11 @@ export default class ColumnSettings extends React.PureComponent {
 | 
			
		|||
  render () {
 | 
			
		||||
    const { settings, pushSettings, onChange, onClear } = this.props;
 | 
			
		||||
 | 
			
		||||
    const alertStr = <FormattedMessage id='notifications.column_settings.alert' defaultMessage='Desktop notifications' />;
 | 
			
		||||
    const showStr  = <FormattedMessage id='notifications.column_settings.show' defaultMessage='Show in column' />;
 | 
			
		||||
    const soundStr = <FormattedMessage id='notifications.column_settings.sound' defaultMessage='Play sound' />;
 | 
			
		||||
    const filterShowStr = <FormattedMessage id='notifications.column_settings.filter_bar.show' defaultMessage='Show' />;
 | 
			
		||||
    const filterAdvancedStr = <FormattedMessage id='notifications.column_settings.filter_bar.advanced' defaultMessage='Display all categories' />;
 | 
			
		||||
    const alertStr  = <FormattedMessage id='notifications.column_settings.alert' defaultMessage='Desktop notifications' />;
 | 
			
		||||
    const showStr   = <FormattedMessage id='notifications.column_settings.show' defaultMessage='Show in column' />;
 | 
			
		||||
    const soundStr  = <FormattedMessage id='notifications.column_settings.sound' defaultMessage='Play sound' />;
 | 
			
		||||
 | 
			
		||||
    const showPushSettings = pushSettings.get('browserSupport') && pushSettings.get('isSubscribed');
 | 
			
		||||
    const pushStr = showPushSettings && <FormattedMessage id='notifications.column_settings.push' defaultMessage='Push notifications' />;
 | 
			
		||||
| 
						 | 
				
			
			@ -34,6 +36,16 @@ export default class ColumnSettings extends React.PureComponent {
 | 
			
		|||
          <ClearColumnButton onClick={onClear} />
 | 
			
		||||
        </div>
 | 
			
		||||
 | 
			
		||||
        <div role='group' aria-labelledby='notifications-filter-bar'>
 | 
			
		||||
          <span id='notifications-filter-bar' className='column-settings__section'>
 | 
			
		||||
            <FormattedMessage id='notifications.column_settings.filter_bar.category' defaultMessage='Quick filter bar' />
 | 
			
		||||
          </span>
 | 
			
		||||
          <div className='column-settings__row'>
 | 
			
		||||
            <SettingToggle id='show-filter-bar' prefix='notifications' settings={settings} settingPath={['quickFilter', 'show']} onChange={onChange} label={filterShowStr} />
 | 
			
		||||
            <SettingToggle id='show-filter-bar' prefix='notifications' settings={settings} settingPath={['quickFilter', 'advanced']} onChange={onChange} label={filterAdvancedStr} />
 | 
			
		||||
          </div>
 | 
			
		||||
        </div>
 | 
			
		||||
 | 
			
		||||
        <div role='group' aria-labelledby='notifications-follow'>
 | 
			
		||||
          <span id='notifications-follow' className='column-settings__section'><FormattedMessage id='notifications.column_settings.follow' defaultMessage='New followers:' /></span>
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -0,0 +1,93 @@
 | 
			
		|||
import React from 'react';
 | 
			
		||||
import PropTypes from 'prop-types';
 | 
			
		||||
import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
 | 
			
		||||
 | 
			
		||||
const tooltips = defineMessages({
 | 
			
		||||
  mentions: { id: 'notifications.filter.mentions', defaultMessage: 'Mentions' },
 | 
			
		||||
  favourites: { id: 'notifications.filter.favourites', defaultMessage: 'Favourites' },
 | 
			
		||||
  boosts: { id: 'notifications.filter.boosts', defaultMessage: 'Boosts' },
 | 
			
		||||
  follows: { id: 'notifications.filter.follows', defaultMessage: 'Follows' },
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
export default @injectIntl
 | 
			
		||||
class FilterBar extends React.PureComponent {
 | 
			
		||||
 | 
			
		||||
  static propTypes = {
 | 
			
		||||
    selectFilter: PropTypes.func.isRequired,
 | 
			
		||||
    selectedFilter: PropTypes.string.isRequired,
 | 
			
		||||
    advancedMode: PropTypes.bool.isRequired,
 | 
			
		||||
    intl: PropTypes.object.isRequired,
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  onClick (notificationType) {
 | 
			
		||||
    return () => this.props.selectFilter(notificationType);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  render () {
 | 
			
		||||
    const { selectedFilter, advancedMode, intl } = this.props;
 | 
			
		||||
    const renderedElement = !advancedMode ? (
 | 
			
		||||
      <div className='notification__filter-bar'>
 | 
			
		||||
        <button
 | 
			
		||||
          className={selectedFilter === 'all' ? 'active' : ''}
 | 
			
		||||
          onClick={this.onClick('all')}
 | 
			
		||||
        >
 | 
			
		||||
          <FormattedMessage
 | 
			
		||||
            id='notifications.filter.all'
 | 
			
		||||
            defaultMessage='All'
 | 
			
		||||
          />
 | 
			
		||||
        </button>
 | 
			
		||||
        <button
 | 
			
		||||
          className={selectedFilter === 'mention' ? 'active' : ''}
 | 
			
		||||
          onClick={this.onClick('mention')}
 | 
			
		||||
        >
 | 
			
		||||
          <FormattedMessage
 | 
			
		||||
            id='notifications.filter.mentions'
 | 
			
		||||
            defaultMessage='Mentions'
 | 
			
		||||
          />
 | 
			
		||||
        </button>
 | 
			
		||||
      </div>
 | 
			
		||||
    ) : (
 | 
			
		||||
      <div className='notification__filter-bar'>
 | 
			
		||||
        <button
 | 
			
		||||
          className={selectedFilter === 'all' ? 'active' : ''}
 | 
			
		||||
          onClick={this.onClick('all')}
 | 
			
		||||
        >
 | 
			
		||||
          <FormattedMessage
 | 
			
		||||
            id='notifications.filter.all'
 | 
			
		||||
            defaultMessage='All'
 | 
			
		||||
          />
 | 
			
		||||
        </button>
 | 
			
		||||
        <button
 | 
			
		||||
          className={selectedFilter === 'mention' ? 'active' : ''}
 | 
			
		||||
          onClick={this.onClick('mention')}
 | 
			
		||||
          title={intl.formatMessage(tooltips.mentions)}
 | 
			
		||||
        >
 | 
			
		||||
          <i className='fa fa-fw fa-at' />
 | 
			
		||||
        </button>
 | 
			
		||||
        <button
 | 
			
		||||
          className={selectedFilter === 'favourite' ? 'active' : ''}
 | 
			
		||||
          onClick={this.onClick('favourite')}
 | 
			
		||||
          title={intl.formatMessage(tooltips.favourites)}
 | 
			
		||||
        >
 | 
			
		||||
          <i className='fa fa-fw fa-star' />
 | 
			
		||||
        </button>
 | 
			
		||||
        <button
 | 
			
		||||
          className={selectedFilter === 'reblog' ? 'active' : ''}
 | 
			
		||||
          onClick={this.onClick('reblog')}
 | 
			
		||||
          title={intl.formatMessage(tooltips.boosts)}
 | 
			
		||||
        >
 | 
			
		||||
          <i className='fa fa-fw fa-retweet' />
 | 
			
		||||
        </button>
 | 
			
		||||
        <button
 | 
			
		||||
          className={selectedFilter === 'follow' ? 'active' : ''}
 | 
			
		||||
          onClick={this.onClick('follow')}
 | 
			
		||||
          title={intl.formatMessage(tooltips.follows)}
 | 
			
		||||
        >
 | 
			
		||||
          <i className='fa fa-fw fa-user-plus' />
 | 
			
		||||
        </button>
 | 
			
		||||
      </div>
 | 
			
		||||
    );
 | 
			
		||||
    return renderedElement;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			@ -2,6 +2,7 @@ import { connect } from 'react-redux';
 | 
			
		|||
import { defineMessages, injectIntl } from 'react-intl';
 | 
			
		||||
import ColumnSettings from '../components/column_settings';
 | 
			
		||||
import { changeSetting } from '../../../actions/settings';
 | 
			
		||||
import { setFilter } from '../../../actions/notifications';
 | 
			
		||||
import { clearNotifications } from '../../../actions/notifications';
 | 
			
		||||
import { changeAlerts as changePushNotifications } from '../../../actions/push_notifications';
 | 
			
		||||
import { openModal } from '../../../actions/modal';
 | 
			
		||||
| 
						 | 
				
			
			@ -21,6 +22,9 @@ const mapDispatchToProps = (dispatch, { intl }) => ({
 | 
			
		|||
  onChange (path, checked) {
 | 
			
		||||
    if (path[0] === 'push') {
 | 
			
		||||
      dispatch(changePushNotifications(path.slice(1), checked));
 | 
			
		||||
    } else if (path[0] === 'quickFilter') {
 | 
			
		||||
      dispatch(changeSetting(['notifications', ...path], checked));
 | 
			
		||||
      dispatch(setFilter('all'));
 | 
			
		||||
    } else {
 | 
			
		||||
      dispatch(changeSetting(['notifications', ...path], checked));
 | 
			
		||||
    }
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -0,0 +1,16 @@
 | 
			
		|||
import { connect } from 'react-redux';
 | 
			
		||||
import FilterBar from '../components/filter_bar';
 | 
			
		||||
import { setFilter } from '../../../actions/notifications';
 | 
			
		||||
 | 
			
		||||
const makeMapStateToProps = state => ({
 | 
			
		||||
  selectedFilter: state.getIn(['settings', 'notifications', 'quickFilter', 'active']),
 | 
			
		||||
  advancedMode: state.getIn(['settings', 'notifications', 'quickFilter', 'advanced']),
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
const mapDispatchToProps = (dispatch) => ({
 | 
			
		||||
  selectFilter (newActiveFilter) {
 | 
			
		||||
    dispatch(setFilter(newActiveFilter));
 | 
			
		||||
  },
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
export default connect(makeMapStateToProps, mapDispatchToProps)(FilterBar);
 | 
			
		||||
| 
						 | 
				
			
			@ -9,6 +9,7 @@ import { addColumn, removeColumn, moveColumn } from '../../actions/columns';
 | 
			
		|||
import NotificationContainer from './containers/notification_container';
 | 
			
		||||
import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
 | 
			
		||||
import ColumnSettingsContainer from './containers/column_settings_container';
 | 
			
		||||
import FilterBarContainer from './containers/filter_bar_container';
 | 
			
		||||
import { createSelector } from 'reselect';
 | 
			
		||||
import { List as ImmutableList } from 'immutable';
 | 
			
		||||
import { debounce } from 'lodash';
 | 
			
		||||
| 
						 | 
				
			
			@ -20,11 +21,22 @@ const messages = defineMessages({
 | 
			
		|||
});
 | 
			
		||||
 | 
			
		||||
const getNotifications = createSelector([
 | 
			
		||||
  state => state.getIn(['settings', 'notifications', 'quickFilter', 'show']),
 | 
			
		||||
  state => state.getIn(['settings', 'notifications', 'quickFilter', 'active']),
 | 
			
		||||
  state => ImmutableList(state.getIn(['settings', 'notifications', 'shows']).filter(item => !item).keys()),
 | 
			
		||||
  state => state.getIn(['notifications', 'items']),
 | 
			
		||||
], (excludedTypes, notifications) => notifications.filterNot(item => item !== null && excludedTypes.includes(item.get('type'))));
 | 
			
		||||
], (showFilterBar, allowedType, excludedTypes, notifications) => {
 | 
			
		||||
  if (!showFilterBar || allowedType === 'all') {
 | 
			
		||||
    // used if user changed the notification settings after loading the notifications from the server
 | 
			
		||||
    // otherwise a list of notifications will come pre-filtered from the backend
 | 
			
		||||
    // we need to turn it off for FilterBar in order not to block ourselves from seeing a specific category
 | 
			
		||||
    return notifications.filterNot(item => item !== null && excludedTypes.includes(item.get('type')));
 | 
			
		||||
  }
 | 
			
		||||
  return notifications.filter(item => item !== null && allowedType === item.get('type'));
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
const mapStateToProps = state => ({
 | 
			
		||||
  showFilterBar: state.getIn(['settings', 'notifications', 'quickFilter', 'show']),
 | 
			
		||||
  notifications: getNotifications(state),
 | 
			
		||||
  isLoading: state.getIn(['notifications', 'isLoading'], true),
 | 
			
		||||
  isUnread: state.getIn(['notifications', 'unread']) > 0,
 | 
			
		||||
| 
						 | 
				
			
			@ -38,6 +50,7 @@ class Notifications extends React.PureComponent {
 | 
			
		|||
  static propTypes = {
 | 
			
		||||
    columnId: PropTypes.string,
 | 
			
		||||
    notifications: ImmutablePropTypes.list.isRequired,
 | 
			
		||||
    showFilterBar: PropTypes.bool.isRequired,
 | 
			
		||||
    dispatch: PropTypes.func.isRequired,
 | 
			
		||||
    shouldUpdateScroll: PropTypes.func,
 | 
			
		||||
    intl: PropTypes.object.isRequired,
 | 
			
		||||
| 
						 | 
				
			
			@ -117,12 +130,16 @@ class Notifications extends React.PureComponent {
 | 
			
		|||
  }
 | 
			
		||||
 | 
			
		||||
  render () {
 | 
			
		||||
    const { intl, notifications, shouldUpdateScroll, isLoading, isUnread, columnId, multiColumn, hasMore } = this.props;
 | 
			
		||||
    const { intl, notifications, shouldUpdateScroll, isLoading, isUnread, columnId, multiColumn, hasMore, showFilterBar } = this.props;
 | 
			
		||||
    const pinned = !!columnId;
 | 
			
		||||
    const emptyMessage = <FormattedMessage id='empty_column.notifications' defaultMessage="You don't have any notifications yet. Interact with others to start the conversation." />;
 | 
			
		||||
 | 
			
		||||
    let scrollableContent = null;
 | 
			
		||||
 | 
			
		||||
    const filterBarContainer = showFilterBar
 | 
			
		||||
      ? (<FilterBarContainer />)
 | 
			
		||||
      : null;
 | 
			
		||||
 | 
			
		||||
    if (isLoading && this.scrollableContent) {
 | 
			
		||||
      scrollableContent = this.scrollableContent;
 | 
			
		||||
    } else if (notifications.size > 0 || hasMore) {
 | 
			
		||||
| 
						 | 
				
			
			@ -179,7 +196,7 @@ class Notifications extends React.PureComponent {
 | 
			
		|||
        >
 | 
			
		||||
          <ColumnSettingsContainer />
 | 
			
		||||
        </ColumnHeader>
 | 
			
		||||
 | 
			
		||||
        {filterBarContainer}
 | 
			
		||||
        {scrollContainer}
 | 
			
		||||
      </Column>
 | 
			
		||||
    );
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -223,6 +223,14 @@
 | 
			
		|||
  "notification.reblog": "{name} boosted your status",
 | 
			
		||||
  "notifications.clear": "Clear notifications",
 | 
			
		||||
  "notifications.clear_confirmation": "Are you sure you want to permanently clear all your notifications?",
 | 
			
		||||
  "notifications.filter.all": "All",
 | 
			
		||||
  "notifications.filter.mentions": "Mentions",
 | 
			
		||||
  "notifications.filter.favourites": "Favourites",
 | 
			
		||||
  "notifications.filter.boosts": "Boosts",
 | 
			
		||||
  "notifications.filter.follows": "Follows",
 | 
			
		||||
  "notifications.column_settings.filter_bar.category": "Quick filter bar",
 | 
			
		||||
  "notifications.column_settings.filter_bar.show": "Show",
 | 
			
		||||
  "notifications.column_settings.filter_bar.advanced": "Display all categories",
 | 
			
		||||
  "notifications.column_settings.alert": "Desktop notifications",
 | 
			
		||||
  "notifications.column_settings.favourite": "Favourites:",
 | 
			
		||||
  "notifications.column_settings.follow": "New followers:",
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -223,6 +223,14 @@
 | 
			
		|||
  "notification.reblog": "{name} podbił(a) Twój wpis",
 | 
			
		||||
  "notifications.clear": "Wyczyść powiadomienia",
 | 
			
		||||
  "notifications.clear_confirmation": "Czy na pewno chcesz bezpowrotnie usunąć wszystkie powiadomienia?",
 | 
			
		||||
  "notifications.filter.all": "Wszystkie",
 | 
			
		||||
  "notifications.filter.mentions": "Wspomnienia",
 | 
			
		||||
  "notifications.filter.favourites": "Ulubione",
 | 
			
		||||
  "notifications.filter.boosts": "Podbicia",
 | 
			
		||||
  "notifications.filter.follows": "Śledzenia",
 | 
			
		||||
  "notifications.column_settings.filter_bar.category": "Szybkie filtrowanie",
 | 
			
		||||
  "notifications.column_settings.filter_bar.show": "Pokaż",
 | 
			
		||||
  "notifications.column_settings.filter_bar.advanced": "Wyświetl wszystkie kategorie",
 | 
			
		||||
  "notifications.column_settings.alert": "Powiadomienia na pulpicie",
 | 
			
		||||
  "notifications.column_settings.favourite": "Dodanie do ulubionych:",
 | 
			
		||||
  "notifications.column_settings.follow": "Nowi śledzący:",
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -3,6 +3,7 @@ import {
 | 
			
		|||
  NOTIFICATIONS_EXPAND_SUCCESS,
 | 
			
		||||
  NOTIFICATIONS_EXPAND_REQUEST,
 | 
			
		||||
  NOTIFICATIONS_EXPAND_FAIL,
 | 
			
		||||
  NOTIFICATIONS_FILTER_SET,
 | 
			
		||||
  NOTIFICATIONS_CLEAR,
 | 
			
		||||
  NOTIFICATIONS_SCROLL_TOP,
 | 
			
		||||
} from '../actions/notifications';
 | 
			
		||||
| 
						 | 
				
			
			@ -98,6 +99,8 @@ export default function notifications(state = initialState, action) {
 | 
			
		|||
    return state.set('isLoading', true);
 | 
			
		||||
  case NOTIFICATIONS_EXPAND_FAIL:
 | 
			
		||||
    return state.set('isLoading', false);
 | 
			
		||||
  case NOTIFICATIONS_FILTER_SET:
 | 
			
		||||
    return state.set('items', ImmutableList()).set('hasMore', true);
 | 
			
		||||
  case NOTIFICATIONS_SCROLL_TOP:
 | 
			
		||||
    return updateTop(state, action.top);
 | 
			
		||||
  case NOTIFICATIONS_UPDATE:
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,4 +1,5 @@
 | 
			
		|||
import { SETTING_CHANGE, SETTING_SAVE } from '../actions/settings';
 | 
			
		||||
import { NOTIFICATIONS_FILTER_SET } from '../actions/notifications';
 | 
			
		||||
import { COLUMN_ADD, COLUMN_REMOVE, COLUMN_MOVE, COLUMN_PARAMS_CHANGE } from '../actions/columns';
 | 
			
		||||
import { STORE_HYDRATE } from '../actions/store';
 | 
			
		||||
import { EMOJI_USE } from '../actions/emojis';
 | 
			
		||||
| 
						 | 
				
			
			@ -32,6 +33,12 @@ const initialState = ImmutableMap({
 | 
			
		|||
      mention: true,
 | 
			
		||||
    }),
 | 
			
		||||
 | 
			
		||||
    quickFilter: ImmutableMap({
 | 
			
		||||
      active: 'all',
 | 
			
		||||
      show: true,
 | 
			
		||||
      advanced: false,
 | 
			
		||||
    }),
 | 
			
		||||
 | 
			
		||||
    shows: ImmutableMap({
 | 
			
		||||
      follow: true,
 | 
			
		||||
      favourite: true,
 | 
			
		||||
| 
						 | 
				
			
			@ -112,6 +119,7 @@ export default function settings(state = initialState, action) {
 | 
			
		|||
  switch(action.type) {
 | 
			
		||||
  case STORE_HYDRATE:
 | 
			
		||||
    return hydrate(state, action.state.get('settings'));
 | 
			
		||||
  case NOTIFICATIONS_FILTER_SET:
 | 
			
		||||
  case SETTING_CHANGE:
 | 
			
		||||
    return state
 | 
			
		||||
      .setIn(action.path, action.value)
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1484,6 +1484,52 @@ a.account__display-name {
 | 
			
		|||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.notification__filter-bar {
 | 
			
		||||
  display: flex;
 | 
			
		||||
  flex-wrap: wrap;
 | 
			
		||||
  justify-content: space-between;
 | 
			
		||||
  background: $ui-base-color;
 | 
			
		||||
 | 
			
		||||
  & > button {
 | 
			
		||||
    position: relative;
 | 
			
		||||
    flex-grow: 1;
 | 
			
		||||
    color: $primary-text-color;
 | 
			
		||||
    padding: 10px 5px 12px;
 | 
			
		||||
    text-decoration: none;
 | 
			
		||||
    font-weight: 400;
 | 
			
		||||
    font-size: 15px;
 | 
			
		||||
    line-height: 18px;
 | 
			
		||||
    background: darken($ui-base-color, 4%);
 | 
			
		||||
    border: 0;
 | 
			
		||||
    border-bottom: 1px solid lighten($ui-base-color, 8%);
 | 
			
		||||
    cursor: default;
 | 
			
		||||
 | 
			
		||||
    &.active {
 | 
			
		||||
      color: $secondary-text-color;
 | 
			
		||||
 | 
			
		||||
      &::before,
 | 
			
		||||
      &::after {
 | 
			
		||||
        display: block;
 | 
			
		||||
        content: "";
 | 
			
		||||
        position: absolute;
 | 
			
		||||
        bottom: 0;
 | 
			
		||||
        left: 50%;
 | 
			
		||||
        width: 0;
 | 
			
		||||
        height: 0;
 | 
			
		||||
        transform: translateX(-50%);
 | 
			
		||||
        border-style: solid;
 | 
			
		||||
        border-width: 0 10px 10px;
 | 
			
		||||
        border-color: transparent transparent lighten($ui-base-color, 8%);
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      &::after {
 | 
			
		||||
        bottom: -1px;
 | 
			
		||||
        border-color: transparent transparent $ui-base-color;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.notification__message {
 | 
			
		||||
  margin: 0 10px 0 68px;
 | 
			
		||||
  padding: 8px 0 0;
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
		Loading…
	
	Add table
		
		Reference in a new issue