mirror of
https://github.com/grafana/grafana.git
synced 2025-02-16 18:34:52 -06:00
223 lines
5.7 KiB
TypeScript
223 lines
5.7 KiB
TypeScript
// Libraries
|
|
import React, { PureComponent } from 'react';
|
|
|
|
// Utils & Services
|
|
import { getAngularLoader, AngularComponent } from 'app/core/services/AngularLoader';
|
|
|
|
// Components
|
|
import { EditorTabBody } from './EditorTabBody';
|
|
import { VizTypePicker } from './VizTypePicker';
|
|
import { FadeIn } from 'app/core/components/Animations/FadeIn';
|
|
import { PanelOptionSection } from './PanelOptionSection';
|
|
|
|
// Types
|
|
import { PanelModel } from '../panel_model';
|
|
import { DashboardModel } from '../dashboard_model';
|
|
import { PanelPlugin } from 'app/types/plugins';
|
|
|
|
interface Props {
|
|
panel: PanelModel;
|
|
dashboard: DashboardModel;
|
|
plugin: PanelPlugin;
|
|
angularPanel?: AngularComponent;
|
|
onTypeChanged: (newType: PanelPlugin) => void;
|
|
}
|
|
|
|
interface State {
|
|
isVizPickerOpen: boolean;
|
|
searchQuery: string;
|
|
}
|
|
|
|
export class VisualizationTab extends PureComponent<Props, State> {
|
|
element: HTMLElement;
|
|
angularOptions: AngularComponent;
|
|
searchInput: HTMLElement;
|
|
|
|
constructor(props) {
|
|
super(props);
|
|
|
|
this.state = {
|
|
isVizPickerOpen: false,
|
|
searchQuery: '',
|
|
};
|
|
}
|
|
|
|
getPanelDefaultOptions = () => {
|
|
const { panel, plugin } = this.props;
|
|
|
|
if (plugin.exports.PanelDefaults) {
|
|
return panel.getOptions(plugin.exports.PanelDefaults.options);
|
|
}
|
|
|
|
return panel.getOptions(plugin.exports.PanelDefaults);
|
|
};
|
|
|
|
renderPanelOptions() {
|
|
const { plugin, angularPanel } = this.props;
|
|
const { PanelOptions } = plugin.exports;
|
|
|
|
if (angularPanel) {
|
|
return <div ref={element => (this.element = element)} />;
|
|
}
|
|
|
|
return (
|
|
<PanelOptionSection>
|
|
{PanelOptions ? (
|
|
<PanelOptions options={this.getPanelDefaultOptions()} onChange={this.onPanelOptionsChanged} />
|
|
) : (
|
|
<p>Visualization has no options</p>
|
|
)}
|
|
</PanelOptionSection>
|
|
);
|
|
}
|
|
|
|
componentDidMount() {
|
|
if (this.shouldLoadAngularOptions()) {
|
|
this.loadAngularOptions();
|
|
}
|
|
}
|
|
|
|
componentDidUpdate(prevProps: Props) {
|
|
if (this.props.plugin !== prevProps.plugin) {
|
|
this.cleanUpAngularOptions();
|
|
}
|
|
|
|
if (this.shouldLoadAngularOptions()) {
|
|
this.loadAngularOptions();
|
|
}
|
|
}
|
|
|
|
shouldLoadAngularOptions() {
|
|
return this.props.angularPanel && this.element && !this.angularOptions;
|
|
}
|
|
|
|
loadAngularOptions() {
|
|
const { angularPanel } = this.props;
|
|
|
|
const scope = angularPanel.getScope();
|
|
|
|
// When full page reloading in edit mode the angular panel has on fully compiled & instantiated yet
|
|
if (!scope.$$childHead) {
|
|
setTimeout(() => {
|
|
this.forceUpdate();
|
|
});
|
|
return;
|
|
}
|
|
|
|
const panelCtrl = scope.$$childHead.ctrl;
|
|
|
|
let template = '';
|
|
for (let i = 0; i < panelCtrl.editorTabs.length; i++) {
|
|
template +=
|
|
`
|
|
<div class="panel-option-section" ng-cloak>` +
|
|
(i > 0 ? `<div class="panel-option-section__header">{{ctrl.editorTabs[${i}].title}}</div>` : '') +
|
|
`<div class="panel-option-section__body">
|
|
<panel-editor-tab editor-tab="ctrl.editorTabs[${i}]" ctrl="ctrl"></panel-editor-tab>
|
|
</div>
|
|
</div>
|
|
`;
|
|
}
|
|
|
|
const loader = getAngularLoader();
|
|
const scopeProps = { ctrl: panelCtrl };
|
|
|
|
this.angularOptions = loader.load(this.element, scopeProps, template);
|
|
}
|
|
|
|
componentWillUnmount() {
|
|
this.cleanUpAngularOptions();
|
|
}
|
|
|
|
cleanUpAngularOptions() {
|
|
if (this.angularOptions) {
|
|
this.angularOptions.destroy();
|
|
this.angularOptions = null;
|
|
}
|
|
}
|
|
|
|
onPanelOptionsChanged = (options: any) => {
|
|
this.props.panel.updateOptions(options);
|
|
this.forceUpdate();
|
|
};
|
|
|
|
onOpenVizPicker = () => {
|
|
this.setState({ isVizPickerOpen: true });
|
|
};
|
|
|
|
onCloseVizPicker = () => {
|
|
this.setState({ isVizPickerOpen: false });
|
|
};
|
|
|
|
onSearchQueryChange = evt => {
|
|
const value = evt.target.value;
|
|
this.setState({
|
|
searchQuery: value,
|
|
});
|
|
};
|
|
|
|
renderToolbar = (): JSX.Element => {
|
|
const { plugin } = this.props;
|
|
const { searchQuery } = this.state;
|
|
|
|
if (this.state.isVizPickerOpen) {
|
|
return (
|
|
<>
|
|
<label className="gf-form--has-input-icon">
|
|
<input
|
|
type="text"
|
|
className="gf-form-input width-13"
|
|
placeholder=""
|
|
onChange={this.onSearchQueryChange}
|
|
value={searchQuery}
|
|
ref={elem => elem && elem.focus()}
|
|
/>
|
|
<i className="gf-form-input-icon fa fa-search" />
|
|
</label>
|
|
<div className="flex-grow" />
|
|
<button className="btn btn-link" onClick={this.onCloseVizPicker}>
|
|
<i className="fa fa-chevron-up" />
|
|
</button>
|
|
</>
|
|
);
|
|
} else {
|
|
return (
|
|
<div className="toolbar__main" onClick={this.onOpenVizPicker}>
|
|
<img className="toolbar__main-image" src={plugin.info.logos.small} />
|
|
<div className="toolbar__main-name">{plugin.name}</div>
|
|
<i className="fa fa-caret-down" />
|
|
</div>
|
|
);
|
|
}
|
|
};
|
|
|
|
onTypeChanged = (plugin: PanelPlugin) => {
|
|
if (plugin.id === this.props.plugin.id) {
|
|
this.setState({ isVizPickerOpen: false });
|
|
} else {
|
|
this.props.onTypeChanged(plugin);
|
|
}
|
|
};
|
|
|
|
render() {
|
|
const { plugin } = this.props;
|
|
const { isVizPickerOpen, searchQuery } = this.state;
|
|
|
|
return (
|
|
<EditorTabBody heading="Visualization" renderToolbar={this.renderToolbar}>
|
|
<>
|
|
<FadeIn in={isVizPickerOpen} duration={200} unmountOnExit={true}>
|
|
<VizTypePicker
|
|
current={plugin}
|
|
onTypeChanged={this.onTypeChanged}
|
|
searchQuery={searchQuery}
|
|
onClose={this.onCloseVizPicker}
|
|
/>
|
|
</FadeIn>
|
|
{this.renderPanelOptions()}
|
|
</>
|
|
</EditorTabBody>
|
|
);
|
|
}
|
|
}
|