87d95a1eb5
This PR adds a new notification cleaning mode, super perfectly tuned for accessibility, and removes the previous notification cleaning functionality as it's now redundant. * w.i.p. notif clearing mode * Better CSS for selected notification and shorter text if Stretch is off * wip for rebase ~ * all working in notif clearing mode, except the actual removal * bulk delete route for piggo * cleaning + refactor. endpoint gives 422 for some reason * formatting * use the right route * fix broken destroy_multiple * load more notifs after succ cleaning * satisfy eslint * Removed CSS for the old notif delete button * Tabindex=0 is mandatory In order to make it possible to tab to this element you must have tab index = 0. Removing this violates WCAG and makes it impossible to use the interface without good eyesight and a mouse. So nobody with certain mobility impairments, vision impairments, or brain injuries would be able to use this feature if you don't have tabindex=0 * Corrected aria-label Previous label implied a different behavior from what actually happens * aria role localization & made the overlay behave like a checkbox * checkboxes css and better contrast * color tuning for the notif overlay * fanceh checkboxes etc and nice backgrounds * SHUT UP TRAVIS
169 lines
5.5 KiB
JavaScript
169 lines
5.5 KiB
JavaScript
import React from 'react';
|
|
import PropTypes from 'prop-types';
|
|
import classNames from 'classnames';
|
|
import { defineMessages, FormattedMessage, injectIntl } from 'react-intl';
|
|
import ImmutablePropTypes from 'react-immutable-proptypes';
|
|
|
|
// Glitch imports
|
|
import NotificationPurgeButtonsContainer from '../../glitch/components/column/notif_cleaning_widget/container';
|
|
|
|
const messages = defineMessages({
|
|
titleNotifClearing: { id: 'column.notifications_clearing', defaultMessage: 'Dismiss selected notifications:' },
|
|
titleNotifClearingShort: { id: 'column.notifications_clearing_short', defaultMessage: 'Dismiss selected:' },
|
|
});
|
|
|
|
@injectIntl
|
|
export default class ColumnHeader extends React.PureComponent {
|
|
|
|
static contextTypes = {
|
|
router: PropTypes.object,
|
|
};
|
|
|
|
static propTypes = {
|
|
title: PropTypes.node.isRequired,
|
|
icon: PropTypes.string.isRequired,
|
|
active: PropTypes.bool,
|
|
localSettings : ImmutablePropTypes.map,
|
|
multiColumn: PropTypes.bool,
|
|
showBackButton: PropTypes.bool,
|
|
notifCleaning: PropTypes.bool, // true only for the notification column
|
|
notifCleaningActive: PropTypes.bool,
|
|
children: PropTypes.node,
|
|
pinned: PropTypes.bool,
|
|
onPin: PropTypes.func,
|
|
onMove: PropTypes.func,
|
|
onClick: PropTypes.func,
|
|
intl: PropTypes.object.isRequired,
|
|
};
|
|
|
|
state = {
|
|
collapsed: true,
|
|
animating: false,
|
|
};
|
|
|
|
handleToggleClick = (e) => {
|
|
e.stopPropagation();
|
|
this.setState({ collapsed: !this.state.collapsed, animating: true });
|
|
}
|
|
|
|
handleTitleClick = () => {
|
|
this.props.onClick();
|
|
}
|
|
|
|
handleMoveLeft = () => {
|
|
this.props.onMove(-1);
|
|
}
|
|
|
|
handleMoveRight = () => {
|
|
this.props.onMove(1);
|
|
}
|
|
|
|
handleBackClick = () => {
|
|
// if history is exhausted, or we would leave mastodon, just go to root.
|
|
if (window.history && (window.history.length === 1 || window.history.length === window._mastoInitialHistoryLen)) {
|
|
this.context.router.history.push('/');
|
|
} else {
|
|
this.context.router.history.goBack();
|
|
}
|
|
}
|
|
|
|
handleTransitionEnd = () => {
|
|
this.setState({ animating: false });
|
|
}
|
|
|
|
render () {
|
|
const { intl, icon, active, children, pinned, onPin, multiColumn, showBackButton, notifCleaning, localSettings } = this.props;
|
|
const { collapsed, animating } = this.state;
|
|
|
|
let title = this.props.title;
|
|
if (notifCleaning && this.props.notifCleaningActive) {
|
|
title = intl.formatMessage(localSettings.getIn(['stretch']) ?
|
|
messages.titleNotifClearing :
|
|
messages.titleNotifClearingShort);
|
|
}
|
|
|
|
const wrapperClassName = classNames('column-header__wrapper', {
|
|
'active': active,
|
|
});
|
|
|
|
const buttonClassName = classNames('column-header', {
|
|
'active': active,
|
|
});
|
|
|
|
const collapsibleClassName = classNames('column-header__collapsible', {
|
|
'collapsed': collapsed,
|
|
'animating': animating,
|
|
});
|
|
|
|
const collapsibleButtonClassName = classNames('column-header__button', {
|
|
'active': !collapsed,
|
|
});
|
|
|
|
let extraContent, pinButton, moveButtons, backButton, collapseButton;
|
|
|
|
if (children) {
|
|
extraContent = (
|
|
<div key='extra-content' className='column-header__collapsible__extra'>
|
|
{children}
|
|
</div>
|
|
);
|
|
}
|
|
|
|
if (multiColumn && pinned) {
|
|
pinButton = <button key='pin-button' className='text-btn column-header__setting-btn' onClick={onPin}><i className='fa fa fa-times' /> <FormattedMessage id='column_header.unpin' defaultMessage='Unpin' /></button>;
|
|
|
|
moveButtons = (
|
|
<div key='move-buttons' className='column-header__setting-arrows'>
|
|
<button className='text-btn column-header__setting-btn' onClick={this.handleMoveLeft}><i className='fa fa-chevron-left' /></button>
|
|
<button className='text-btn column-header__setting-btn' onClick={this.handleMoveRight}><i className='fa fa-chevron-right' /></button>
|
|
</div>
|
|
);
|
|
} else if (multiColumn) {
|
|
pinButton = <button key='pin-button' className='text-btn column-header__setting-btn' onClick={onPin}><i className='fa fa fa-plus' /> <FormattedMessage id='column_header.pin' defaultMessage='Pin' /></button>;
|
|
}
|
|
|
|
if (!pinned && (multiColumn || showBackButton)) {
|
|
backButton = (
|
|
<button onClick={this.handleBackClick} className='column-header__back-button'>
|
|
<i className='fa fa-fw fa-chevron-left column-back-button__icon' />
|
|
<FormattedMessage id='column_back_button.label' defaultMessage='Back' />
|
|
</button>
|
|
);
|
|
}
|
|
|
|
const collapsedContent = [
|
|
extraContent,
|
|
];
|
|
|
|
if (multiColumn) {
|
|
collapsedContent.push(moveButtons);
|
|
collapsedContent.push(pinButton);
|
|
}
|
|
|
|
if (children || multiColumn) {
|
|
collapseButton = <button className={collapsibleButtonClassName} onClick={this.handleToggleClick}><i className='fa fa-sliders' /></button>;
|
|
}
|
|
|
|
return (
|
|
<div className={wrapperClassName}>
|
|
<div role='button heading' tabIndex='0' className={buttonClassName} onClick={this.handleTitleClick}>
|
|
<i className={`fa fa-fw fa-${icon} column-header__icon`} />
|
|
{title}
|
|
|
|
<div className='column-header__buttons'>
|
|
{notifCleaning ? (<NotificationPurgeButtonsContainer />) : null}
|
|
{backButton}
|
|
{collapseButton}
|
|
</div>
|
|
</div>
|
|
|
|
<div className={collapsibleClassName} onTransitionEnd={this.handleTransitionEnd}>
|
|
<div className='column-header__collapsible-inner'>
|
|
{(!collapsed || animating) && collapsedContent}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
}
|