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
				
			
		|  | @ -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> | ||||
|     ); | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue