forked from p34709852/monkey
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:
parent
bf6db078a6
commit
e04e11e4ac
|
@ -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'
|
||||||
|
|
|
@ -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>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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 {
|
||||||
|
|
Loading…
Reference in New Issue