2020-02-28 11:04:40 +01:00
|
|
|
import React, { PureComponent } from 'react';
|
2021-04-28 15:40:01 +03:00
|
|
|
import { connect, ConnectedProps } from 'react-redux';
|
2020-02-28 11:04:40 +01:00
|
|
|
|
|
|
|
|
import { PanelPlugin, PanelPluginMeta } from '@grafana/data';
|
2022-04-22 14:33:13 +01:00
|
|
|
import { AngularComponent, getAngularLoader } from '@grafana/runtime';
|
|
|
|
|
import { PanelCtrl } from 'app/angular/panel/panel_ctrl';
|
2021-10-13 08:53:36 +02:00
|
|
|
import { changePanelPlugin } from 'app/features/panel/state/actions';
|
2022-04-22 14:33:13 +01:00
|
|
|
import { getPanelStateForModel } from 'app/features/panel/state/selectors';
|
2020-02-28 11:04:40 +01:00
|
|
|
import { StoreState } from 'app/types';
|
2022-04-22 14:33:13 +01:00
|
|
|
|
|
|
|
|
import { PanelModel, DashboardModel } from '../../state';
|
|
|
|
|
|
2020-04-20 08:47:25 +02:00
|
|
|
import { getSectionOpenState, saveSectionOpenState } from './state/utils';
|
2020-02-28 11:04:40 +01:00
|
|
|
|
|
|
|
|
interface OwnProps {
|
|
|
|
|
panel: PanelModel;
|
|
|
|
|
dashboard: DashboardModel;
|
|
|
|
|
plugin: PanelPlugin;
|
|
|
|
|
}
|
|
|
|
|
|
2021-04-28 15:40:01 +03:00
|
|
|
const mapStateToProps = (state: StoreState, props: OwnProps) => ({
|
2021-10-13 08:53:36 +02:00
|
|
|
angularPanelComponent: getPanelStateForModel(state, props.panel)?.angularComponent,
|
2021-04-28 15:40:01 +03:00
|
|
|
});
|
2020-02-28 11:04:40 +01:00
|
|
|
|
2021-04-28 15:40:01 +03:00
|
|
|
const mapDispatchToProps = { changePanelPlugin };
|
2020-02-28 11:04:40 +01:00
|
|
|
|
2021-04-28 15:40:01 +03:00
|
|
|
const connector = connect(mapStateToProps, mapDispatchToProps);
|
|
|
|
|
type Props = ConnectedProps<typeof connector> & OwnProps;
|
2020-02-28 11:04:40 +01:00
|
|
|
|
|
|
|
|
export class AngularPanelOptionsUnconnected extends PureComponent<Props> {
|
Chore: Fix all Typescript strict null errors (#26204)
* Chore: Fix typescript strict null errors
* Added new limit
* Fixed ts issue
* fixed tests
* trying to fix type inference
* Fixing more ts errors
* Revert tsconfig option
* Fix
* Fixed code
* More fixes
* fix tests
* Updated snapshot
* Chore: More ts strict null fixes
* More fixes in some really messed up azure config components
* More fixes, current count: 441
* 419
* More fixes
* Fixed invalid initial state in explore
* Fixing tests
* Fixed tests
* Explore fix
* More fixes
* Progress
* Sub 300
* Now at 218
* Progress
* Update
* Progress
* Updated tests
* at 159
* fixed tests
* Progress
* YAy blow 100! at 94
* 10,9,8,7,6,5,4,3,2,1... lift off
* Fixed tests
* Fixed more type errors
Co-authored-by: Ryan McKinley <ryantxu@gmail.com>
2020-07-10 12:46:59 +02:00
|
|
|
element?: HTMLElement | null;
|
|
|
|
|
angularOptions?: AngularComponent | null;
|
2020-02-28 11:04:40 +01:00
|
|
|
|
|
|
|
|
constructor(props: Props) {
|
|
|
|
|
super(props);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
componentDidMount() {
|
|
|
|
|
this.loadAngularOptions();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
componentDidUpdate(prevProps: Props) {
|
2021-10-13 08:53:36 +02:00
|
|
|
if (
|
|
|
|
|
this.props.plugin !== prevProps.plugin ||
|
|
|
|
|
this.props.angularPanelComponent !== prevProps.angularPanelComponent
|
|
|
|
|
) {
|
2020-02-28 11:04:40 +01:00
|
|
|
this.cleanUpAngularOptions();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
this.loadAngularOptions();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
componentWillUnmount() {
|
|
|
|
|
this.cleanUpAngularOptions();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
cleanUpAngularOptions() {
|
|
|
|
|
if (this.angularOptions) {
|
|
|
|
|
this.angularOptions.destroy();
|
|
|
|
|
this.angularOptions = null;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
loadAngularOptions() {
|
|
|
|
|
const { panel, angularPanelComponent, changePanelPlugin } = this.props;
|
|
|
|
|
|
|
|
|
|
if (!this.element || !angularPanelComponent || this.angularOptions) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const scope = angularPanelComponent.getScope();
|
|
|
|
|
|
2021-04-01 08:42:57 -07:00
|
|
|
// When full page reloading in edit mode the angular panel has on fully compiled and instantiated yet
|
2020-02-28 11:04:40 +01:00
|
|
|
if (!scope.$$childHead) {
|
|
|
|
|
setTimeout(() => {
|
|
|
|
|
this.forceUpdate();
|
|
|
|
|
});
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const panelCtrl: PanelCtrl = scope.$$childHead.ctrl;
|
|
|
|
|
panelCtrl.initEditMode();
|
|
|
|
|
panelCtrl.onPluginTypeChange = (plugin: PanelPluginMeta) => {
|
2021-10-25 13:55:06 +02:00
|
|
|
changePanelPlugin({ panel, pluginId: plugin.id });
|
2020-02-28 11:04:40 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
let template = '';
|
|
|
|
|
for (let i = 0; i < panelCtrl.editorTabs.length; i++) {
|
2020-04-20 08:47:25 +02:00
|
|
|
const tab = panelCtrl.editorTabs[i];
|
|
|
|
|
tab.isOpen = getSectionOpenState(tab.title, i === 0);
|
|
|
|
|
|
|
|
|
|
template += `
|
2021-04-06 14:51:35 +02:00
|
|
|
<div class="panel-options-group" ng-cloak>
|
2020-04-20 08:47:25 +02:00
|
|
|
<div class="panel-options-group__header" ng-click="toggleOptionGroup(${i})" aria-label="${tab.title} section">
|
|
|
|
|
<div class="panel-options-group__icon">
|
|
|
|
|
<icon name="ctrl.editorTabs[${i}].isOpen ? 'angle-down' : 'angle-right'"></icon>
|
|
|
|
|
</div>
|
|
|
|
|
<div class="panel-options-group__title">${tab.title}</div>
|
|
|
|
|
</div>
|
|
|
|
|
<div class="panel-options-group__body" ng-if="ctrl.editorTabs[${i}].isOpen">
|
2020-02-28 11:04:40 +01:00
|
|
|
<panel-editor-tab editor-tab="ctrl.editorTabs[${i}]" ctrl="ctrl"></panel-editor-tab>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
`;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const loader = getAngularLoader();
|
2020-04-20 08:47:25 +02:00
|
|
|
const scopeProps = {
|
|
|
|
|
ctrl: panelCtrl,
|
|
|
|
|
toggleOptionGroup: (index: number) => {
|
|
|
|
|
const tab = panelCtrl.editorTabs[index];
|
|
|
|
|
tab.isOpen = !tab.isOpen;
|
2022-11-11 16:29:49 +00:00
|
|
|
saveSectionOpenState(tab.title, Boolean(tab.isOpen));
|
2020-04-20 08:47:25 +02:00
|
|
|
},
|
|
|
|
|
};
|
2020-02-28 11:04:40 +01:00
|
|
|
|
|
|
|
|
this.angularOptions = loader.load(this.element, scopeProps, template);
|
2021-03-25 08:33:13 +01:00
|
|
|
this.angularOptions.digest();
|
2020-02-28 11:04:40 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
render() {
|
2021-01-20 07:59:48 +01:00
|
|
|
return <div ref={(elem) => (this.element = elem)} />;
|
2020-02-28 11:04:40 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export const AngularPanelOptions = connect(mapStateToProps, mapDispatchToProps)(AngularPanelOptionsUnconnected);
|