EditorTabBody.tsx 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138
  1. // Libraries
  2. import React, { PureComponent } from 'react';
  3. // Components
  4. import { CustomScrollbar, PanelOptionsGroup } from '@grafana/ui';
  5. import { FadeIn } from 'app/core/components/Animations/FadeIn';
  6. interface Props {
  7. children: JSX.Element;
  8. heading: string;
  9. renderToolbar?: () => JSX.Element;
  10. toolbarItems?: EditorToolbarView[];
  11. scrollTop?: number;
  12. setScrollTop?: (value: React.MouseEvent<HTMLElement>) => void;
  13. }
  14. export interface EditorToolbarView {
  15. title?: string;
  16. heading?: string;
  17. icon?: string;
  18. disabled?: boolean;
  19. onClick?: () => void;
  20. render?: () => JSX.Element;
  21. action?: () => void;
  22. btnType?: 'danger';
  23. }
  24. interface State {
  25. openView?: EditorToolbarView;
  26. isOpen: boolean;
  27. fadeIn: boolean;
  28. }
  29. export class EditorTabBody extends PureComponent<Props, State> {
  30. static defaultProps = {
  31. toolbarItems: [],
  32. };
  33. constructor(props) {
  34. super(props);
  35. this.state = {
  36. openView: null,
  37. fadeIn: false,
  38. isOpen: false,
  39. };
  40. }
  41. componentDidMount() {
  42. this.setState({ fadeIn: true });
  43. }
  44. onToggleToolBarView = (item: EditorToolbarView) => {
  45. this.setState({
  46. openView: item,
  47. isOpen: this.state.openView !== item || !this.state.isOpen,
  48. });
  49. };
  50. onCloseOpenView = () => {
  51. this.setState({ isOpen: false });
  52. };
  53. static getDerivedStateFromProps(props, state) {
  54. if (state.openView) {
  55. const activeToolbarItem = props.toolbarItems.find(
  56. item => item.title === state.openView.title && item.icon === state.openView.icon
  57. );
  58. if (activeToolbarItem) {
  59. return {
  60. ...state,
  61. openView: activeToolbarItem,
  62. };
  63. }
  64. }
  65. return state;
  66. }
  67. renderButton(view: EditorToolbarView) {
  68. const onClick = () => {
  69. if (view.onClick) {
  70. view.onClick();
  71. }
  72. if (view.render) {
  73. this.onToggleToolBarView(view);
  74. }
  75. };
  76. return (
  77. <div className="nav-buttons" key={view.title + view.icon}>
  78. <button className="btn navbar-button" onClick={onClick} disabled={view.disabled}>
  79. {view.icon && <i className={view.icon} />} {view.title}
  80. </button>
  81. </div>
  82. );
  83. }
  84. renderOpenView(view: EditorToolbarView) {
  85. return (
  86. <PanelOptionsGroup title={view.title || view.heading} onClose={this.onCloseOpenView}>
  87. {view.render()}
  88. </PanelOptionsGroup>
  89. );
  90. }
  91. render() {
  92. const { children, renderToolbar, heading, toolbarItems, scrollTop, setScrollTop } = this.props;
  93. const { openView, fadeIn, isOpen } = this.state;
  94. return (
  95. <>
  96. <div className="toolbar">
  97. <div className="toolbar__heading">{heading}</div>
  98. {renderToolbar && renderToolbar()}
  99. {toolbarItems.length > 0 && (
  100. <>
  101. <div className="gf-form--grow" />
  102. {toolbarItems.map(item => this.renderButton(item))}
  103. </>
  104. )}
  105. </div>
  106. <div className="panel-editor__scroll">
  107. <CustomScrollbar autoHide={false} scrollTop={scrollTop} setScrollTop={setScrollTop}>
  108. <div className="panel-editor__content">
  109. <FadeIn in={isOpen} duration={200} unmountOnExit={true}>
  110. {openView && this.renderOpenView(openView)}
  111. </FadeIn>
  112. <FadeIn in={fadeIn} duration={50}>
  113. {children}
  114. </FadeIn>
  115. </div>
  116. </CustomScrollbar>
  117. </div>
  118. </>
  119. );
  120. }
  121. }