@perses-dev/dashboards 0.52.0-beta.5 → 0.52.0-rc.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (108) hide show
  1. package/dist/cjs/components/GridLayout/GridLayout.js +78 -126
  2. package/dist/cjs/components/GridLayout/Row.js +150 -0
  3. package/dist/cjs/components/GridLayout/index.js +1 -0
  4. package/dist/cjs/components/LeaveDialog/LeaveDialog.js +44 -0
  5. package/dist/cjs/components/LeaveDialog/index.js +30 -0
  6. package/dist/cjs/components/Panel/Panel.js +5 -1
  7. package/dist/cjs/components/Panel/PanelActions.js +3 -3
  8. package/dist/cjs/components/Panel/PanelHeader.js +32 -15
  9. package/dist/cjs/components/PanelDrawer/PanelDrawer.js +58 -21
  10. package/dist/cjs/components/PanelGroupDialog/PanelGroupDialog.js +3 -0
  11. package/dist/cjs/components/PanelGroupDialog/PanelGroupEditorForm.js +35 -15
  12. package/dist/cjs/components/index.js +1 -0
  13. package/dist/cjs/context/DashboardProvider/DashboardProvider.js +7 -8
  14. package/dist/cjs/context/DashboardProvider/DashboardProviderWithQueryParams.js +3 -3
  15. package/dist/cjs/context/DashboardProvider/panel-editor-slice.js +1 -0
  16. package/dist/cjs/context/DashboardProvider/panel-group-editor-slice.js +6 -2
  17. package/dist/cjs/context/DashboardProvider/panel-group-slice.js +1 -0
  18. package/dist/cjs/context/DashboardProvider/view-panel-slice.js +10 -3
  19. package/dist/cjs/context/VariableProvider/VariableProvider.js +1 -1
  20. package/dist/cjs/context/useDashboard.js +5 -4
  21. package/dist/cjs/views/ViewDashboard/DashboardApp.js +7 -3
  22. package/dist/cjs/views/ViewDashboard/ViewDashboard.js +9 -8
  23. package/dist/components/DashboardToolbar/DashboardToolbar.d.ts +2 -2
  24. package/dist/components/DashboardToolbar/DashboardToolbar.d.ts.map +1 -1
  25. package/dist/components/DashboardToolbar/DashboardToolbar.js.map +1 -1
  26. package/dist/components/GridLayout/GridLayout.d.ts +8 -0
  27. package/dist/components/GridLayout/GridLayout.d.ts.map +1 -1
  28. package/dist/components/GridLayout/GridLayout.js +72 -126
  29. package/dist/components/GridLayout/GridLayout.js.map +1 -1
  30. package/dist/components/GridLayout/Row.d.ts +17 -0
  31. package/dist/components/GridLayout/Row.d.ts.map +1 -0
  32. package/dist/components/GridLayout/Row.js +142 -0
  33. package/dist/components/GridLayout/Row.js.map +1 -0
  34. package/dist/components/GridLayout/index.d.ts +1 -0
  35. package/dist/components/GridLayout/index.d.ts.map +1 -1
  36. package/dist/components/GridLayout/index.js +1 -0
  37. package/dist/components/GridLayout/index.js.map +1 -1
  38. package/dist/components/LeaveDialog/LeaveDialog.d.ts +7 -0
  39. package/dist/components/LeaveDialog/LeaveDialog.d.ts.map +1 -0
  40. package/dist/components/LeaveDialog/LeaveDialog.js +36 -0
  41. package/dist/components/LeaveDialog/LeaveDialog.js.map +1 -0
  42. package/dist/components/LeaveDialog/index.d.ts +2 -0
  43. package/dist/components/LeaveDialog/index.d.ts.map +1 -0
  44. package/dist/components/LeaveDialog/index.js +15 -0
  45. package/dist/components/LeaveDialog/index.js.map +1 -0
  46. package/dist/components/Panel/Panel.d.ts +5 -0
  47. package/dist/components/Panel/Panel.d.ts.map +1 -1
  48. package/dist/components/Panel/Panel.js +5 -1
  49. package/dist/components/Panel/Panel.js.map +1 -1
  50. package/dist/components/Panel/PanelActions.d.ts +2 -0
  51. package/dist/components/Panel/PanelActions.d.ts.map +1 -1
  52. package/dist/components/Panel/PanelActions.js +3 -3
  53. package/dist/components/Panel/PanelActions.js.map +1 -1
  54. package/dist/components/Panel/PanelHeader.d.ts +6 -1
  55. package/dist/components/Panel/PanelHeader.d.ts.map +1 -1
  56. package/dist/components/Panel/PanelHeader.js +33 -16
  57. package/dist/components/Panel/PanelHeader.js.map +1 -1
  58. package/dist/components/PanelDrawer/PanelDrawer.d.ts.map +1 -1
  59. package/dist/components/PanelDrawer/PanelDrawer.js +59 -22
  60. package/dist/components/PanelDrawer/PanelDrawer.js.map +1 -1
  61. package/dist/components/PanelGroupDialog/PanelGroupDialog.d.ts.map +1 -1
  62. package/dist/components/PanelGroupDialog/PanelGroupDialog.js +3 -0
  63. package/dist/components/PanelGroupDialog/PanelGroupDialog.js.map +1 -1
  64. package/dist/components/PanelGroupDialog/PanelGroupEditorForm.d.ts +1 -0
  65. package/dist/components/PanelGroupDialog/PanelGroupEditorForm.d.ts.map +1 -1
  66. package/dist/components/PanelGroupDialog/PanelGroupEditorForm.js +36 -16
  67. package/dist/components/PanelGroupDialog/PanelGroupEditorForm.js.map +1 -1
  68. package/dist/components/index.d.ts +1 -0
  69. package/dist/components/index.d.ts.map +1 -1
  70. package/dist/components/index.js +1 -0
  71. package/dist/components/index.js.map +1 -1
  72. package/dist/context/DashboardProvider/DashboardProvider.d.ts +5 -5
  73. package/dist/context/DashboardProvider/DashboardProvider.d.ts.map +1 -1
  74. package/dist/context/DashboardProvider/DashboardProvider.js +7 -8
  75. package/dist/context/DashboardProvider/DashboardProvider.js.map +1 -1
  76. package/dist/context/DashboardProvider/DashboardProviderWithQueryParams.js +4 -4
  77. package/dist/context/DashboardProvider/DashboardProviderWithQueryParams.js.map +1 -1
  78. package/dist/context/DashboardProvider/common.d.ts +1 -1
  79. package/dist/context/DashboardProvider/common.d.ts.map +1 -1
  80. package/dist/context/DashboardProvider/common.js.map +1 -1
  81. package/dist/context/DashboardProvider/panel-editor-slice.d.ts +1 -0
  82. package/dist/context/DashboardProvider/panel-editor-slice.d.ts.map +1 -1
  83. package/dist/context/DashboardProvider/panel-editor-slice.js +1 -0
  84. package/dist/context/DashboardProvider/panel-editor-slice.js.map +1 -1
  85. package/dist/context/DashboardProvider/panel-group-editor-slice.d.ts +1 -0
  86. package/dist/context/DashboardProvider/panel-group-editor-slice.d.ts.map +1 -1
  87. package/dist/context/DashboardProvider/panel-group-editor-slice.js +6 -2
  88. package/dist/context/DashboardProvider/panel-group-editor-slice.js.map +1 -1
  89. package/dist/context/DashboardProvider/panel-group-slice.d.ts +3 -0
  90. package/dist/context/DashboardProvider/panel-group-slice.d.ts.map +1 -1
  91. package/dist/context/DashboardProvider/panel-group-slice.js +1 -0
  92. package/dist/context/DashboardProvider/panel-group-slice.js.map +1 -1
  93. package/dist/context/DashboardProvider/view-panel-slice.d.ts +6 -2
  94. package/dist/context/DashboardProvider/view-panel-slice.d.ts.map +1 -1
  95. package/dist/context/DashboardProvider/view-panel-slice.js +10 -3
  96. package/dist/context/DashboardProvider/view-panel-slice.js.map +1 -1
  97. package/dist/context/VariableProvider/VariableProvider.js +1 -1
  98. package/dist/context/VariableProvider/VariableProvider.js.map +1 -1
  99. package/dist/context/useDashboard.js +5 -4
  100. package/dist/context/useDashboard.js.map +1 -1
  101. package/dist/views/ViewDashboard/DashboardApp.d.ts +7 -6
  102. package/dist/views/ViewDashboard/DashboardApp.d.ts.map +1 -1
  103. package/dist/views/ViewDashboard/DashboardApp.js +8 -4
  104. package/dist/views/ViewDashboard/DashboardApp.js.map +1 -1
  105. package/dist/views/ViewDashboard/ViewDashboard.d.ts.map +1 -1
  106. package/dist/views/ViewDashboard/ViewDashboard.js +9 -8
  107. package/dist/views/ViewDashboard/ViewDashboard.js.map +1 -1
  108. package/package.json +5 -5
@@ -23,6 +23,7 @@ Object.defineProperty(exports, "PanelDrawer", {
23
23
  const _jsxruntime = require("react/jsx-runtime");
24
24
  const _react = require("react");
25
25
  const _components = require("@perses-dev/components");
26
+ const _pluginsystem = require("@perses-dev/plugin-system");
26
27
  const _context = require("../../context");
27
28
  const _PanelEditorForm = require("./PanelEditorForm");
28
29
  const PanelDrawer = ()=>{
@@ -30,42 +31,78 @@ const PanelDrawer = ()=>{
30
31
  // When the user clicks close, start closing but don't call the store yet to keep values stable during animtation
31
32
  const [isClosing, setIsClosing] = (0, _react.useState)(false);
32
33
  // Drawer is open if we have a model and we're not transitioning out
33
- const isOpen = panelEditor !== undefined && isClosing === false;
34
- function handleSave(values) {
34
+ const isOpen = panelEditor !== undefined && !isClosing;
35
+ const handleSave = (0, _react.useCallback)((values)=>{
35
36
  // This shouldn't happen since we don't render the submit button until we have a model, but check to make TS happy
36
37
  if (panelEditor === undefined || values === undefined) {
37
38
  throw new Error('Cannot apply changes');
38
39
  }
39
40
  panelEditor.applyChanges(values);
40
41
  setIsClosing(true);
41
- }
42
+ }, [
43
+ panelEditor
44
+ ]);
42
45
  const handleClose = ()=>{
43
46
  setIsClosing(true);
44
47
  };
45
48
  // Don't call closeDrawer on the store until the Drawer has completely transitioned out and reset close state
46
- const handleExited = ()=>{
49
+ const handleExited = (0, _react.useCallback)(()=>{
47
50
  panelEditor?.close();
48
51
  setIsClosing(false);
49
- };
52
+ }, [
53
+ panelEditor
54
+ ]);
50
55
  // Disables closing on click out. This is a quick-win solution to avoid losing draft changes.
51
56
  // -> TODO find a way to enable closing by clicking-out in edit view, with a discard confirmation modal popping up
52
57
  const handleClickOut = ()=>{
53
58
  /* do nothing */ };
54
- return /*#__PURE__*/ (0, _jsxruntime.jsx)(_components.Drawer, {
55
- isOpen: isOpen,
56
- onClose: handleClickOut,
57
- SlideProps: {
58
- onExited: handleExited
59
- },
60
- "data-testid": "panel-editor",
61
- children: panelEditor && /*#__PURE__*/ (0, _jsxruntime.jsx)(_components.ErrorBoundary, {
62
- FallbackComponent: _components.ErrorAlert,
63
- children: /*#__PURE__*/ (0, _jsxruntime.jsx)(_PanelEditorForm.PanelEditorForm, {
64
- initialAction: panelEditor.mode,
65
- initialValues: panelEditor.initialValues,
66
- onSave: handleSave,
67
- onClose: handleClose
59
+ const drawer = (0, _react.useMemo)(()=>{
60
+ return /*#__PURE__*/ (0, _jsxruntime.jsx)(_components.Drawer, {
61
+ isOpen: isOpen,
62
+ onClose: handleClickOut,
63
+ SlideProps: {
64
+ onExited: handleExited
65
+ },
66
+ "data-testid": "panel-editor",
67
+ children: panelEditor && /*#__PURE__*/ (0, _jsxruntime.jsx)(_components.ErrorBoundary, {
68
+ FallbackComponent: _components.ErrorAlert,
69
+ children: /*#__PURE__*/ (0, _jsxruntime.jsx)(_PanelEditorForm.PanelEditorForm, {
70
+ initialAction: panelEditor.mode,
71
+ initialValues: panelEditor.initialValues,
72
+ onSave: handleSave,
73
+ onClose: handleClose
74
+ })
68
75
  })
69
- })
70
- });
76
+ });
77
+ }, [
78
+ handleExited,
79
+ handleSave,
80
+ isOpen,
81
+ panelEditor
82
+ ]);
83
+ // If the panel editor is using a repeat variable, we need to wrap the drawer in a VariableContext.Provider
84
+ if (panelEditor?.panelGroupItemId?.repeatVariable) {
85
+ return /*#__PURE__*/ (0, _jsxruntime.jsx)(RepeatVariableWrapper, {
86
+ repeatVariable: panelEditor.panelGroupItemId.repeatVariable,
87
+ children: drawer
88
+ });
89
+ }
90
+ return drawer;
71
91
  };
92
+ // Wraps the drawer in a VariableContext.Provider to provide the repeat variable value
93
+ // This is necessary for previewing panels that use repeat variables and query editor
94
+ function RepeatVariableWrapper({ repeatVariable, children }) {
95
+ const variables = (0, _pluginsystem.useVariableValues)();
96
+ return /*#__PURE__*/ (0, _jsxruntime.jsx)(_pluginsystem.VariableContext.Provider, {
97
+ value: {
98
+ state: {
99
+ ...variables,
100
+ [repeatVariable[0]]: {
101
+ value: repeatVariable[1],
102
+ loading: false
103
+ }
104
+ }
105
+ },
106
+ children: children
107
+ });
108
+ }
@@ -24,6 +24,7 @@ const _jsxruntime = require("react/jsx-runtime");
24
24
  const _material = require("@mui/material");
25
25
  const _Close = /*#__PURE__*/ _interop_require_default(require("mdi-material-ui/Close"));
26
26
  const _react = require("react");
27
+ const _pluginsystem = require("@perses-dev/plugin-system");
27
28
  const _context = require("../../context");
28
29
  const _PanelGroupEditorForm = require("./PanelGroupEditorForm");
29
30
  function _interop_require_default(obj) {
@@ -33,6 +34,7 @@ function _interop_require_default(obj) {
33
34
  }
34
35
  function PanelGroupDialog() {
35
36
  const panelGroupEditor = (0, _context.usePanelGroupEditor)();
37
+ const variables = (0, _pluginsystem.useVariableValues)();
36
38
  // When the user clicks close, start closing but don't call the store yet to keep values stable during animtation
37
39
  const [isClosing, setIsClosing] = (0, _react.useState)(false);
38
40
  const handleClose = ()=>setIsClosing(true);
@@ -81,6 +83,7 @@ function PanelGroupDialog() {
81
83
  },
82
84
  children: /*#__PURE__*/ (0, _jsxruntime.jsx)(_PanelGroupEditorForm.PanelGroupEditorForm, {
83
85
  initialValues: panelGroupEditor.initialValues,
86
+ variables: Object.keys(variables),
84
87
  onSubmit: handleSubmit
85
88
  })
86
89
  }),
@@ -32,18 +32,16 @@ const _jsxruntime = require("react/jsx-runtime");
32
32
  const _react = require("react");
33
33
  const _material = require("@mui/material");
34
34
  function PanelGroupEditorForm(props) {
35
- const { initialValues, onSubmit } = props;
35
+ const { initialValues, variables, onSubmit } = props;
36
36
  const [title, setTitle] = (0, _react.useState)(initialValues.title);
37
37
  const [isCollapsed, setIsCollapsed] = (0, _react.useState)(initialValues.isCollapsed);
38
- const handleCollapsedChange = (e)=>{
39
- const next = e.target.value;
40
- setIsCollapsed(next === 'Closed');
41
- };
38
+ const [repeatVariable, setRepeatVariable] = (0, _react.useState)(initialValues.repeatVariable);
42
39
  const handleSubmit = (e)=>{
43
40
  e.preventDefault();
44
41
  onSubmit({
45
42
  title,
46
- isCollapsed
43
+ isCollapsed,
44
+ repeatVariable
47
45
  });
48
46
  };
49
47
  return /*#__PURE__*/ (0, _jsxruntime.jsxs)("form", {
@@ -58,25 +56,21 @@ function PanelGroupEditorForm(props) {
58
56
  label: "Name",
59
57
  variant: "outlined",
60
58
  value: title,
61
- onChange: (e)=>setTitle(e.target.value)
59
+ onChange: (e)=>setTitle(e.target.value),
60
+ "data-testid": "panel-group-editor-name"
62
61
  })
63
62
  }),
64
63
  /*#__PURE__*/ (0, _jsxruntime.jsxs)(_material.FormControl, {
65
64
  fullWidth: true,
66
65
  margin: "normal",
67
66
  children: [
68
- /*#__PURE__*/ (0, _jsxruntime.jsx)(_material.InputLabel, {
69
- id: "select-collapse-state",
70
- children: "Collapse State"
71
- }),
72
- /*#__PURE__*/ (0, _jsxruntime.jsxs)(_material.Select, {
67
+ /*#__PURE__*/ (0, _jsxruntime.jsxs)(_material.TextField, {
68
+ select: true,
73
69
  required: true,
74
- displayEmpty: true,
75
- labelId: "select-collapse-state",
76
70
  label: "Collapse State",
77
71
  size: "small",
78
72
  value: isCollapsed ? 'Closed' : 'Open',
79
- onChange: handleCollapsedChange,
73
+ onChange: (e)=>setIsCollapsed(e.target.value === 'Closed'),
80
74
  children: [
81
75
  /*#__PURE__*/ (0, _jsxruntime.jsx)(_material.MenuItem, {
82
76
  value: "Open",
@@ -87,6 +81,32 @@ function PanelGroupEditorForm(props) {
87
81
  children: "Closed"
88
82
  })
89
83
  ]
84
+ }),
85
+ /*#__PURE__*/ (0, _jsxruntime.jsx)(_material.FormControl, {
86
+ fullWidth: true,
87
+ margin: "normal",
88
+ children: /*#__PURE__*/ (0, _jsxruntime.jsxs)(_material.TextField, {
89
+ select: true,
90
+ label: "Repeat Variable",
91
+ variant: "outlined",
92
+ value: repeatVariable ?? '',
93
+ onChange: (e)=>setRepeatVariable(e.target.value === '' ? undefined : e.target.value),
94
+ children: [
95
+ /*#__PURE__*/ (0, _jsxruntime.jsx)(_material.MenuItem, {
96
+ value: "",
97
+ children: /*#__PURE__*/ (0, _jsxruntime.jsx)(_material.Typography, {
98
+ sx: {
99
+ fontStyle: 'italic'
100
+ },
101
+ children: "None"
102
+ })
103
+ }),
104
+ variables?.sort((a, b)=>a.localeCompare(b)).map((variable)=>/*#__PURE__*/ (0, _jsxruntime.jsx)(_material.MenuItem, {
105
+ value: variable,
106
+ children: variable
107
+ }, variable))
108
+ ]
109
+ })
90
110
  })
91
111
  ]
92
112
  })
@@ -28,6 +28,7 @@ _export_star(require("./EditJsonButton"), exports);
28
28
  _export_star(require("./EditJsonDialog"), exports);
29
29
  _export_star(require("./EmptyDashboard"), exports);
30
30
  _export_star(require("./GridLayout"), exports);
31
+ _export_star(require("./LeaveDialog"), exports);
31
32
  _export_star(require("./Panel"), exports);
32
33
  _export_star(require("./PanelDrawer"), exports);
33
34
  _export_star(require("./PanelGroupDialog"), exports);
@@ -61,14 +61,15 @@ function useDashboardStore(selector) {
61
61
  return (0, _traditional.useStoreWithEqualityFn)(store, selector, _shallow.shallow);
62
62
  }
63
63
  function DashboardProvider(props) {
64
+ // Prevent calling createDashboardStore every time it rerenders
64
65
  const createDashboardStore = (0, _react.useCallback)(initStore, [
65
66
  props
66
67
  ]);
68
+ const [store] = (0, _react.useState)(createDashboardStore(props));
67
69
  // load plugin to retrieve initial spec if default panel kind is defined
68
70
  const { defaultPluginKinds } = (0, _pluginsystem.usePluginRegistry)();
69
71
  const defaultPanelKind = defaultPluginKinds?.['Panel'] ?? '';
70
72
  const { data: plugin } = (0, _pluginsystem.usePlugin)('Panel', defaultPanelKind);
71
- const [store] = (0, _react.useState)(createDashboardStore(props)); // prevent calling createDashboardStore every time it rerenders
72
73
  (0, _react.useEffect)(()=>{
73
74
  if (plugin === undefined) return;
74
75
  const defaultPanelSpec = plugin.createInitialOptions ? plugin.createInitialOptions() : {};
@@ -90,12 +91,8 @@ function DashboardProvider(props) {
90
91
  }
91
92
  function initStore(props) {
92
93
  const { initialState: { dashboardResource, isEditMode, viewPanelRef, setViewPanelRef } } = props;
93
- const { kind, metadata, spec: { display, duration, refreshInterval = _core.DEFAULT_REFRESH_INTERVAL, datasources } } = dashboardResource;
94
+ const { kind, metadata, spec: { display, duration, refreshInterval = _core.DEFAULT_REFRESH_INTERVAL, datasources, layouts = [], panels = {} } } = dashboardResource;
94
95
  const ttl = 'ttl' in dashboardResource.spec ? dashboardResource.spec.ttl : undefined;
95
- let { spec: { layouts, panels } } = dashboardResource;
96
- // Set fallbacks in case the frontend is used with a non-Perses backend
97
- layouts = layouts ?? [];
98
- panels = panels ?? {};
99
96
  const store = (0, _zustand.createStore)()((0, _immer.immer)((0, _middleware.devtools)((...args)=>{
100
97
  const [set] = args;
101
98
  return {
@@ -118,9 +115,11 @@ function initStore(props) {
118
115
  datasources,
119
116
  ttl,
120
117
  isEditMode: !!isEditMode,
121
- setEditMode: (isEditMode)=>set({
118
+ setEditMode: (isEditMode)=>{
119
+ set({
122
120
  isEditMode
123
- }),
121
+ });
122
+ },
124
123
  setDashboard: ({ kind, metadata, spec: { display, panels = {}, layouts = [], duration, refreshInterval, datasources = {} } })=>{
125
124
  set((state)=>{
126
125
  state.kind = kind;
@@ -24,12 +24,12 @@ const _jsxruntime = require("react/jsx-runtime");
24
24
  const _usequeryparams = require("use-query-params");
25
25
  const _DashboardProvider = require("./DashboardProvider");
26
26
  function DashboardProviderWithQueryParams({ children, initialState }) {
27
- const [viewPanelRef, setViewPanelRef] = (0, _usequeryparams.useQueryParam)('viewPanelRef', _usequeryparams.StringParam);
27
+ const [viewPanelRef, setViewPanelRef] = (0, _usequeryparams.useQueryParam)('viewPanelRef', _usequeryparams.JsonParam);
28
28
  return /*#__PURE__*/ (0, _jsxruntime.jsx)(_DashboardProvider.DashboardProvider, {
29
29
  initialState: {
30
+ ...initialState,
30
31
  viewPanelRef: viewPanelRef ?? undefined,
31
- setViewPanelRef: setViewPanelRef,
32
- ...initialState
32
+ setViewPanelRef: setViewPanelRef
33
33
  },
34
34
  children: children
35
35
  });
@@ -42,6 +42,7 @@ function createPanelEditorSlice() {
42
42
  }
43
43
  const editorState = {
44
44
  mode: 'update',
45
+ panelGroupItemId: panelGroupItemId,
45
46
  initialValues: {
46
47
  groupId: panelGroupItemId.panelGroupId,
47
48
  panelDefinition: panelToEdit
@@ -29,12 +29,14 @@ const createPanelGroupEditorSlice = (set, get)=>({
29
29
  mode: 'Add',
30
30
  initialValues: {
31
31
  title: '',
32
- isCollapsed: false
32
+ isCollapsed: false,
33
+ repeatVariable: ''
33
34
  },
34
35
  applyChanges (next) {
35
36
  const newGroup = (0, _panelgroupslice.createEmptyPanelGroup)();
36
37
  newGroup.title = next.title;
37
38
  newGroup.isCollapsed = next.isCollapsed;
39
+ newGroup.repeatVariable = next.repeatVariable;
38
40
  set((draft)=>{
39
41
  (0, _panelgroupslice.addPanelGroup)(draft, newGroup);
40
42
  });
@@ -60,7 +62,8 @@ const createPanelGroupEditorSlice = (set, get)=>({
60
62
  mode: 'Edit',
61
63
  initialValues: {
62
64
  title: existingGroup.title ?? '',
63
- isCollapsed: existingGroup.isCollapsed
65
+ isCollapsed: existingGroup.isCollapsed,
66
+ repeatVariable: existingGroup.repeatVariable ?? ''
64
67
  },
65
68
  applyChanges (next) {
66
69
  set((draft)=>{
@@ -70,6 +73,7 @@ const createPanelGroupEditorSlice = (set, get)=>({
70
73
  }
71
74
  group.title = next.title;
72
75
  group.isCollapsed = next.isCollapsed;
76
+ group.repeatVariable = next.repeatVariable;
73
77
  });
74
78
  },
75
79
  close () {
@@ -100,6 +100,7 @@ function convertLayoutsToPanelGroups(layouts) {
100
100
  panelGroups[panelGroupId] = {
101
101
  id: panelGroupId,
102
102
  isCollapsed: layout.spec.display?.collapse?.open === false,
103
+ repeatVariable: layout.spec.repeatVariable,
103
104
  title: layout.spec.display?.title,
104
105
  itemLayouts,
105
106
  itemPanelKeys
@@ -55,12 +55,13 @@ function getViewPanelGroupId(panelGroups, panelGroupItemId, panelRef) {
55
55
  // Find the PanelGroupItemId of a Panel from a PanelRef
56
56
  function findPanelGroupItemIdOfPanelRef(panelGroups, panelRef) {
57
57
  for (const panelGroup of Object.values(panelGroups)){
58
- const itemPanel = Object.entries(panelGroup.itemPanelKeys ?? []).find(([_, value])=>value === panelRef);
58
+ const itemPanel = Object.entries(panelGroup.itemPanelKeys ?? []).find(([_, value])=>value === panelRef.ref);
59
59
  if (itemPanel) {
60
60
  const [key] = itemPanel;
61
61
  return {
62
62
  panelGroupId: panelGroup.id,
63
- panelGroupItemLayoutId: key
63
+ panelGroupItemLayoutId: key,
64
+ repeatVariable: panelRef.repeatVariable
64
65
  };
65
66
  }
66
67
  }
@@ -73,7 +74,13 @@ function findPanelRefOfPanelGroupItemId(panelGroups, panelGroupItemId) {
73
74
  }
74
75
  const panelGroup = panelGroups[panelGroupItemId.panelGroupId];
75
76
  if (panelGroup) {
76
- return panelGroup.itemPanelKeys[panelGroupItemId.panelGroupItemLayoutId];
77
+ const panelRef = panelGroup.itemPanelKeys[panelGroupItemId.panelGroupItemLayoutId];
78
+ if (panelRef) {
79
+ return {
80
+ ref: panelRef,
81
+ repeatVariable: panelGroupItemId.repeatVariable
82
+ };
83
+ }
77
84
  }
78
85
  return undefined;
79
86
  }
@@ -362,7 +362,7 @@ function createVariableDefinitionStore({ initialVariableDefinitions = [], extern
362
362
  return (0, _utils.checkSavedDefaultVariableStatus)(get().variableDefinitions, get().variableState);
363
363
  }
364
364
  }))));
365
- return store; // TODO: @Gladorme check if we can avoid this cast
365
+ return store;
366
366
  }
367
367
  function VariableProvider({ children, initialVariableDefinitions = [], externalVariableDefinitions = [], builtinVariableDefinitions = [] }) {
368
368
  const [store] = (0, _react.useState)(()=>createVariableDefinitionStore({
@@ -82,11 +82,11 @@ function convertPanelGroupsToLayouts(panelGroups, panelGroupOrder) {
82
82
  if (group === undefined) {
83
83
  throw new Error('panel group not found');
84
84
  }
85
- const { title, isCollapsed, itemLayouts, itemPanelKeys } = group;
85
+ const { title, isCollapsed, repeatVariable, itemLayouts, itemPanelKeys } = group;
86
86
  let display = undefined;
87
- if (title) {
87
+ if (title || isCollapsed !== undefined) {
88
88
  display = {
89
- title,
89
+ title: title ?? '',
90
90
  collapse: {
91
91
  open: !isCollapsed
92
92
  }
@@ -108,7 +108,8 @@ function convertPanelGroupsToLayouts(panelGroups, panelGroupOrder) {
108
108
  height: layout.h,
109
109
  content: (0, _core.createPanelRef)(panelKey)
110
110
  };
111
- })
111
+ }),
112
+ repeatVariable: repeatVariable
112
113
  }
113
114
  };
114
115
  layouts.push(layout);
@@ -28,7 +28,7 @@ const _pluginsystem = require("@perses-dev/plugin-system");
28
28
  const _components1 = require("../../components");
29
29
  const _context = require("../../context");
30
30
  const DashboardApp = (props)=>{
31
- const { dashboardResource, dashboardTitleComponent, emptyDashboardProps, onSave, onDiscard, initialVariableIsSticky, isReadonly, isVariableEnabled, isDatasourceEnabled, isCreating } = props;
31
+ const { dashboardResource, emptyDashboardProps, isReadonly, isVariableEnabled, isDatasourceEnabled, isCreating, isInitialVariableSticky, isLeavingConfirmDialogEnabled, dashboardTitleComponent, onSave, onDiscard } = props;
32
32
  const chartsTheme = (0, _components.useChartsTheme)();
33
33
  const { isEditMode, setEditMode } = (0, _context.useEditMode)();
34
34
  const { dashboard, setDashboard } = (0, _context.useDashboard)();
@@ -78,7 +78,7 @@ const DashboardApp = (props)=>{
78
78
  /*#__PURE__*/ (0, _jsxruntime.jsx)(_components1.DashboardToolbar, {
79
79
  dashboardName: dashboardResource.metadata.name,
80
80
  dashboardTitleComponent: dashboardTitleComponent,
81
- initialVariableIsSticky: initialVariableIsSticky,
81
+ initialVariableIsSticky: isInitialVariableSticky,
82
82
  onSave: onSave,
83
83
  isReadonly: isReadonly,
84
84
  isVariableEnabled: isVariableEnabled,
@@ -116,7 +116,11 @@ const DashboardApp = (props)=>{
116
116
  isReadonly: !isEditMode,
117
117
  disableMetadataEdition: !isCreating
118
118
  }),
119
- /*#__PURE__*/ (0, _jsxruntime.jsx)(_components1.SaveChangesConfirmationDialog, {})
119
+ /*#__PURE__*/ (0, _jsxruntime.jsx)(_components1.SaveChangesConfirmationDialog, {}),
120
+ isLeavingConfirmDialogEnabled && isEditMode && /*#__PURE__*/ (0, _jsxruntime.jsx)(_components1.LeaveDialog, {
121
+ original: originalDashboard,
122
+ current: dashboard
123
+ })
120
124
  ]
121
125
  })
122
126
  ]
@@ -30,7 +30,7 @@ const _context = require("../../context");
30
30
  const _DashboardProviderWithQueryParams = require("../../context/DashboardProvider/DashboardProviderWithQueryParams");
31
31
  const _DashboardApp = require("./DashboardApp");
32
32
  function ViewDashboard(props) {
33
- const { dashboardResource, datasourceApi, externalVariableDefinitions, dashboardTitleComponent, emptyDashboardProps, onSave, onDiscard, initialVariableIsSticky, isReadonly, isVariableEnabled, isDatasourceEnabled, isEditing, isCreating, sx, ...others } = props;
33
+ const { dashboardResource, datasourceApi, externalVariableDefinitions, emptyDashboardProps, isReadonly, isVariableEnabled, isDatasourceEnabled, isEditing, isCreating, isInitialVariableSticky, isLeavingConfirmDialogEnabled, dashboardTitleComponent, onSave, onDiscard, sx, ...others } = props;
34
34
  const { spec } = dashboardResource;
35
35
  const dashboardDuration = spec.duration ?? _core.DEFAULT_DASHBOARD_DURATION;
36
36
  const dashboardRefreshInterval = spec.refreshInterval ?? _core.DEFAULT_REFRESH_INTERVAL;
@@ -80,8 +80,8 @@ function ViewDashboard(props) {
80
80
  datasourceApi: datasourceApi,
81
81
  children: /*#__PURE__*/ (0, _jsxruntime.jsx)(_DashboardProviderWithQueryParams.DashboardProviderWithQueryParams, {
82
82
  initialState: {
83
- dashboardResource,
84
- isEditMode: !!isEditing
83
+ isEditMode: !!isEditing,
84
+ dashboardResource
85
85
  },
86
86
  children: /*#__PURE__*/ (0, _jsxruntime.jsx)(_pluginsystem.TimeRangeProviderWithQueryParams, {
87
87
  initialTimeRange: initialTimeRange,
@@ -103,15 +103,16 @@ function ViewDashboard(props) {
103
103
  FallbackComponent: _components.ErrorAlert,
104
104
  children: /*#__PURE__*/ (0, _jsxruntime.jsx)(_DashboardApp.DashboardApp, {
105
105
  dashboardResource: dashboardResource,
106
- dashboardTitleComponent: dashboardTitleComponent,
107
106
  emptyDashboardProps: emptyDashboardProps,
108
- onSave: onSave,
109
- onDiscard: onDiscard,
110
- initialVariableIsSticky: initialVariableIsSticky,
111
107
  isReadonly: isReadonly,
112
108
  isVariableEnabled: isVariableEnabled,
113
109
  isDatasourceEnabled: isDatasourceEnabled,
114
- isCreating: isCreating
110
+ isCreating: isCreating,
111
+ isInitialVariableSticky: isInitialVariableSticky,
112
+ isLeavingConfirmDialogEnabled: isLeavingConfirmDialogEnabled,
113
+ dashboardTitleComponent: dashboardTitleComponent,
114
+ onSave: onSave,
115
+ onDiscard: onDiscard
115
116
  })
116
117
  })
117
118
  })
@@ -1,8 +1,8 @@
1
- import { ReactElement } from 'react';
1
+ import { ReactElement, ReactNode } from 'react';
2
2
  import { OnSaveDashboard } from '../../context';
3
3
  export interface DashboardToolbarProps {
4
4
  dashboardName: string;
5
- dashboardTitleComponent?: JSX.Element;
5
+ dashboardTitleComponent?: ReactNode;
6
6
  initialVariableIsSticky?: boolean;
7
7
  isReadonly: boolean;
8
8
  isVariableEnabled: boolean;
@@ -1 +1 @@
1
- {"version":3,"file":"DashboardToolbar.d.ts","sourceRoot":"","sources":["../../../src/components/DashboardToolbar/DashboardToolbar.tsx"],"names":[],"mappings":"AAgBA,OAAO,EAAE,YAAY,EAAE,MAAM,OAAO,CAAC;AACrC,OAAO,EAAE,eAAe,EAAe,MAAM,eAAe,CAAC;AAW7D,MAAM,WAAW,qBAAqB;IACpC,aAAa,EAAE,MAAM,CAAC;IACtB,uBAAuB,CAAC,EAAE,GAAG,CAAC,OAAO,CAAC;IACtC,uBAAuB,CAAC,EAAE,OAAO,CAAC;IAClC,UAAU,EAAE,OAAO,CAAC;IACpB,iBAAiB,EAAE,OAAO,CAAC;IAC3B,mBAAmB,EAAE,OAAO,CAAC;IAC7B,iBAAiB,EAAE,MAAM,IAAI,CAAC;IAC9B,mBAAmB,EAAE,MAAM,IAAI,CAAC;IAChC,MAAM,CAAC,EAAE,eAAe,CAAC;CAC1B;AAED,eAAO,MAAM,gBAAgB,UAAW,qBAAqB,KAAG,YAgG/D,CAAC"}
1
+ {"version":3,"file":"DashboardToolbar.d.ts","sourceRoot":"","sources":["../../../src/components/DashboardToolbar/DashboardToolbar.tsx"],"names":[],"mappings":"AAgBA,OAAO,EAAE,YAAY,EAAE,SAAS,EAAE,MAAM,OAAO,CAAC;AAChD,OAAO,EAAE,eAAe,EAAe,MAAM,eAAe,CAAC;AAW7D,MAAM,WAAW,qBAAqB;IACpC,aAAa,EAAE,MAAM,CAAC;IACtB,uBAAuB,CAAC,EAAE,SAAS,CAAC;IACpC,uBAAuB,CAAC,EAAE,OAAO,CAAC;IAClC,UAAU,EAAE,OAAO,CAAC;IACpB,iBAAiB,EAAE,OAAO,CAAC;IAC3B,mBAAmB,EAAE,OAAO,CAAC;IAC7B,iBAAiB,EAAE,MAAM,IAAI,CAAC;IAC9B,mBAAmB,EAAE,MAAM,IAAI,CAAC;IAChC,MAAM,CAAC,EAAE,eAAe,CAAC;CAC1B;AAED,eAAO,MAAM,gBAAgB,UAAW,qBAAqB,KAAG,YAgG/D,CAAC"}
@@ -1 +1 @@
1
- {"version":3,"sources":["../../../src/components/DashboardToolbar/DashboardToolbar.tsx"],"sourcesContent":["// Copyright 2023 The Perses Authors\n// Licensed under the Apache License, Version 2.0 (the \"License\");\n// you may not use this file except in compliance with the License.\n// You may obtain a copy of the License at\n//\n// http://www.apache.org/licenses/LICENSE-2.0\n//\n// Unless required by applicable law or agreed to in writing, software\n// distributed under the License is distributed on an \"AS IS\" BASIS,\n// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n// See the License for the specific language governing permissions and\n// limitations under the License.\n\nimport { Typography, Stack, Button, Box, useTheme, useMediaQuery, Alert } from '@mui/material';\nimport { ErrorBoundary, ErrorAlert } from '@perses-dev/components';\nimport { TimeRangeControls } from '@perses-dev/plugin-system';\nimport { ReactElement } from 'react';\nimport { OnSaveDashboard, useEditMode } from '../../context';\nimport { AddPanelButton } from '../AddPanelButton';\nimport { AddGroupButton } from '../AddGroupButton';\nimport { DownloadButton } from '../DownloadButton';\nimport { EditVariablesButton } from '../Variables';\nimport { EditDatasourcesButton } from '../Datasources';\nimport { EditButton } from '../EditButton';\nimport { EditJsonButton } from '../EditJsonButton';\nimport { SaveDashboardButton } from '../SaveDashboardButton';\nimport { DashboardStickyToolbar } from '../DashboardStickyToolbar';\n\nexport interface DashboardToolbarProps {\n dashboardName: string;\n dashboardTitleComponent?: JSX.Element;\n initialVariableIsSticky?: boolean;\n isReadonly: boolean;\n isVariableEnabled: boolean;\n isDatasourceEnabled: boolean;\n onEditButtonClick: () => void;\n onCancelButtonClick: () => void;\n onSave?: OnSaveDashboard;\n}\n\nexport const DashboardToolbar = (props: DashboardToolbarProps): ReactElement => {\n const {\n dashboardName,\n dashboardTitleComponent,\n initialVariableIsSticky,\n isReadonly,\n isVariableEnabled,\n isDatasourceEnabled,\n onEditButtonClick,\n onCancelButtonClick,\n onSave,\n } = props;\n\n const { isEditMode } = useEditMode();\n\n const isBiggerThanSm = useMediaQuery(useTheme().breakpoints.up('sm'));\n const isBiggerThanMd = useMediaQuery(useTheme().breakpoints.up('md'));\n\n const dashboardTitle = dashboardTitleComponent ? (\n dashboardTitleComponent\n ) : (\n <Typography variant=\"h2\">{dashboardName}</Typography>\n );\n\n const testId = 'dashboard-toolbar';\n\n return (\n <>\n <Stack data-testid={testId}>\n <Box\n px={2}\n py={1.5}\n display=\"flex\"\n sx={{ backgroundColor: (theme) => theme.palette.primary.main + (isEditMode ? '30' : '0') }}\n >\n {dashboardTitle}\n {isEditMode ? (\n <Stack direction=\"row\" gap={1} ml=\"auto\">\n {isReadonly && (\n <Alert severity=\"warning\" sx={{ backgroundColor: 'transparent', padding: 0 }}>\n Dashboard managed via code only. Download JSON and commit changes to save.\n </Alert>\n )}\n <Stack direction=\"row\" spacing={0.5} ml={1} whiteSpace=\"nowrap\">\n {isVariableEnabled && <EditVariablesButton />}\n {isDatasourceEnabled && <EditDatasourcesButton />}\n <AddPanelButton />\n <AddGroupButton />\n </Stack>\n <SaveDashboardButton onSave={onSave} isDisabled={isReadonly} />\n <Button variant=\"outlined\" onClick={onCancelButtonClick}>\n Cancel\n </Button>\n </Stack>\n ) : (\n <>\n {isBiggerThanSm && (\n <Stack direction=\"row\" gap={1} ml=\"auto\">\n <EditButton onClick={onEditButtonClick} />\n </Stack>\n )}\n </>\n )}\n </Box>\n <Box\n sx={{\n display: 'flex',\n width: '100%',\n alignItems: 'start',\n padding: (theme) => theme.spacing(1, 2, 0, 2),\n flexDirection: isBiggerThanMd ? 'row' : 'column',\n flexWrap: 'nowrap',\n gap: 1,\n }}\n >\n <Box width=\"100%\">\n <ErrorBoundary FallbackComponent={ErrorAlert}>\n <DashboardStickyToolbar\n initialVariableIsSticky={initialVariableIsSticky}\n sx={{\n backgroundColor: ({ palette }) => palette.background.default,\n }}\n />\n </ErrorBoundary>\n </Box>\n <Stack direction=\"row\" ml=\"auto\" flexWrap=\"wrap\" justifyContent=\"end\">\n <Stack direction=\"row\" spacing={1} mt={1} ml={1}>\n <TimeRangeControls />\n <DownloadButton />\n <EditJsonButton isReadonly={!isEditMode} />\n </Stack>\n </Stack>\n </Box>\n </Stack>\n </>\n );\n};\n"],"names":["Typography","Stack","Button","Box","useTheme","useMediaQuery","Alert","ErrorBoundary","ErrorAlert","TimeRangeControls","useEditMode","AddPanelButton","AddGroupButton","DownloadButton","EditVariablesButton","EditDatasourcesButton","EditButton","EditJsonButton","SaveDashboardButton","DashboardStickyToolbar","DashboardToolbar","props","dashboardName","dashboardTitleComponent","initialVariableIsSticky","isReadonly","isVariableEnabled","isDatasourceEnabled","onEditButtonClick","onCancelButtonClick","onSave","isEditMode","isBiggerThanSm","breakpoints","up","isBiggerThanMd","dashboardTitle","variant","testId","data-testid","px","py","display","sx","backgroundColor","theme","palette","primary","main","direction","gap","ml","severity","padding","spacing","whiteSpace","isDisabled","onClick","width","alignItems","flexDirection","flexWrap","FallbackComponent","background","default","justifyContent","mt"],"mappings":"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"}
1
+ {"version":3,"sources":["../../../src/components/DashboardToolbar/DashboardToolbar.tsx"],"sourcesContent":["// Copyright 2023 The Perses Authors\n// Licensed under the Apache License, Version 2.0 (the \"License\");\n// you may not use this file except in compliance with the License.\n// You may obtain a copy of the License at\n//\n// http://www.apache.org/licenses/LICENSE-2.0\n//\n// Unless required by applicable law or agreed to in writing, software\n// distributed under the License is distributed on an \"AS IS\" BASIS,\n// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n// See the License for the specific language governing permissions and\n// limitations under the License.\n\nimport { Typography, Stack, Button, Box, useTheme, useMediaQuery, Alert } from '@mui/material';\nimport { ErrorBoundary, ErrorAlert } from '@perses-dev/components';\nimport { TimeRangeControls } from '@perses-dev/plugin-system';\nimport { ReactElement, ReactNode } from 'react';\nimport { OnSaveDashboard, useEditMode } from '../../context';\nimport { AddPanelButton } from '../AddPanelButton';\nimport { AddGroupButton } from '../AddGroupButton';\nimport { DownloadButton } from '../DownloadButton';\nimport { EditVariablesButton } from '../Variables';\nimport { EditDatasourcesButton } from '../Datasources';\nimport { EditButton } from '../EditButton';\nimport { EditJsonButton } from '../EditJsonButton';\nimport { SaveDashboardButton } from '../SaveDashboardButton';\nimport { DashboardStickyToolbar } from '../DashboardStickyToolbar';\n\nexport interface DashboardToolbarProps {\n dashboardName: string;\n dashboardTitleComponent?: ReactNode;\n initialVariableIsSticky?: boolean;\n isReadonly: boolean;\n isVariableEnabled: boolean;\n isDatasourceEnabled: boolean;\n onEditButtonClick: () => void;\n onCancelButtonClick: () => void;\n onSave?: OnSaveDashboard;\n}\n\nexport const DashboardToolbar = (props: DashboardToolbarProps): ReactElement => {\n const {\n dashboardName,\n dashboardTitleComponent,\n initialVariableIsSticky,\n isReadonly,\n isVariableEnabled,\n isDatasourceEnabled,\n onEditButtonClick,\n onCancelButtonClick,\n onSave,\n } = props;\n\n const { isEditMode } = useEditMode();\n\n const isBiggerThanSm = useMediaQuery(useTheme().breakpoints.up('sm'));\n const isBiggerThanMd = useMediaQuery(useTheme().breakpoints.up('md'));\n\n const dashboardTitle = dashboardTitleComponent ? (\n dashboardTitleComponent\n ) : (\n <Typography variant=\"h2\">{dashboardName}</Typography>\n );\n\n const testId = 'dashboard-toolbar';\n\n return (\n <>\n <Stack data-testid={testId}>\n <Box\n px={2}\n py={1.5}\n display=\"flex\"\n sx={{ backgroundColor: (theme) => theme.palette.primary.main + (isEditMode ? '30' : '0') }}\n >\n {dashboardTitle}\n {isEditMode ? (\n <Stack direction=\"row\" gap={1} ml=\"auto\">\n {isReadonly && (\n <Alert severity=\"warning\" sx={{ backgroundColor: 'transparent', padding: 0 }}>\n Dashboard managed via code only. Download JSON and commit changes to save.\n </Alert>\n )}\n <Stack direction=\"row\" spacing={0.5} ml={1} whiteSpace=\"nowrap\">\n {isVariableEnabled && <EditVariablesButton />}\n {isDatasourceEnabled && <EditDatasourcesButton />}\n <AddPanelButton />\n <AddGroupButton />\n </Stack>\n <SaveDashboardButton onSave={onSave} isDisabled={isReadonly} />\n <Button variant=\"outlined\" onClick={onCancelButtonClick}>\n Cancel\n </Button>\n </Stack>\n ) : (\n <>\n {isBiggerThanSm && (\n <Stack direction=\"row\" gap={1} ml=\"auto\">\n <EditButton onClick={onEditButtonClick} />\n </Stack>\n )}\n </>\n )}\n </Box>\n <Box\n sx={{\n display: 'flex',\n width: '100%',\n alignItems: 'start',\n padding: (theme) => theme.spacing(1, 2, 0, 2),\n flexDirection: isBiggerThanMd ? 'row' : 'column',\n flexWrap: 'nowrap',\n gap: 1,\n }}\n >\n <Box width=\"100%\">\n <ErrorBoundary FallbackComponent={ErrorAlert}>\n <DashboardStickyToolbar\n initialVariableIsSticky={initialVariableIsSticky}\n sx={{\n backgroundColor: ({ palette }) => palette.background.default,\n }}\n />\n </ErrorBoundary>\n </Box>\n <Stack direction=\"row\" ml=\"auto\" flexWrap=\"wrap\" justifyContent=\"end\">\n <Stack direction=\"row\" spacing={1} mt={1} ml={1}>\n <TimeRangeControls />\n <DownloadButton />\n <EditJsonButton isReadonly={!isEditMode} />\n </Stack>\n </Stack>\n </Box>\n </Stack>\n </>\n );\n};\n"],"names":["Typography","Stack","Button","Box","useTheme","useMediaQuery","Alert","ErrorBoundary","ErrorAlert","TimeRangeControls","useEditMode","AddPanelButton","AddGroupButton","DownloadButton","EditVariablesButton","EditDatasourcesButton","EditButton","EditJsonButton","SaveDashboardButton","DashboardStickyToolbar","DashboardToolbar","props","dashboardName","dashboardTitleComponent","initialVariableIsSticky","isReadonly","isVariableEnabled","isDatasourceEnabled","onEditButtonClick","onCancelButtonClick","onSave","isEditMode","isBiggerThanSm","breakpoints","up","isBiggerThanMd","dashboardTitle","variant","testId","data-testid","px","py","display","sx","backgroundColor","theme","palette","primary","main","direction","gap","ml","severity","padding","spacing","whiteSpace","isDisabled","onClick","width","alignItems","flexDirection","flexWrap","FallbackComponent","background","default","justifyContent","mt"],"mappings":"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"}
@@ -1,6 +1,7 @@
1
1
  import { ReactElement } from 'react';
2
2
  import { PanelGroupId } from '@perses-dev/core';
3
3
  import { PanelOptions } from '../Panel';
4
+ import { RowProps } from './Row';
4
5
  export interface GridLayoutProps {
5
6
  panelGroupId: PanelGroupId;
6
7
  panelOptions?: PanelOptions;
@@ -10,4 +11,11 @@ export interface GridLayoutProps {
10
11
  * Layout component that arranges children in a Grid based on the definition.
11
12
  */
12
13
  export declare function GridLayout(props: GridLayoutProps): ReactElement;
14
+ export interface RepeatGridLayoutProps extends RowProps {
15
+ repeatVariableName: string;
16
+ }
17
+ /**
18
+ * Renders a grid layout for a repeated variable, where each value of the variable will create a new row.
19
+ */
20
+ export declare function RepeatGridLayout({ repeatVariableName, panelGroupId, groupDefinition, gridColWidth, panelFullHeight, panelOptions, isEditMode, onLayoutChange, onWidthChange, }: RepeatGridLayoutProps): ReactElement | null;
13
21
  //# sourceMappingURL=GridLayout.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"GridLayout.d.ts","sourceRoot":"","sources":["../../../src/components/GridLayout/GridLayout.tsx"],"names":[],"mappings":"AAYA,OAAO,EAAE,YAAY,EAAgC,MAAM,OAAO,CAAC;AAInE,OAAO,EAAE,YAAY,EAAE,MAAM,kBAAkB,CAAC;AAUhD,OAAO,EAAE,YAAY,EAAE,MAAM,UAAU,CAAC;AAOxC,MAAM,WAAW,eAAe;IAC9B,YAAY,EAAE,YAAY,CAAC;IAC3B,YAAY,CAAC,EAAE,YAAY,CAAC;IAC5B,eAAe,CAAC,EAAE,MAAM,CAAC;CAC1B;AAED;;GAEG;AACH,wBAAgB,UAAU,CAAC,KAAK,EAAE,eAAe,GAAG,YAAY,CAmI/D"}
1
+ {"version":3,"file":"GridLayout.d.ts","sourceRoot":"","sources":["../../../src/components/GridLayout/GridLayout.tsx"],"names":[],"mappings":"AAYA,OAAO,EAAE,YAAY,EAAY,MAAM,OAAO,CAAC;AAE/C,OAAO,EAAE,YAAY,EAAE,MAAM,kBAAkB,CAAC;AAUhD,OAAO,EAAE,YAAY,EAAE,MAAM,UAAU,CAAC;AACxC,OAAO,EAAO,QAAQ,EAAE,MAAM,OAAO,CAAC;AAEtC,MAAM,WAAW,eAAe;IAC9B,YAAY,EAAE,YAAY,CAAC;IAC3B,YAAY,CAAC,EAAE,YAAY,CAAC;IAC5B,eAAe,CAAC,EAAE,MAAM,CAAC;CAC1B;AAED;;GAEG;AACH,wBAAgB,UAAU,CAAC,KAAK,EAAE,eAAe,GAAG,YAAY,CAmE/D;AAED,MAAM,WAAW,qBAAsB,SAAQ,QAAQ;IACrD,kBAAkB,EAAE,MAAM,CAAC;CAC5B;AAED;;GAEG;AACH,wBAAgB,gBAAgB,CAAC,EAC/B,kBAAkB,EAClB,YAAY,EACZ,eAAe,EACf,YAAY,EACZ,eAAe,EACf,YAAY,EACZ,UAAkB,EAClB,cAAc,EACd,aAAa,GACd,EAAE,qBAAqB,GAAG,YAAY,GAAG,IAAI,CA0C7C"}