basic filter form component set up
This commit is contained in:
parent
eeaf9f0b8e
commit
bdd9204b3c
3 changed files with 174 additions and 6 deletions
|
@ -168,10 +168,10 @@ class SidebarMenu extends ImmutablePureComponent {
|
||||||
<Icon id='times-circle' />
|
<Icon id='times-circle' />
|
||||||
<span className='sidebar-menu-item__title'>{intl.formatMessage(messages.mutes)}</span>
|
<span className='sidebar-menu-item__title'>{intl.formatMessage(messages.mutes)}</span>
|
||||||
</NavLink>
|
</NavLink>
|
||||||
{/* <NavLink className='sidebar-menu-item' to='/filters' onClick={onClose}>
|
<NavLink className='sidebar-menu-item' to='/filters' onClick={onClose}>
|
||||||
<Icon id='filter' />
|
<Icon id='filter' />
|
||||||
<span className='sidebar-menu-item__title'>{intl.formatMessage(messages.filters)}</span>
|
<span className='sidebar-menu-item__title'>{intl.formatMessage(messages.filters)}</span>
|
||||||
</NavLink> */}
|
</NavLink>
|
||||||
{ isStaff && <a className='sidebar-menu-item' href={'/pleroma/admin/'} onClick={onClose}>
|
{ isStaff && <a className='sidebar-menu-item' href={'/pleroma/admin/'} onClick={onClose}>
|
||||||
<Icon id='shield' />
|
<Icon id='shield' />
|
||||||
<span className='sidebar-menu-item__title'>{intl.formatMessage(messages.admin_settings)}</span>
|
<span className='sidebar-menu-item__title'>{intl.formatMessage(messages.admin_settings)}</span>
|
||||||
|
|
|
@ -76,7 +76,7 @@ class ActionBar extends React.PureComponent {
|
||||||
menu.push({ text: intl.formatMessage(messages.mutes), to: '/mutes' });
|
menu.push({ text: intl.formatMessage(messages.mutes), to: '/mutes' });
|
||||||
menu.push({ text: intl.formatMessage(messages.blocks), to: '/blocks' });
|
menu.push({ text: intl.formatMessage(messages.blocks), to: '/blocks' });
|
||||||
menu.push({ text: intl.formatMessage(messages.domain_blocks), to: '/domain_blocks' });
|
menu.push({ text: intl.formatMessage(messages.domain_blocks), to: '/domain_blocks' });
|
||||||
// menu.push({ text: intl.formatMessage(messages.filters), to: '/filters' });
|
menu.push({ text: intl.formatMessage(messages.filters), to: '/filters' });
|
||||||
menu.push(null);
|
menu.push(null);
|
||||||
menu.push({ text: intl.formatMessage(messages.keyboard_shortcuts), action: this.handleHotkeyClick });
|
menu.push({ text: intl.formatMessage(messages.keyboard_shortcuts), action: this.handleHotkeyClick });
|
||||||
if (isStaff) {
|
if (isStaff) {
|
||||||
|
|
|
@ -4,16 +4,49 @@ import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
|
||||||
import ImmutablePureComponent from 'react-immutable-pure-component';
|
import ImmutablePureComponent from 'react-immutable-pure-component';
|
||||||
import PropTypes from 'prop-types';
|
import PropTypes from 'prop-types';
|
||||||
import Column from '../ui/components/column';
|
import Column from '../ui/components/column';
|
||||||
import { fetchFilters } from '../../actions/filters';
|
import { fetchFilters, createFilter } from '../../actions/filters';
|
||||||
|
import ScrollableList from '../../components/scrollable_list';
|
||||||
|
import Button from 'soapbox/components/button';
|
||||||
|
import {
|
||||||
|
SimpleForm,
|
||||||
|
SimpleInput,
|
||||||
|
FieldsGroup,
|
||||||
|
TextInput,
|
||||||
|
SelectDropdown,
|
||||||
|
Checkbox,
|
||||||
|
} from 'soapbox/features/forms';
|
||||||
|
import { showAlert } from 'soapbox/actions/alerts';
|
||||||
|
|
||||||
const messages = defineMessages({
|
const messages = defineMessages({
|
||||||
heading: { id: 'column.filters', defaultMessage: 'Muted words' },
|
heading: { id: 'column.filters', defaultMessage: 'Muted words' },
|
||||||
|
keyword: { id: 'column.filters.keyword', defaultMessage: 'Keyword or phrase' },
|
||||||
|
expires: { id: 'column.filters.expires', defaultMessage: 'Expire after' },
|
||||||
|
home_timeline: { id: 'column.filters.home_timeline', defaultMessage: 'Home timeline' },
|
||||||
|
public_timeline: { id: 'column.filters.public_timeline', defaultMessage: 'Public timeline' },
|
||||||
|
notifications: { id: 'column.filters.notifications', defaultMessage: 'Notifications' },
|
||||||
|
conversations: { id: 'column.filters.conversations', defaultMessage: 'Conversations' },
|
||||||
|
drop_header: { id: 'column.filters.drop_header', defaultMessage: 'Drop instead of hide' },
|
||||||
|
drop_hint: { id: 'column.filters.drop_hint', defaultMessage: 'Filtered posts will disappear irreversibly, even if filter is later removed' },
|
||||||
|
whole_word_header: { id: 'column.filters.whole_word_header', defaultMessage: 'Whole word' },
|
||||||
|
whole_word_hint: { id: 'column.filters.whole_word_hint', defaultMessage: 'When the keyword or phrase is alphanumeric only, it will only be applied if it matches the whole word' },
|
||||||
|
add_new: { id: 'column.filters.add_new', defaultMessage: 'Add New Muted Word' },
|
||||||
|
error: { id: 'column.filters.error', defaultMessage: 'Error adding filter' },
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const expirations = {
|
||||||
|
1800: 'Never',
|
||||||
|
3600: '30 minutes',
|
||||||
|
21600: '1 hour',
|
||||||
|
43200: '12 hours',
|
||||||
|
86400 : '1 day',
|
||||||
|
604800: '1 week',
|
||||||
|
};
|
||||||
|
|
||||||
const mapStateToProps = state => ({
|
const mapStateToProps = state => ({
|
||||||
filters: state.get('filters'),
|
filters: state.get('filters'),
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
||||||
export default @connect(mapStateToProps)
|
export default @connect(mapStateToProps)
|
||||||
@injectIntl
|
@injectIntl
|
||||||
class Filters extends ImmutablePureComponent {
|
class Filters extends ImmutablePureComponent {
|
||||||
|
@ -24,17 +57,152 @@ class Filters extends ImmutablePureComponent {
|
||||||
intl: PropTypes.object.isRequired,
|
intl: PropTypes.object.isRequired,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
state = {
|
||||||
|
phrase: '',
|
||||||
|
expires_at: '',
|
||||||
|
context: {
|
||||||
|
home_timeline: false,
|
||||||
|
public_timeline: false,
|
||||||
|
notifications: false,
|
||||||
|
conversations: false,
|
||||||
|
},
|
||||||
|
irreversible: false,
|
||||||
|
whole_word: true,
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
componentDidMount() {
|
componentDidMount() {
|
||||||
this.props.dispatch(fetchFilters());
|
this.props.dispatch(fetchFilters());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
handleInputChange = e => {
|
||||||
|
this.setState({ [e.target.name]: e.target.value });
|
||||||
|
}
|
||||||
|
|
||||||
|
handleSelectChange = e => {
|
||||||
|
this.setState({ [e.target.name]: e.target.value });
|
||||||
|
}
|
||||||
|
|
||||||
|
handleCheckboxChange = e => {
|
||||||
|
this.setState({ [e.target.name]: e.target.checked });
|
||||||
|
}
|
||||||
|
|
||||||
|
handleAddNew = e => {
|
||||||
|
e.preventDefault();
|
||||||
|
const { intl, dispatch } = this.state;
|
||||||
|
const { phrase, context, whole_word, expires_at } = this.state;
|
||||||
|
dispatch(createFilter(phrase, context, whole_word, expires_at)).then(response => {
|
||||||
|
dispatch(fetchFilters());
|
||||||
|
}).catch(error => {
|
||||||
|
dispatch(showAlert('', intl.formatMessage(messages.error)));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
const { intl } = this.props;
|
const { intl, filters } = this.props;
|
||||||
const emptyMessage = <FormattedMessage id='empty_column.filters' defaultMessage="You haven't created any muted words yet." />;
|
const emptyMessage = <FormattedMessage id='empty_column.filters' defaultMessage="You haven't created any muted words yet." />;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Column icon='filter' heading={intl.formatMessage(messages.heading)} backBtnSlim>
|
<Column icon='filter' heading={intl.formatMessage(messages.heading)} backBtnSlim>
|
||||||
{emptyMessage}
|
|
||||||
|
<SimpleForm>
|
||||||
|
<div className='filter-settings-panel'>
|
||||||
|
<h1 className='filter-settings-panel__add-new'>
|
||||||
|
<FormattedMessage id='filters.add_new_title' defaultMessage='Add New Filter' />
|
||||||
|
</h1>
|
||||||
|
|
||||||
|
<fieldset disabled={false}>
|
||||||
|
<FieldsGroup>
|
||||||
|
|
||||||
|
<SimpleInput
|
||||||
|
label={intl.formatMessage(messages.keyword)}
|
||||||
|
required
|
||||||
|
type='text'
|
||||||
|
name='custom_filter_phrase'
|
||||||
|
onChange={this.handleInputChange}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<SelectDropdown
|
||||||
|
label={intl.formatMessage(messages.expires)}
|
||||||
|
items={expirations}
|
||||||
|
defaultValue={expirations.never}
|
||||||
|
onChange={this.handleSelectChange}
|
||||||
|
/>
|
||||||
|
</FieldsGroup>
|
||||||
|
|
||||||
|
<FieldsGroup>
|
||||||
|
<label className='checkboxes required'>
|
||||||
|
<FormattedMessage id='filters.context_header' defaultMessage='Filter contexts' />
|
||||||
|
</label>
|
||||||
|
<span className='hint'>
|
||||||
|
<FormattedMessage id='filters.context_hint' defaultMessage='One or multiple contexts where the filter should apply' />
|
||||||
|
</span>
|
||||||
|
<Checkbox
|
||||||
|
label={intl.formatMessage(messages.home_timeline)}
|
||||||
|
name='home_timeline'
|
||||||
|
checked={this.state.context.home_timeline}
|
||||||
|
onChange={this.handleCheckboxChange}
|
||||||
|
/>
|
||||||
|
<Checkbox
|
||||||
|
label={intl.formatMessage(messages.public_timeline)}
|
||||||
|
name='public_timeline'
|
||||||
|
checked={this.state.context.public_timeline}
|
||||||
|
onChange={this.handleCheckboxChange}
|
||||||
|
/>
|
||||||
|
<Checkbox
|
||||||
|
label={intl.formatMessage(messages.notifications)}
|
||||||
|
name='notifications'
|
||||||
|
checked={this.state.context.notifications}
|
||||||
|
onChange={this.handleCheckboxChange}
|
||||||
|
/>
|
||||||
|
<Checkbox
|
||||||
|
label={intl.formatMessage(messages.conversations)}
|
||||||
|
name='conversations'
|
||||||
|
checked={this.state.context.conversations}
|
||||||
|
onChange={this.handleCheckboxChange}
|
||||||
|
/>
|
||||||
|
|
||||||
|
</FieldsGroup>
|
||||||
|
|
||||||
|
<FieldsGroup>
|
||||||
|
<Checkbox
|
||||||
|
label={intl.formatMessage(messages.drop_header)}
|
||||||
|
hint={intl.formatMessage(messages.drop_hint)}
|
||||||
|
name='irreversible'
|
||||||
|
checked={this.state.irreversible}
|
||||||
|
onChange={this.handleCheckboxChange}
|
||||||
|
/>
|
||||||
|
<Checkbox
|
||||||
|
label={intl.formatMessage(messages.whole_word_header)}
|
||||||
|
hint={intl.formatMessage(messages.whole_word_hint)}
|
||||||
|
name='whole_word'
|
||||||
|
checked={this.state.whole_word}
|
||||||
|
onChange={this.handleCheckboxChange}
|
||||||
|
/>
|
||||||
|
</FieldsGroup>
|
||||||
|
</fieldset>
|
||||||
|
|
||||||
|
<Button className='button button-primary setup' text={intl.formatMessage(messages.add_new)} onClick={this.props.handleAddNew} />
|
||||||
|
|
||||||
|
<ScrollableList
|
||||||
|
scrollKey='mutes'
|
||||||
|
onLoadMore={this.handleLoadMore}
|
||||||
|
emptyMessage={emptyMessage}
|
||||||
|
>
|
||||||
|
{filters.map((filter, i) => (
|
||||||
|
<div key={i} className='backup_code'>
|
||||||
|
<div className='backup_code'>{filter}</div>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</ScrollableList>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
</SimpleForm>
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
</Column>
|
</Column>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue