mirror of
https://github.com/prometheus/prometheus.git
synced 2025-01-24 20:21:35 -08:00
React UI: Fix issue when changing query then time, the old query executed (#6437)
* React UI: Fix issue when changing query then time, the old query is executed Signed-off-by: Dustin Hooten <dhooten@splunk.com> * pr feedback Signed-off-by: Dustin Hooten <dhooten@splunk.com> * more pr feedback Signed-off-by: Dustin Hooten <dhooten@splunk.com>
This commit is contained in:
parent
67838643ee
commit
fce2e131db
|
@ -23,6 +23,7 @@ describe('ExpressionInput', () => {
|
||||||
'Metric Names': metricNames,
|
'Metric Names': metricNames,
|
||||||
},
|
},
|
||||||
executeQuery: (): void => {},
|
executeQuery: (): void => {},
|
||||||
|
onExpressionChange: (): void => {},
|
||||||
loading: false,
|
loading: false,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -62,7 +63,7 @@ describe('ExpressionInput', () => {
|
||||||
expect(input.prop('type')).toEqual('textarea');
|
expect(input.prop('type')).toEqual('textarea');
|
||||||
expect(input.prop('rows')).toEqual('1');
|
expect(input.prop('rows')).toEqual('1');
|
||||||
expect(input.prop('placeholder')).toEqual('Expression (press Shift+Enter for newlines)');
|
expect(input.prop('placeholder')).toEqual('Expression (press Shift+Enter for newlines)');
|
||||||
expect(expressionInput.state('value')).toEqual('node_cpu');
|
expect(input.prop('value')).toEqual('node_cpu');
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('when autosuggest is closed', () => {
|
describe('when autosuggest is closed', () => {
|
||||||
|
@ -94,6 +95,14 @@ describe('ExpressionInput', () => {
|
||||||
instance.handleInput();
|
instance.handleInput();
|
||||||
expect(stateSpy).toHaveBeenCalled();
|
expect(stateSpy).toHaveBeenCalled();
|
||||||
});
|
});
|
||||||
|
it('should call onExpressionChange', () => {
|
||||||
|
const spyOnExpressionChange = jest.fn();
|
||||||
|
const props = { ...expressionInputProps, onExpressionChange: spyOnExpressionChange };
|
||||||
|
const wrapper = mount(<ExpressionInput {...props} />);
|
||||||
|
const input = wrapper.find(Input);
|
||||||
|
input.simulate('input', { target: { value: 'prometheus_engine_' } });
|
||||||
|
expect(spyOnExpressionChange).toHaveBeenCalledTimes(1);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('onSelect', () => {
|
describe('onSelect', () => {
|
||||||
|
@ -101,7 +110,18 @@ describe('ExpressionInput', () => {
|
||||||
const instance: any = expressionInput.instance();
|
const instance: any = expressionInput.instance();
|
||||||
const stateSpy = jest.spyOn(instance, 'setState');
|
const stateSpy = jest.spyOn(instance, 'setState');
|
||||||
instance.setValue('foo');
|
instance.setValue('foo');
|
||||||
expect(stateSpy).toHaveBeenCalledWith({ value: 'foo', height: 'auto' }, expect.anything());
|
expect(stateSpy).toHaveBeenCalledWith({ height: 'auto' }, expect.anything());
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('onClick', () => {
|
||||||
|
it('executes the query', () => {
|
||||||
|
const spyExecuteQuery = jest.fn();
|
||||||
|
const props = { ...expressionInputProps, executeQuery: spyExecuteQuery };
|
||||||
|
const wrapper = mount(<ExpressionInput {...props} />);
|
||||||
|
const btn = wrapper.find(Button).filterWhere(btn => btn.hasClass('execute-btn'));
|
||||||
|
btn.simulate('click');
|
||||||
|
expect(spyExecuteQuery).toHaveBeenCalledTimes(1);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -10,14 +10,14 @@ import { faSearch, faSpinner } from '@fortawesome/free-solid-svg-icons';
|
||||||
|
|
||||||
interface ExpressionInputProps {
|
interface ExpressionInputProps {
|
||||||
value: string;
|
value: string;
|
||||||
|
onExpressionChange: (expr: string) => void;
|
||||||
autocompleteSections: { [key: string]: string[] };
|
autocompleteSections: { [key: string]: string[] };
|
||||||
executeQuery: (expr: string) => void;
|
executeQuery: () => void;
|
||||||
loading: boolean;
|
loading: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
interface ExpressionInputState {
|
interface ExpressionInputState {
|
||||||
height: number | string;
|
height: number | string;
|
||||||
value: string;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
class ExpressionInput extends Component<ExpressionInputProps, ExpressionInputState> {
|
class ExpressionInput extends Component<ExpressionInputProps, ExpressionInputState> {
|
||||||
|
@ -26,7 +26,6 @@ class ExpressionInput extends Component<ExpressionInputProps, ExpressionInputSta
|
||||||
constructor(props: ExpressionInputProps) {
|
constructor(props: ExpressionInputProps) {
|
||||||
super(props);
|
super(props);
|
||||||
this.state = {
|
this.state = {
|
||||||
value: props.value,
|
|
||||||
height: 'auto',
|
height: 'auto',
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
@ -46,7 +45,9 @@ class ExpressionInput extends Component<ExpressionInputProps, ExpressionInputSta
|
||||||
};
|
};
|
||||||
|
|
||||||
setValue = (value: string) => {
|
setValue = (value: string) => {
|
||||||
this.setState({ value, height: 'auto' }, this.setHeight);
|
const { onExpressionChange } = this.props;
|
||||||
|
onExpressionChange(value);
|
||||||
|
this.setState({ height: 'auto' }, this.setHeight);
|
||||||
};
|
};
|
||||||
|
|
||||||
componentDidUpdate(prevProps: ExpressionInputProps) {
|
componentDidUpdate(prevProps: ExpressionInputProps) {
|
||||||
|
@ -57,14 +58,13 @@ class ExpressionInput extends Component<ExpressionInputProps, ExpressionInputSta
|
||||||
}
|
}
|
||||||
|
|
||||||
handleKeyPress = (event: React.KeyboardEvent<HTMLInputElement>) => {
|
handleKeyPress = (event: React.KeyboardEvent<HTMLInputElement>) => {
|
||||||
|
const { executeQuery } = this.props;
|
||||||
if (event.key === 'Enter' && !event.shiftKey) {
|
if (event.key === 'Enter' && !event.shiftKey) {
|
||||||
this.executeQuery();
|
executeQuery();
|
||||||
event.preventDefault();
|
event.preventDefault();
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
executeQuery = () => this.props.executeQuery(this.exprInputRef.current!.value);
|
|
||||||
|
|
||||||
getSearchMatches = (input: string, expressions: string[]) => {
|
getSearchMatches = (input: string, expressions: string[]) => {
|
||||||
return fuzzy.filter(input.replace(/ /g, ''), expressions, {
|
return fuzzy.filter(input.replace(/ /g, ''), expressions, {
|
||||||
pre: '<strong>',
|
pre: '<strong>',
|
||||||
|
@ -125,7 +125,8 @@ class ExpressionInput extends Component<ExpressionInputProps, ExpressionInputSta
|
||||||
};
|
};
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
const { value, height } = this.state;
|
const { executeQuery, value } = this.props;
|
||||||
|
const { height } = this.state;
|
||||||
return (
|
return (
|
||||||
<Downshift onSelect={this.setValue}>
|
<Downshift onSelect={this.setValue}>
|
||||||
{downshift => (
|
{downshift => (
|
||||||
|
@ -175,7 +176,7 @@ class ExpressionInput extends Component<ExpressionInputProps, ExpressionInputSta
|
||||||
value={value}
|
value={value}
|
||||||
/>
|
/>
|
||||||
<InputGroupAddon addonType="append">
|
<InputGroupAddon addonType="append">
|
||||||
<Button className="execute-btn" color="primary" onClick={this.executeQuery}>
|
<Button className="execute-btn" color="primary" onClick={executeQuery}>
|
||||||
Execute
|
Execute
|
||||||
</Button>
|
</Button>
|
||||||
</InputGroupAddon>
|
</InputGroupAddon>
|
||||||
|
|
|
@ -8,8 +8,7 @@ import TimeInput from './TimeInput';
|
||||||
import DataTable from './DataTable';
|
import DataTable from './DataTable';
|
||||||
import { GraphTabContent } from './graph/GraphTabContent';
|
import { GraphTabContent } from './graph/GraphTabContent';
|
||||||
|
|
||||||
describe('Panel', () => {
|
const defaultProps = {
|
||||||
const props = {
|
|
||||||
options: {
|
options: {
|
||||||
expr: 'prometheus_engine',
|
expr: 'prometheus_engine',
|
||||||
type: PanelType.Table,
|
type: PanelType.Table,
|
||||||
|
@ -27,8 +26,10 @@ describe('Panel', () => {
|
||||||
],
|
],
|
||||||
removePanel: (): void => {},
|
removePanel: (): void => {},
|
||||||
onExecuteQuery: (): void => {},
|
onExecuteQuery: (): void => {},
|
||||||
};
|
};
|
||||||
const panel = shallow(<Panel {...props} />);
|
|
||||||
|
describe('Panel', () => {
|
||||||
|
const panel = shallow(<Panel {...defaultProps} />);
|
||||||
|
|
||||||
it('renders an ExpressionInput', () => {
|
it('renders an ExpressionInput', () => {
|
||||||
const input = panel.find(ExpressionInput);
|
const input = panel.find(ExpressionInput);
|
||||||
|
@ -48,7 +49,7 @@ describe('Panel', () => {
|
||||||
const onOptionsChanged = (opts: PanelOptions): void => {
|
const onOptionsChanged = (opts: PanelOptions): void => {
|
||||||
results.push(opts);
|
results.push(opts);
|
||||||
};
|
};
|
||||||
const panel = shallow(<Panel {...props} onOptionsChanged={onOptionsChanged} />);
|
const panel = shallow(<Panel {...defaultProps} onOptionsChanged={onOptionsChanged} />);
|
||||||
const links = panel.find(NavLink);
|
const links = panel.find(NavLink);
|
||||||
[{ panelType: 'Table', active: true }, { panelType: 'Graph', active: false }].forEach(
|
[{ panelType: 'Table', active: true }, { panelType: 'Graph', active: false }].forEach(
|
||||||
(tc: { panelType: string; active: boolean }, i: number) => {
|
(tc: { panelType: string; active: boolean }, i: number) => {
|
||||||
|
@ -66,8 +67,8 @@ describe('Panel', () => {
|
||||||
it('renders a TabPane with a TimeInput and a DataTable when in table mode', () => {
|
it('renders a TabPane with a TimeInput and a DataTable when in table mode', () => {
|
||||||
const tab = panel.find(TabPane).filterWhere(tab => tab.prop('tabId') === 'table');
|
const tab = panel.find(TabPane).filterWhere(tab => tab.prop('tabId') === 'table');
|
||||||
const timeInput = tab.find(TimeInput);
|
const timeInput = tab.find(TimeInput);
|
||||||
expect(timeInput.prop('time')).toEqual(props.options.endTime);
|
expect(timeInput.prop('time')).toEqual(defaultProps.options.endTime);
|
||||||
expect(timeInput.prop('range')).toEqual(props.options.range);
|
expect(timeInput.prop('range')).toEqual(defaultProps.options.range);
|
||||||
expect(timeInput.prop('placeholder')).toEqual('Evaluation time');
|
expect(timeInput.prop('placeholder')).toEqual('Evaluation time');
|
||||||
expect(tab.find(DataTable)).toHaveLength(1);
|
expect(tab.find(DataTable)).toHaveLength(1);
|
||||||
});
|
});
|
||||||
|
@ -81,7 +82,7 @@ describe('Panel', () => {
|
||||||
resolution: 28,
|
resolution: 28,
|
||||||
stacked: false,
|
stacked: false,
|
||||||
};
|
};
|
||||||
const graphPanel = mount(<Panel {...props} options={options} />);
|
const graphPanel = mount(<Panel {...defaultProps} options={options} />);
|
||||||
const controls = graphPanel.find(GraphControls);
|
const controls = graphPanel.find(GraphControls);
|
||||||
graphPanel.setState({ data: { resultType: 'matrix', result: [] } });
|
graphPanel.setState({ data: { resultType: 'matrix', result: [] } });
|
||||||
const graph = graphPanel.find(GraphTabContent);
|
const graph = graphPanel.find(GraphTabContent);
|
||||||
|
@ -91,4 +92,40 @@ describe('Panel', () => {
|
||||||
expect(controls.prop('stacked')).toEqual(options.stacked);
|
expect(controls.prop('stacked')).toEqual(options.stacked);
|
||||||
expect(graph.prop('stacked')).toEqual(options.stacked);
|
expect(graph.prop('stacked')).toEqual(options.stacked);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('when switching between modes', () => {
|
||||||
|
[{ from: PanelType.Table, to: PanelType.Graph }, { from: PanelType.Graph, to: PanelType.Table }].forEach(
|
||||||
|
({ from, to }: { from: PanelType; to: PanelType }) => {
|
||||||
|
it(`${from} -> ${to} nulls out data`, () => {
|
||||||
|
const props = {
|
||||||
|
...defaultProps,
|
||||||
|
options: { ...defaultProps.options, type: from },
|
||||||
|
};
|
||||||
|
const panel = shallow(<Panel {...props} />);
|
||||||
|
const instance: any = panel.instance();
|
||||||
|
panel.setState({ data: 'somedata' });
|
||||||
|
expect(panel.state('data')).toEqual('somedata');
|
||||||
|
instance.handleChangeType(to);
|
||||||
|
expect(panel.state('data')).toBeNull();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('when changing query then time', () => {
|
||||||
|
it('executes the new query', () => {
|
||||||
|
const initialExpr = 'time()';
|
||||||
|
const newExpr = 'time() - time()';
|
||||||
|
const panel = shallow(<Panel {...defaultProps} options={{ ...defaultProps.options, expr: initialExpr }} />);
|
||||||
|
const instance: any = panel.instance();
|
||||||
|
instance.executeQuery();
|
||||||
|
const executeQuerySpy = jest.spyOn(instance, 'executeQuery');
|
||||||
|
//change query without executing
|
||||||
|
panel.setProps({ options: { ...defaultProps.options, expr: newExpr } });
|
||||||
|
expect(executeQuerySpy).toHaveBeenCalledTimes(0);
|
||||||
|
//execute query implicitly with time change
|
||||||
|
panel.setProps({ options: { ...defaultProps.options, expr: newExpr, endTime: 1575744840 } });
|
||||||
|
expect(executeQuerySpy).toHaveBeenCalledTimes(1);
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -68,36 +68,26 @@ class Panel extends Component<PanelProps & PathPrefixProps, PanelState> {
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
componentDidUpdate(prevProps: PanelProps, prevState: PanelState) {
|
componentDidUpdate({ options: prevOpts }: PanelProps) {
|
||||||
const prevOpts = prevProps.options;
|
const { endTime, range, resolution, type } = this.props.options;
|
||||||
const opts = this.props.options;
|
|
||||||
if (
|
if (
|
||||||
prevOpts.type !== opts.type ||
|
prevOpts.endTime !== endTime ||
|
||||||
prevOpts.range !== opts.range ||
|
prevOpts.range !== range ||
|
||||||
prevOpts.endTime !== opts.endTime ||
|
prevOpts.resolution !== resolution ||
|
||||||
prevOpts.resolution !== opts.resolution ||
|
prevOpts.type !== type
|
||||||
prevOpts.expr !== opts.expr
|
|
||||||
) {
|
) {
|
||||||
if (prevOpts.type !== opts.type) {
|
this.executeQuery();
|
||||||
// If the other options change, we still want to show the old data until the new
|
|
||||||
// query completes, but this is not a good idea when we actually change between
|
|
||||||
// table and graph view, since not all queries work well in both.
|
|
||||||
this.setState({ data: null });
|
|
||||||
}
|
|
||||||
this.executeQuery(opts.expr);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
componentDidMount() {
|
componentDidMount() {
|
||||||
this.executeQuery(this.props.options.expr);
|
this.executeQuery();
|
||||||
}
|
}
|
||||||
|
|
||||||
executeQuery = (expr: string): void => {
|
executeQuery = (): void => {
|
||||||
|
const { expr } = this.props.options;
|
||||||
const queryStart = Date.now();
|
const queryStart = Date.now();
|
||||||
this.props.onExecuteQuery(expr);
|
this.props.onExecuteQuery(expr);
|
||||||
if (this.props.options.expr !== expr) {
|
|
||||||
this.setOptions({ expr: expr });
|
|
||||||
}
|
|
||||||
if (expr === '') {
|
if (expr === '') {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -209,6 +199,11 @@ class Panel extends Component<PanelProps & PathPrefixProps, PanelState> {
|
||||||
this.setOptions({ resolution: resolution });
|
this.setOptions({ resolution: resolution });
|
||||||
};
|
};
|
||||||
|
|
||||||
|
handleChangeType = (type: PanelType) => {
|
||||||
|
this.setState({ data: null });
|
||||||
|
this.setOptions({ type: type });
|
||||||
|
};
|
||||||
|
|
||||||
handleChangeStacking = (stacked: boolean) => {
|
handleChangeStacking = (stacked: boolean) => {
|
||||||
this.setOptions({ stacked: stacked });
|
this.setOptions({ stacked: stacked });
|
||||||
};
|
};
|
||||||
|
@ -221,6 +216,7 @@ class Panel extends Component<PanelProps & PathPrefixProps, PanelState> {
|
||||||
<Col>
|
<Col>
|
||||||
<ExpressionInput
|
<ExpressionInput
|
||||||
value={options.expr}
|
value={options.expr}
|
||||||
|
onExpressionChange={this.handleExpressionChange}
|
||||||
executeQuery={this.executeQuery}
|
executeQuery={this.executeQuery}
|
||||||
loading={this.state.loading}
|
loading={this.state.loading}
|
||||||
autocompleteSections={{
|
autocompleteSections={{
|
||||||
|
@ -239,9 +235,7 @@ class Panel extends Component<PanelProps & PathPrefixProps, PanelState> {
|
||||||
<NavItem>
|
<NavItem>
|
||||||
<NavLink
|
<NavLink
|
||||||
className={options.type === 'table' ? 'active' : ''}
|
className={options.type === 'table' ? 'active' : ''}
|
||||||
onClick={() => {
|
onClick={() => this.handleChangeType(PanelType.Table)}
|
||||||
this.setOptions({ type: 'table' });
|
|
||||||
}}
|
|
||||||
>
|
>
|
||||||
Table
|
Table
|
||||||
</NavLink>
|
</NavLink>
|
||||||
|
@ -249,9 +243,7 @@ class Panel extends Component<PanelProps & PathPrefixProps, PanelState> {
|
||||||
<NavItem>
|
<NavItem>
|
||||||
<NavLink
|
<NavLink
|
||||||
className={options.type === 'graph' ? 'active' : ''}
|
className={options.type === 'graph' ? 'active' : ''}
|
||||||
onClick={() => {
|
onClick={() => this.handleChangeType(PanelType.Graph)}
|
||||||
this.setOptions({ type: 'graph' });
|
|
||||||
}}
|
|
||||||
>
|
>
|
||||||
Graph
|
Graph
|
||||||
</NavLink>
|
</NavLink>
|
||||||
|
|
|
@ -27,7 +27,6 @@ interface TimeInputProps {
|
||||||
time: number | null; // Timestamp in milliseconds.
|
time: number | null; // Timestamp in milliseconds.
|
||||||
range: number; // Range in seconds.
|
range: number; // Range in seconds.
|
||||||
placeholder: string;
|
placeholder: string;
|
||||||
|
|
||||||
onChangeTime: (time: number | null) => void;
|
onChangeTime: (time: number | null) => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,9 +1,11 @@
|
||||||
import * as React from 'react';
|
import * as React from 'react';
|
||||||
import { mount, shallow } from 'enzyme';
|
import { mount, shallow } from 'enzyme';
|
||||||
import PanelList from './pages/PanelList';
|
import PanelList from './PanelList';
|
||||||
import Checkbox from './Checkbox';
|
import Checkbox from '../Checkbox';
|
||||||
import { Alert, Button } from 'reactstrap';
|
import { Alert, Button } from 'reactstrap';
|
||||||
import Panel from './Panel';
|
import Panel, { PanelDefaultOptions } from '../Panel';
|
||||||
|
import { decodePanelOptionsFromQueryString } from '../utils/urlParams';
|
||||||
|
import ExpressionInput from '../ExpressionInput';
|
||||||
|
|
||||||
describe('PanelList', () => {
|
describe('PanelList', () => {
|
||||||
it('renders a query history checkbox', () => {
|
it('renders a query history checkbox', () => {
|
||||||
|
@ -37,4 +39,12 @@ describe('PanelList', () => {
|
||||||
expect(btn.prop('color')).toEqual('primary');
|
expect(btn.prop('color')).toEqual('primary');
|
||||||
expect(btn.children().text()).toEqual('Add Panel');
|
expect(btn.children().text()).toEqual('Add Panel');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('updates URL when a query is executed', () => {
|
||||||
|
const panelList = mount(<PanelList />);
|
||||||
|
const instance: any = panelList.instance();
|
||||||
|
const updateURLSpy = jest.spyOn(instance, 'updateURL');
|
||||||
|
instance.handleExecuteQuery('new');
|
||||||
|
expect(updateURLSpy).toHaveBeenCalledTimes(1);
|
||||||
|
});
|
||||||
});
|
});
|
|
@ -96,7 +96,7 @@ class PanelList extends Component<RouteComponentProps & PathPrefixProps, PanelLi
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
handleQueryHistory = (query: string) => {
|
handleExecuteQuery = (query: string) => {
|
||||||
const isSimpleMetric = this.state.metricNames.indexOf(query) !== -1;
|
const isSimpleMetric = this.state.metricNames.indexOf(query) !== -1;
|
||||||
if (isSimpleMetric || !query.length) {
|
if (isSimpleMetric || !query.length) {
|
||||||
return;
|
return;
|
||||||
|
@ -110,6 +110,7 @@ class PanelList extends Component<RouteComponentProps & PathPrefixProps, PanelLi
|
||||||
);
|
);
|
||||||
localStorage.setItem('history', JSON.stringify(extendedItems.slice(0, 50)));
|
localStorage.setItem('history', JSON.stringify(extendedItems.slice(0, 50)));
|
||||||
this.updatePastQueries();
|
this.updatePastQueries();
|
||||||
|
this.updateURL();
|
||||||
};
|
};
|
||||||
|
|
||||||
updateURL() {
|
updateURL() {
|
||||||
|
@ -119,7 +120,7 @@ class PanelList extends Component<RouteComponentProps & PathPrefixProps, PanelLi
|
||||||
|
|
||||||
handleOptionsChanged = (id: string, options: PanelOptions) => {
|
handleOptionsChanged = (id: string, options: PanelOptions) => {
|
||||||
const updatedPanels = this.state.panels.map(p => (id === p.id ? { ...p, options } : p));
|
const updatedPanels = this.state.panels.map(p => (id === p.id ? { ...p, options } : p));
|
||||||
this.setState({ panels: updatedPanels }, this.updateURL);
|
this.setState({ panels: updatedPanels });
|
||||||
};
|
};
|
||||||
|
|
||||||
addPanel = () => {
|
addPanel = () => {
|
||||||
|
@ -181,7 +182,7 @@ class PanelList extends Component<RouteComponentProps & PathPrefixProps, PanelLi
|
||||||
</Row>
|
</Row>
|
||||||
{panels.map(({ id, options }) => (
|
{panels.map(({ id, options }) => (
|
||||||
<Panel
|
<Panel
|
||||||
onExecuteQuery={this.handleQueryHistory}
|
onExecuteQuery={this.handleExecuteQuery}
|
||||||
key={id}
|
key={id}
|
||||||
options={options}
|
options={options}
|
||||||
onOptionsChanged={opts => this.handleOptionsChanged(id, opts)}
|
onOptionsChanged={opts => this.handleOptionsChanged(id, opts)}
|
||||||
|
|
Loading…
Reference in a new issue