ui: add "reset to safe defaults" in AdvancedMultiSelect

If the user selects an unsafe exploit or post breach action, a yellow
warning button appears that allows the user to reset to safe defaults.
This commit is contained in:
Mike Salvatore 2021-01-21 15:07:19 -05:00
parent bf6db078a6
commit e04e11e4ac
3 changed files with 62 additions and 10 deletions

View File

@ -1,5 +1,5 @@
import React from 'react'; import React from 'react';
import {Form} from 'react-bootstrap'; import {Button, Card, Form} from 'react-bootstrap';
import {cloneDeep} from 'lodash'; import {cloneDeep} from 'lodash';
@ -7,21 +7,45 @@ import {getComponentHeight} from './utils/HeightCalculator';
import InfoPane from './InfoPane'; import InfoPane from './InfoPane';
import {MasterCheckbox, MasterCheckboxState} from './MasterCheckbox'; import {MasterCheckbox, MasterCheckboxState} from './MasterCheckbox';
import ChildCheckbox from './ChildCheckbox'; import ChildCheckbox from './ChildCheckbox';
import {getFullDefinitionByKey, getDefaultPaneParams} from './JsonSchemaHelpers.js'; import {getFullDefinitionByKey, getDefaultPaneParams} from './JsonSchemaHelpers';
function AdvancedMultiSelectHeader(props) {
const {
title,
disabled,
onCheckboxClick,
checkboxState,
hideReset,
onResetClick
} = props;
return (
<Card.Header className="d-flex justify-content-between">
<MasterCheckbox title={title} disabled={disabled} onClick={onCheckboxClick} checkboxState={checkboxState}/>
<Button className={'reset-safe-defaults'} type={'reset'} variant={'warning'}
hidden={hideReset} onClick={onResetClick}>
Reset to safe defaults
</Button>
</Card.Header>
);
}
class AdvancedMultiSelect extends React.Component { class AdvancedMultiSelect extends React.Component {
constructor(props) { constructor(props) {
super(props); super(props);
this.enumOptions = props.options.enumOptions; this.enumOptions = props.options.enumOptions;
this.defaultValues = props.schema.default;
this.state = { this.state = {
masterCheckboxState: this.getMasterCheckboxState(props.value), masterCheckboxState: this.getMasterCheckboxState(props.value),
hideReset: this.getHideResetState(props.value),
infoPaneParams: getDefaultPaneParams(props.schema.items.$ref, props.registry) infoPaneParams: getDefaultPaneParams(props.schema.items.$ref, props.registry)
}; };
this.onMasterCheckboxClick = this.onMasterCheckboxClick.bind(this); this.onMasterCheckboxClick = this.onMasterCheckboxClick.bind(this);
this.onChildCheckboxClick = this.onChildCheckboxClick.bind(this); this.onChildCheckboxClick = this.onChildCheckboxClick.bind(this);
this.onResetClick = this.onResetClick.bind(this);
this.setPaneInfo = this.setPaneInfo.bind(this, props.schema.items.$ref, props.registry); this.setPaneInfo = this.setPaneInfo.bind(this, props.schema.items.$ref, props.registry);
} }
@ -34,6 +58,7 @@ class AdvancedMultiSelect extends React.Component {
this.props.onChange(newValues); this.props.onChange(newValues);
this.setMasterCheckboxState(newValues); this.setMasterCheckboxState(newValues);
this.setHideResetState(newValues);
} }
onChildCheckboxClick(value) { onChildCheckboxClick(value) {
@ -41,6 +66,7 @@ class AdvancedMultiSelect extends React.Component {
this.props.onChange(selectValues); this.props.onChange(selectValues);
this.setMasterCheckboxState(selectValues); this.setMasterCheckboxState(selectValues);
this.setHideResetState(selectValues);
} }
getSelectValuesAfterClick(clickedValue) { getSelectValuesAfterClick(clickedValue) {
@ -72,11 +98,34 @@ class AdvancedMultiSelect extends React.Component {
return MasterCheckboxState.ALL; return MasterCheckboxState.ALL;
} }
onResetClick() {
this.props.onChange(this.defaultValues);
this.setHideResetState(this.defaultValues);
this.setMasterCheckboxState(this.defaultValues);
this.setPaneInfoToDefault();
}
setHideResetState(selectValues) {
this.setState(() => ({
hideReset: this.getHideResetState(selectValues)
}));
}
getHideResetState(selectValues) {
return selectValues.every((value) => this.defaultValues.includes(value));
}
setPaneInfo(refString, registry, itemKey) { setPaneInfo(refString, registry, itemKey) {
let definitionObj = getFullDefinitionByKey(refString, registry, itemKey); let definitionObj = getFullDefinitionByKey(refString, registry, itemKey);
this.setState({infoPaneParams: {title: definitionObj.title, content: definitionObj.info, link: definitionObj.link}}); this.setState({infoPaneParams: {title: definitionObj.title, content: definitionObj.info, link: definitionObj.link}});
} }
setPaneInfoToDefault() {
this.setState(() => ({
infoPaneParams: getDefaultPaneParams(this.props.schema.items.$ref, this.props.registry)
}));
}
render() { render() {
const { const {
schema, schema,
@ -90,9 +139,10 @@ class AdvancedMultiSelect extends React.Component {
return ( return (
<div className={'advanced-multi-select'}> <div className={'advanced-multi-select'}>
<MasterCheckbox title={schema.title} <AdvancedMultiSelectHeader title={schema.title}
disabled={disabled} onClick={this.onMasterCheckboxClick} disabled={disabled} onCheckboxClick={this.onMasterCheckboxClick}
checkboxState={this.state.masterCheckboxState}/> checkboxState={this.state.masterCheckboxState}
hideReset={this.state.hideReset} onResetClick={this.onResetClick}/>
<Form.Group <Form.Group
style={{height: `${getComponentHeight(this.enumOptions.length)}px`}} style={{height: `${getComponentHeight(this.enumOptions.length)}px`}}
id={id} multiple={multiple} className='choice-block form-control' id={id} multiple={multiple} className='choice-block form-control'

View File

@ -1,5 +1,5 @@
import React from 'react'; import React from 'react';
import {Card, Button} from 'react-bootstrap'; import {Button} from 'react-bootstrap';
import {FontAwesomeIcon} from '@fortawesome/react-fontawesome'; import {FontAwesomeIcon} from '@fortawesome/react-fontawesome';
import {faCheckSquare} from '@fortawesome/free-solid-svg-icons'; import {faCheckSquare} from '@fortawesome/free-solid-svg-icons';
@ -29,12 +29,12 @@ function MasterCheckbox(props) {
} }
return ( return (
<Card.Header> <div className={'master-checkbox'}>
<Button key={`${title}-button`} variant={'link'} disabled={disabled} onClick={onClick}> <Button key={`${title}-button`} variant={'link'} disabled={disabled} onClick={onClick}>
<FontAwesomeIcon icon={newCheckboxIcon}/> <FontAwesomeIcon icon={newCheckboxIcon}/>
</Button> </Button>
<span className={'header-title'}>{title}</span> <span className={'header-title'}>{title}</span>
</Card.Header> </div>
); );
} }

View File

@ -18,12 +18,14 @@
padding-bottom: 5px; padding-bottom: 5px;
} }
.advanced-multi-select .card-header button { .advanced-multi-select .card-header .master-checkbox span {
padding-top: 0; padding-bottom: 0.188rem;
} }
.advanced-multi-select .card-header .header-title { .advanced-multi-select .card-header .header-title {
font-size: 1.2em; font-size: 1.2em;
display: inline-block;
vertical-align: middle;
} }
.advanced-multi-select .choice-block .form-group { .advanced-multi-select .choice-block .form-group {