@evergis/react 3.1.14 → 3.1.16

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 (43) hide show
  1. package/dist/components/Dashboard/components/Chart/styled.d.ts +16 -0
  2. package/dist/components/Dashboard/components/DashboardHeader/index.d.ts +3 -0
  3. package/dist/components/Dashboard/components/FeatureCardButtons/index.d.ts +2 -0
  4. package/dist/components/Dashboard/components/FeatureCardHeader/index.d.ts +2 -0
  5. package/dist/components/Dashboard/components/FeatureCardTitle/index.d.ts +5 -0
  6. package/dist/components/Dashboard/components/Pagination/index.d.ts +1 -1
  7. package/dist/components/Dashboard/components/index.d.ts +4 -0
  8. package/dist/components/Dashboard/containers/RoundedBackgroundContainer/styled.d.ts +2 -0
  9. package/dist/components/Dashboard/headers/DashboardDefaultHeader/index.d.ts +3 -0
  10. package/dist/components/Dashboard/headers/DashboardDefaultHeader/styled.d.ts +8 -0
  11. package/dist/components/Dashboard/headers/FeatureCardDefaultHeader/components/HeaderTitle.d.ts +4 -0
  12. package/dist/components/Dashboard/headers/FeatureCardDefaultHeader/index.d.ts +5 -0
  13. package/dist/components/Dashboard/headers/FeatureCardDefaultHeader/styled.d.ts +12 -0
  14. package/dist/components/Dashboard/headers/FeatureCardGradientHeader/index.d.ts +3 -0
  15. package/dist/components/Dashboard/headers/FeatureCardGradientHeader/styled.d.ts +4 -0
  16. package/dist/components/Dashboard/headers/FeatureCardIconHeader/index.d.ts +3 -0
  17. package/dist/components/Dashboard/headers/FeatureCardIconHeader/styled.d.ts +6 -0
  18. package/dist/components/Dashboard/headers/FeatureCardSlideshowHeader/index.d.ts +3 -0
  19. package/dist/components/Dashboard/headers/FeatureCardSlideshowHeader/styled.d.ts +8 -0
  20. package/dist/components/Dashboard/headers/index.d.ts +7 -0
  21. package/dist/components/Dashboard/hooks/index.d.ts +1 -0
  22. package/dist/components/Dashboard/hooks/useDashboardHeader.d.ts +2 -2
  23. package/dist/components/Dashboard/hooks/useGlobalContext.d.ts +0 -1
  24. package/dist/components/Dashboard/hooks/useHeaderRender.d.ts +2 -0
  25. package/dist/components/Dashboard/hooks/useWidgetContext.d.ts +8 -0
  26. package/dist/components/Dashboard/index.d.ts +1 -0
  27. package/dist/components/Dashboard/styled.d.ts +1 -0
  28. package/dist/components/Dashboard/types.d.ts +2 -0
  29. package/dist/components/Dashboard/utils/getDashboardHeader.d.ts +2 -0
  30. package/dist/components/Dashboard/utils/getFeatureCardHeader.d.ts +2 -0
  31. package/dist/components/Dashboard/utils/index.d.ts +2 -0
  32. package/dist/components/LayerIcon/index.d.ts +5 -0
  33. package/dist/components/LayerIcon/styled.d.ts +2 -0
  34. package/dist/components/LayerTree/types.d.ts +0 -2
  35. package/dist/components/index.d.ts +1 -0
  36. package/dist/contexts/DashboardContext/types.d.ts +8 -2
  37. package/dist/contexts/FeatureCardContext/types.d.ts +3 -1
  38. package/dist/contexts/GlobalContext/types.d.ts +1 -3
  39. package/dist/index.js +1460 -701
  40. package/dist/index.js.map +1 -1
  41. package/dist/react.esm.js +1185 -451
  42. package/dist/react.esm.js.map +1 -1
  43. package/package.json +3 -3
package/dist/index.js CHANGED
@@ -2,7 +2,7 @@
2
2
 
3
3
  var jsxRuntime = require('react/jsx-runtime');
4
4
  var uilibGl = require('@evergis/uilib-gl');
5
- var react = require('react');
5
+ var React = require('react');
6
6
  var styled = require('styled-components');
7
7
  var charts = require('@evergis/charts');
8
8
  var api = require('@evergis/api');
@@ -19,7 +19,7 @@ var turf = require('@turf/turf');
19
19
  var MapGL = require('react-map-gl/mapbox');
20
20
  require('@mapbox/mapbox-gl-draw/dist/mapbox-gl-draw.css');
21
21
  require('mapbox-gl/dist/mapbox-gl.css');
22
- var react$1 = require('swiper/react');
22
+ var react = require('swiper/react');
23
23
 
24
24
  const AddFeatureButton = ({ title, icon = "feature_add" /* , layerName, geometryType*/ }) => {
25
25
  // const [, handleAddFeature] = useFeatureCreator(layerName, geometryType);
@@ -37,14 +37,18 @@ const ChartTooltipTable = styled.table `
37
37
  }
38
38
  `;
39
39
  const ChartTooltip = styled(uilibGl.Flex) `
40
- flex-direction: row;
41
- align-items: center;
42
40
  flex-wrap: nowrap;
41
+ background: rgb(48, 69, 79);
42
+ border-radius: 0.25rem;
43
+ color: white;
43
44
 
44
45
  :not(:last-child) {
45
46
  margin-bottom: 0.25rem;
46
47
  }
47
48
  `;
49
+ const ChartTooltipLabel = styled.div `
50
+ margin-right: 0.25rem;
51
+ `;
48
52
  const ChartTooltipColor = styled.div `
49
53
  width: 0.625rem;
50
54
  height: 0.625rem;
@@ -90,6 +94,107 @@ const ChartWrapperContainer = styled.div `
90
94
  position: relative;
91
95
  width: 100%;
92
96
  `;
97
+ const Tooltip = styled.div `
98
+ position: relative;
99
+ border-radius: 0.25rem;
100
+ background-color: rgba(28, 33, 48);
101
+ padding: 0.5rem;
102
+ box-shadow: 0 0.125rem 0.25rem 0 rgba(0, 0, 0, 0.15);
103
+ font-size: 0.75rem;
104
+ transform: ${({ transform }) => transform};
105
+ color: white;
106
+
107
+ :before {
108
+ position: absolute;
109
+ display: block;
110
+ content: "";
111
+ width: 0;
112
+ height: 0;
113
+ }
114
+ `;
115
+ const LineChartTooltipStyles = styled.createGlobalStyle `
116
+ .dashboardLineChartTooltip.${charts.lineChartClassNames.lineChartMouseLabel} {
117
+ .${charts.lineChartClassNames.lineChartLabelFlex} {
118
+ justify-content: center;
119
+ align-items: flex-end;
120
+
121
+ ${Tooltip} {
122
+ margin: 0 0 12px 0;
123
+
124
+ :before {
125
+ top: auto;
126
+ bottom: 0;
127
+ left: 50%;
128
+ transform: translate(-50%, 100%);
129
+ border-left: 4px solid transparent;
130
+ border-right: 4px solid transparent;
131
+ border-top: 4px solid rgba(28, 33, 48, 0.9);
132
+ }
133
+ }
134
+ }
135
+ }
136
+
137
+ .${charts.lineChartClassNames.lineChartYScaleGlobal}, .${charts.lineChartClassNames.lineChartXScaleGlobal} {
138
+ .domain {
139
+ visibility: hidden;
140
+ }
141
+
142
+ .tick {
143
+ text {
144
+ fill: ${({ theme: { palette } }) => palette.textDisabled};
145
+ }
146
+
147
+ line {
148
+ visibility: hidden;
149
+ }
150
+ }
151
+ }
152
+
153
+ .${charts.lineChartClassNames.lineChartXScaleGlobal} {
154
+ .tick {
155
+ :first-of-type {
156
+ text {
157
+ text-anchor: start;
158
+ }
159
+ }
160
+
161
+ :last-of-type {
162
+ text {
163
+ text-anchor: end;
164
+ }
165
+ }
166
+ }
167
+ }
168
+
169
+ .${charts.lineChartClassNames.lineChartLine} {
170
+ stroke-width: 2px;
171
+ }
172
+
173
+ .${charts.lineChartClassNames.lineChartLabel} {
174
+ color: ${({ theme: { palette } }) => palette.textPrimary};
175
+ }
176
+
177
+ .${charts.lineChartClassNames.lineChartMouseCircle} {
178
+ stroke: #ffffff;
179
+ stroke-width: 2px;
180
+ }
181
+
182
+ .${charts.lineChartClassNames.lineChartGridLineX} {
183
+ stroke: ${({ theme: { palette } }) => palette.element};
184
+ }
185
+
186
+ text {
187
+ fill: ${({ theme: { palette } }) => palette.textDisabled};
188
+ }
189
+ `;
190
+ const BarChartStyles = styled.createGlobalStyle `
191
+ .dashboardBarChart {
192
+ .domain,
193
+ line {
194
+ display: none;
195
+ }
196
+ }
197
+ `;
93
198
 
94
199
  function range(start, stop, step) {
95
200
  start = +start, stop = +stop, step = (n = arguments.length) < 2 ? (stop = start, start = 0, 1) : n < 3 ? 1 : +step;
@@ -3684,31 +3789,31 @@ const isLayerService = (value) => isObject(value) && "name" in value;
3684
3789
 
3685
3790
  const isNumeric = (number) => !isNaN(parseFloat(number)) && isFinite(number);
3686
3791
 
3687
- const ConfigContext = react.createContext({});
3688
- const ConfigProvider = react.memo(({ children, config }) => {
3792
+ const ConfigContext = React.createContext({});
3793
+ const ConfigProvider = React.memo(({ children, config }) => {
3689
3794
  return jsxRuntime.jsx(ConfigContext.Provider, { value: { config }, children: children });
3690
3795
  });
3691
3796
 
3692
- const DashboardContext = react.createContext({});
3693
- const DashboardProvider = react.memo(({ children, ...props }) => {
3797
+ const DashboardContext = React.createContext({});
3798
+ const DashboardProvider = React.memo(({ children, ...props }) => {
3694
3799
  return jsxRuntime.jsx(DashboardContext.Provider, { value: props, children: children });
3695
3800
  });
3696
3801
 
3697
- const FeatureCardContext = react.createContext(null);
3698
- const FeatureCardProvider = react.memo(({ settings, children }) => jsxRuntime.jsx(FeatureCardContext.Provider, { value: settings, children: children }));
3802
+ const FeatureCardContext = React.createContext(null);
3803
+ const FeatureCardProvider = React.memo(({ settings, children }) => jsxRuntime.jsx(FeatureCardContext.Provider, { value: settings, children: children }));
3699
3804
 
3700
- const GlobalContext = react.createContext({});
3701
- const GlobalProvider = react.memo(({ children, ...props }) => {
3805
+ const GlobalContext = React.createContext({});
3806
+ const GlobalProvider = React.memo(({ children, ...props }) => {
3702
3807
  return jsxRuntime.jsx(GlobalContext.Provider, { value: props, children: children });
3703
3808
  });
3704
3809
 
3705
- const MapContext = react.createContext({});
3810
+ const MapContext = React.createContext({});
3706
3811
 
3707
3812
  const MapProvider = ({ basemapItems, defaultBasemap, children }) => {
3708
- const map = react.useRef();
3709
- const draw = react.useRef();
3710
- const [loaded, setLoaded] = react.useState(false);
3711
- const [basemapName, setBasemapName] = react.useState(defaultBasemap);
3813
+ const map = React.useRef();
3814
+ const draw = React.useRef();
3815
+ const [loaded, setLoaded] = React.useState(false);
3816
+ const [basemapName, setBasemapName] = React.useState(defaultBasemap);
3712
3817
  return (jsxRuntime.jsx(MapContext.Provider, { value: {
3713
3818
  map,
3714
3819
  draw,
@@ -3727,12 +3832,12 @@ exports.BaseMapTheme = void 0;
3727
3832
  BaseMapTheme["Dark"] = "dark";
3728
3833
  })(exports.BaseMapTheme || (exports.BaseMapTheme = {}));
3729
3834
 
3730
- const ServerNotificationsContext = react.createContext({});
3835
+ const ServerNotificationsContext = React.createContext({});
3731
3836
 
3732
3837
  const useServerNotifications = (url, initialized) => {
3733
- const hubConnection = react.useRef(null);
3734
- const [connection, setConnection] = react.useState(null);
3735
- react.useEffect(() => {
3838
+ const hubConnection = React.useRef(null);
3839
+ const [connection, setConnection] = React.useState(null);
3840
+ React.useEffect(() => {
3736
3841
  if (!initialized) {
3737
3842
  return;
3738
3843
  }
@@ -3748,7 +3853,7 @@ const useServerNotifications = (url, initialized) => {
3748
3853
  .catch(err => console.info("Ошибка:", err))
3749
3854
  .finally(() => setConnection(hubConnection.current));
3750
3855
  }, [initialized]); // eslint-disable-line
3751
- react.useEffect(() => {
3856
+ React.useEffect(() => {
3752
3857
  if (!connection || connection.state !== "Connected") {
3753
3858
  return;
3754
3859
  }
@@ -3762,7 +3867,7 @@ const useServerNotifications = (url, initialized) => {
3762
3867
 
3763
3868
  const ServerNotificationsProvider = ({ url, initialized, children }) => {
3764
3869
  const connection = useServerNotifications(url, initialized);
3765
- const addSubscription = react.useCallback(async (payload) => {
3870
+ const addSubscription = React.useCallback(async (payload) => {
3766
3871
  if (!connection || connection.state !== "Connected" || !payload) {
3767
3872
  return;
3768
3873
  }
@@ -3776,7 +3881,7 @@ const ServerNotificationsProvider = ({ url, initialized, children }) => {
3776
3881
  return Promise.resolve(null);
3777
3882
  }
3778
3883
  }, [connection]);
3779
- const updateSubscription = react.useCallback(async (id, payload) => {
3884
+ const updateSubscription = React.useCallback(async (id, payload) => {
3780
3885
  if (!connection || connection.state !== "Connected" || !id || !payload) {
3781
3886
  return;
3782
3887
  }
@@ -3787,7 +3892,7 @@ const ServerNotificationsProvider = ({ url, initialized, children }) => {
3787
3892
  console.info(`Ошибка обновления подписки ${id}:`, err);
3788
3893
  }
3789
3894
  }, [connection]);
3790
- const unsubscribeById = react.useCallback(async (id) => {
3895
+ const unsubscribeById = React.useCallback(async (id) => {
3791
3896
  if (!connection || connection.state !== "Connected" || !id) {
3792
3897
  return;
3793
3898
  }
@@ -3802,8 +3907,8 @@ const ServerNotificationsProvider = ({ url, initialized, children }) => {
3802
3907
  };
3803
3908
 
3804
3909
  const useWidgetContext = (type = exports.WidgetType.Dashboard) => {
3805
- const { projectInfo, updateProject, layerInfos, geometryFilter, dashboardLayers, setDashboardLayer, pageIndex: projectPageIndex, selectedTabId: projectSelectedTabId, setSelectedTabId: setProjectSelectedTabId, dataSources: projectDataSources, loading: projectLoading, filters: projectFilters, changeFilters: projectChangeFilters, expandContainer: projectExpandContainer, expandedContainers: projectExpandedContainers, nextPage: projectNextPage, prevPage: projectPrevPage, changePage: projectChangePage, } = react.useContext(DashboardContext) || {};
3806
- const { layerInfo, attributes, pageIndex: featurePageIndex, selectedTabId: featureSelectedTabId, setSelectedTabId: setFeatureSelectedTabId, dataSources: featureDataSources, loading: featureLoading, filters: featureFilters, changeFilters: featureChangeFilters, expandContainer: featureExpandContainer, expandedContainers: featureExpandedContainers, nextPage: featureNextPage, prevPage: featurePrevPage, changePage: featureChangePage, } = react.useContext(FeatureCardContext) || {};
3910
+ const { projectInfo, updateProject, layerInfos, geometryFilter, dashboardLayers, setDashboardLayer, components: dashboardComponents, pageIndex: projectPageIndex, selectedTabId: projectSelectedTabId, setSelectedTabId: setProjectSelectedTabId, dataSources: projectDataSources, loading: projectLoading, filters: projectFilters, changeFilters: projectChangeFilters, expandContainer: projectExpandContainer, expandedContainers: projectExpandedContainers, nextPage: projectNextPage, prevPage: projectPrevPage, changePage: projectChangePage, } = React.useContext(DashboardContext) || {};
3911
+ const { layerInfo, attributes, feature, closeFeatureCard, pageIndex: featurePageIndex, selectedTabId: featureSelectedTabId, setSelectedTabId: setFeatureSelectedTabId, dataSources: featureDataSources, loading: featureLoading, filters: featureFilters, changeFilters: featureChangeFilters, expandContainer: featureExpandContainer, expandedContainers: featureExpandedContainers, nextPage: featureNextPage, prevPage: featurePrevPage, changePage: featureChangePage, } = React.useContext(FeatureCardContext) || {};
3807
3912
  return {
3808
3913
  projectInfo,
3809
3914
  layerInfos,
@@ -3813,6 +3918,9 @@ const useWidgetContext = (type = exports.WidgetType.Dashboard) => {
3813
3918
  geometryFilter,
3814
3919
  layerInfo,
3815
3920
  attributes,
3921
+ feature,
3922
+ closeFeatureCard,
3923
+ components: dashboardComponents,
3816
3924
  isLoading: type === exports.WidgetType.Dashboard ? projectLoading : featureLoading,
3817
3925
  pageIndex: type === exports.WidgetType.Dashboard ? projectPageIndex : featurePageIndex,
3818
3926
  filters: type === exports.WidgetType.Dashboard ? projectFilters : featureFilters,
@@ -3829,15 +3937,14 @@ const useWidgetContext = (type = exports.WidgetType.Dashboard) => {
3829
3937
  };
3830
3938
 
3831
3939
  const useGlobalContext = () => {
3832
- const { t, language, themeName, api, ewktGeometry, LayerItemComponent } = react.useContext(GlobalContext) || {};
3833
- return react.useMemo(() => ({
3940
+ const { t, language, themeName, api, ewktGeometry } = React.useContext(GlobalContext) || {};
3941
+ return React.useMemo(() => ({
3834
3942
  t,
3835
3943
  language,
3836
3944
  themeName,
3837
3945
  api,
3838
3946
  ewktGeometry,
3839
- LayerItemComponent,
3840
- }), [language, t, api, ewktGeometry, themeName, LayerItemComponent]);
3947
+ }), [language, t, api, ewktGeometry, themeName]);
3841
3948
  };
3842
3949
 
3843
3950
  const HEIGHT_OFFSET = 20;
@@ -3920,12 +4027,12 @@ const useChartChange = ({ dataSources, chartId, width, height, relatedAttributes
3920
4027
  const { t } = useGlobalContext();
3921
4028
  const { layerInfos } = useWidgetContext();
3922
4029
  const strokeColors = relatedAttributes.filter(({ chartAxis }) => chartAxis === "y").map(({ axisColor }) => axisColor);
3923
- const ref = react.useRef({
4030
+ const ref = React.useRef({
3924
4031
  path: null,
3925
4032
  area: null,
3926
4033
  points: [],
3927
4034
  });
3928
- const onChange = react.useCallback((range) => {
4035
+ const onChange = React.useCallback((range) => {
3929
4036
  const { path, area, points } = ref.current;
3930
4037
  let filteredPoints = [...points];
3931
4038
  if (range) {
@@ -3937,7 +4044,7 @@ const useChartChange = ({ dataSources, chartId, width, height, relatedAttributes
3937
4044
  path && path.attr("d", lineGenerator(filteredPoints));
3938
4045
  area && area.attr("d", areaGenerator(height)(filteredPoints));
3939
4046
  }, [height, width]);
3940
- const customize = react.useCallback(({ svg }) => {
4047
+ const customize = React.useCallback(({ svg }) => {
3941
4048
  svg.style("overflow", "visible");
3942
4049
  svg
3943
4050
  .selectAll(`.${charts.lineChartClassNames.lineChartXScaleGlobal} line,
@@ -4708,7 +4815,7 @@ const customStyles = [
4708
4815
  ];
4709
4816
 
4710
4817
  const useMapContext = () => {
4711
- return react.useContext(MapContext);
4818
+ return React.useContext(MapContext);
4712
4819
  };
4713
4820
 
4714
4821
  const draw = new MapboxDraw({
@@ -4722,14 +4829,14 @@ const draw = new MapboxDraw({
4722
4829
  });
4723
4830
  const useMapDraw = (triggerDeps = []) => {
4724
4831
  const { map, draw: drawContext, loaded, basemapName } = useMapContext();
4725
- react.useEffect(() => {
4832
+ React.useEffect(() => {
4726
4833
  if (!loaded || !map.current) {
4727
4834
  return;
4728
4835
  }
4729
4836
  drawContext.current = draw;
4730
4837
  map.current.addControl(drawContext.current);
4731
4838
  }, [loaded]); // eslint-disable-line
4732
- react.useEffect(() => {
4839
+ React.useEffect(() => {
4733
4840
  if (map.current && map.current.hasControl(drawContext.current)) {
4734
4841
  map.current.removeControl(drawContext.current);
4735
4842
  map.current.addControl(drawContext.current);
@@ -4739,7 +4846,7 @@ const useMapDraw = (triggerDeps = []) => {
4739
4846
 
4740
4847
  const useRedrawLayer = () => {
4741
4848
  const { map } = useMapContext();
4742
- return react.useCallback((layerName) => {
4849
+ return React.useCallback((layerName) => {
4743
4850
  const layerTileSource = map.current.getSource(layerName);
4744
4851
  layerTileSource.setTiles(layerTileSource.tiles);
4745
4852
  }, [map]);
@@ -4802,7 +4909,7 @@ const convertSpToTurfFeature = (geometry) => {
4802
4909
 
4803
4910
  const useZoomToFeatures = () => {
4804
4911
  const { map } = useMapContext();
4805
- return react.useCallback((features, padding) => {
4912
+ return React.useCallback((features, padding) => {
4806
4913
  if (!features) {
4807
4914
  return;
4808
4915
  }
@@ -4816,7 +4923,7 @@ const useZoomToFeatures = () => {
4816
4923
 
4817
4924
  const useZoomToPoint = () => {
4818
4925
  const { map } = useMapContext();
4819
- return react.useCallback((options, callback) => {
4926
+ return React.useCallback((options, callback) => {
4820
4927
  if (map.current) {
4821
4928
  if (callback) {
4822
4929
  map.current.once("moveend", () => {
@@ -4833,8 +4940,8 @@ const useZoomToPoint = () => {
4833
4940
 
4834
4941
  const useLayerParams = (layer) => {
4835
4942
  const { api } = useGlobalContext();
4836
- const [layerParams, setLayerParams] = react.useState({});
4837
- react.useEffect(() => {
4943
+ const [layerParams, setLayerParams] = React.useState({});
4944
+ React.useEffect(() => {
4838
4945
  if (layer?.type !== "QueryLayerService") {
4839
4946
  return;
4840
4947
  }
@@ -4847,27 +4954,27 @@ const useLayerParams = (layer) => {
4847
4954
  };
4848
4955
 
4849
4956
  const useServerNotificationsContext = () => {
4850
- return react.useContext(ServerNotificationsContext);
4957
+ return React.useContext(ServerNotificationsContext);
4851
4958
  };
4852
4959
 
4853
4960
  const useDebouncedCallback = (interval) => {
4854
- return react.useMemo(() => debounce((cb) => {
4961
+ return React.useMemo(() => debounce((cb) => {
4855
4962
  cb();
4856
4963
  }, interval), [interval]);
4857
4964
  };
4858
4965
 
4859
4966
  const useToggle = (initial) => {
4860
- const [state, setState] = react.useState(initial !== undefined ? initial : false);
4861
- const toggle = react.useCallback(() => setState(!state), [state]);
4967
+ const [state, setState] = React.useState(initial !== undefined ? initial : false);
4968
+ const toggle = React.useCallback(() => setState(!state), [state]);
4862
4969
  return [state, toggle, setState];
4863
4970
  };
4864
4971
 
4865
4972
  const DEBOUNCE_DELAY = 144;
4866
4973
  const useWindowResize = (callback, delay) => {
4867
- const debounceCallback = react.useMemo(() => {
4974
+ const debounceCallback = React.useMemo(() => {
4868
4975
  return callback ? debounce(callback, delay || DEBOUNCE_DELAY) : undefined;
4869
4976
  }, [callback, delay]);
4870
- react.useEffect(() => {
4977
+ React.useEffect(() => {
4871
4978
  debounceCallback && window.addEventListener("resize", debounceCallback);
4872
4979
  return () => debounceCallback && window.removeEventListener("resize", debounceCallback);
4873
4980
  }, [debounceCallback]);
@@ -4875,8 +4982,8 @@ const useWindowResize = (callback, delay) => {
4875
4982
 
4876
4983
  const useLayerGroupMenu = () => {
4877
4984
  const [isMenuOpen, toggleMenu] = useToggle(false);
4878
- const options = react.useMemo(() => [], []);
4879
- const selectOption = react.useCallback(() => {
4985
+ const options = React.useMemo(() => [], []);
4986
+ const selectOption = React.useCallback(() => {
4880
4987
  // ...
4881
4988
  }, [toggleMenu]);
4882
4989
  return {
@@ -4891,7 +4998,7 @@ const LayerGroup = ({ group, onlyMainTools }) => {
4891
4998
  const { projectInfo, updateProject } = useWidgetContext();
4892
4999
  const { pageIndex } = useWidgetPage();
4893
5000
  const { isMenuOpen, toggleMenu, options, selectOption } = useLayerGroupMenu();
4894
- const onToggleExpand = react.useCallback((name, isExpanded) => {
5001
+ const onToggleExpand = React.useCallback((name, isExpanded) => {
4895
5002
  if (!projectInfo) {
4896
5003
  return;
4897
5004
  }
@@ -4900,7 +5007,7 @@ const LayerGroup = ({ group, onlyMainTools }) => {
4900
5007
  page.layers = findAnd.changeProps(projectInfo.content.items, { name }, { isExpanded });
4901
5008
  updateProject(newProjectInfo);
4902
5009
  }, [pageIndex, projectInfo, updateProject]);
4903
- const onToggleVisibility = react.useCallback((name, isVisible) => {
5010
+ const onToggleVisibility = React.useCallback((name, isVisible) => {
4904
5011
  if (!projectInfo) {
4905
5012
  return;
4906
5013
  }
@@ -4944,11 +5051,11 @@ const treeNodesToProjectItems = (currentProjectItems, treeNodes) => {
4944
5051
  return combineProjectItems(treeNodes);
4945
5052
  };
4946
5053
 
4947
- const LayerTree = ({ layers, LayerItemComponent, onlyMainTools }) => {
4948
- const { projectInfo, updateProject } = useWidgetContext();
5054
+ const LayerTree = ({ layers, onlyMainTools }) => {
5055
+ const { projectInfo, updateProject, components: { LayerItem } } = useWidgetContext();
4949
5056
  const { pageIndex } = useWidgetPage();
4950
- const nodes = react.useMemo(() => layers?.map(layer => createTreeNode(layer, LayerItemComponent, onlyMainTools)), [LayerItemComponent, layers, onlyMainTools]);
4951
- const onUpdate = react.useCallback((updatedNodes) => {
5057
+ const nodes = React.useMemo(() => layers?.map(layer => createTreeNode(layer, LayerItem, onlyMainTools)), [LayerItem, layers, onlyMainTools]);
5058
+ const onUpdate = React.useCallback((updatedNodes) => {
4952
5059
  const newProjectInfo = JSON.parse(JSON.stringify(projectInfo));
4953
5060
  const page = getPagesFromProjectInfo(newProjectInfo)?.[pageIndex - 1];
4954
5061
  if (!page) {
@@ -5229,6 +5336,18 @@ const AttributeLabel = styled(uilibGl.Description) `
5229
5336
  margin-bottom: ${({ forCheckbox }) => (forCheckbox ? "0.75rem" : "0.25rem")} !important;
5230
5337
  padding-left: ${({ isEdit }) => (isEdit ? "0.5rem" : "0")};
5231
5338
  `;
5339
+ const FeatureControls = styled(uilibGl.Flex) `
5340
+ align-items: center;
5341
+ gap: 1rem;
5342
+ flex-wrap: nowrap;
5343
+ position: relative;
5344
+ flex-shrink: 0;
5345
+
5346
+ button {
5347
+ padding: 0;
5348
+ width: auto;
5349
+ }
5350
+ `;
5232
5351
 
5233
5352
  const getAttributeByName = (attributeName, attributes) => {
5234
5353
  return Array.isArray(attributeName)
@@ -5254,7 +5373,7 @@ const formatElementValue = ({ t, value, elementConfig, attributes, wrap, }) => {
5254
5373
  : valueOrDefault;
5255
5374
  if (!wrap)
5256
5375
  return resultValue;
5257
- return (jsxRuntime.jsxs(react.Fragment, { children: [tagView ? (jsxRuntime.jsx(DashboardChip$1, { text: resultValue, "$bgColor": bgColor, "$fontColor": fontColor, "$fontSize": fontSize, "$radius": radius, style: style })) : (jsxRuntime.jsx(ElementValueWrapper, { "data-id": id, "data-templatename": templateName, style: style, children: resultValue })), withDivider && jsxRuntime.jsx(uilibGl.Divider, {})] }, id));
5376
+ return (jsxRuntime.jsxs(React.Fragment, { children: [tagView ? (jsxRuntime.jsx(DashboardChip$1, { text: resultValue, "$bgColor": bgColor, "$fontColor": fontColor, "$fontSize": fontSize, "$radius": radius, style: style })) : (jsxRuntime.jsx(ElementValueWrapper, { "data-id": id, "data-templatename": templateName, style: style, children: resultValue })), withDivider && jsxRuntime.jsx(uilibGl.Divider, {})] }, id));
5258
5377
  };
5259
5378
 
5260
5379
  const getAttributeValue = (element, attributes) => {
@@ -5287,7 +5406,7 @@ const getChartMarkers = (items, markers) => {
5287
5406
  })) || []);
5288
5407
  };
5289
5408
 
5290
- const ContainersGroupContainer = react.memo(({ elementConfig, type, renderElement }) => {
5409
+ const ContainersGroupContainer = React.memo(({ elementConfig, type, renderElement }) => {
5291
5410
  const { expandedContainers } = useWidgetContext(type);
5292
5411
  const { id, children, options, style } = elementConfig || {};
5293
5412
  const { column, expandable, expanded } = options || {};
@@ -5296,7 +5415,7 @@ const ContainersGroupContainer = react.memo(({ elementConfig, type, renderElemen
5296
5415
  return (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [jsxRuntime.jsx(ExpandableTitle, { elementConfig: elementConfig, type: type, renderElement: renderElement }), isVisible && (jsxRuntime.jsx(Container, { style: style, isColumn: isColumn, children: jsxRuntime.jsx(ContainerChildren, { items: children, elementConfig: elementConfig, isColumn: isColumn, isMain: id?.startsWith(CONFIG_PAGE_ID), renderElement: renderElement }) }))] }));
5297
5416
  });
5298
5417
 
5299
- const OneColumnContainer = react.memo(({ elementConfig, renderElement }) => {
5418
+ const OneColumnContainer = React.memo(({ elementConfig, renderElement }) => {
5300
5419
  const { options, style } = elementConfig || {};
5301
5420
  const { innerTemplateStyle, hideEmpty } = options || {};
5302
5421
  const value = renderElement({ id: "value" });
@@ -5306,10 +5425,10 @@ const OneColumnContainer = react.memo(({ elementConfig, renderElement }) => {
5306
5425
  return (jsxRuntime.jsxs(Container, { style: innerTemplateStyle || style, children: [jsxRuntime.jsxs(ContainerAlias, { hasBottomMargin: true, children: [renderElement({ id: "alias" }), renderElement({ id: "tooltip" })] }), jsxRuntime.jsxs(ContainerValue, { children: [value, hasUnits && jsxRuntime.jsx(ContainerUnits, { children: renderElement({ id: "units" }) })] })] }));
5307
5426
  });
5308
5427
 
5309
- const TwoColumnContainer = react.memo(({ config, elementConfig, type, renderElement }) => {
5428
+ const TwoColumnContainer = React.memo(({ config, elementConfig, type, renderElement }) => {
5310
5429
  const { selectedTabId, layerInfo, attributes } = useWidgetContext(type);
5311
5430
  const { attributes: renderAttributes } = elementConfig?.options || {};
5312
- const renderContainer = react.useCallback((element, attribute) => {
5431
+ const renderContainer = React.useCallback((element, attribute) => {
5313
5432
  const { options, style, children } = element || {};
5314
5433
  const { hideEmpty, innerTemplateStyle } = options || {};
5315
5434
  const hasUnits = children?.some(({ id }) => id === "units");
@@ -5359,7 +5478,7 @@ const InnerContainerWrapper = styled.div `
5359
5478
  opacity: ${({ hasAnyFilter, isFiltered }) => (isFiltered ? 1 : hasAnyFilter ? FILTERED_VALUE_OPACITY / 100 : 1)};
5360
5479
  `;
5361
5480
 
5362
- const DataSourceInnerContainer = react.memo(({ config, elementConfig, feature, maxValue, type, index, innerComponent }) => {
5481
+ const DataSourceInnerContainer = React.memo(({ config, elementConfig, feature, maxValue, type, index, innerComponent }) => {
5363
5482
  const { t } = useGlobalContext();
5364
5483
  const { expandedContainers, selectedTabId, setSelectedTabId, dataSources } = useWidgetContext(type);
5365
5484
  const { pageIndex, currentPage } = useWidgetPage(type);
@@ -5371,12 +5490,12 @@ const DataSourceInnerContainer = react.memo(({ config, elementConfig, feature, m
5371
5490
  });
5372
5491
  const { options, children } = elementConfig || {};
5373
5492
  const { relatedDataSource, filterName, column } = options || {};
5374
- const aliasElement = react.useMemo(() => children?.find(({ id }) => id === "alias"), [children]);
5375
- const valueElement = react.useMemo(() => children?.find(({ id }) => id === "value"), [children]);
5376
- const aliasAttribute = react.useMemo(() => (aliasElement ? attributes?.find(({ name }) => name === aliasElement.attributeName) : null), [aliasElement, attributes]);
5493
+ const aliasElement = React.useMemo(() => children?.find(({ id }) => id === "alias"), [children]);
5494
+ const valueElement = React.useMemo(() => children?.find(({ id }) => id === "value"), [children]);
5495
+ const aliasAttribute = React.useMemo(() => (aliasElement ? attributes?.find(({ name }) => name === aliasElement.attributeName) : null), [aliasElement, attributes]);
5377
5496
  const value = aliasAttribute?.value;
5378
5497
  const InnerContainer = innerComponent;
5379
- const data = react.useMemo(() => getDataFromRelatedFeatures({
5498
+ const data = React.useMemo(() => getDataFromRelatedFeatures({
5380
5499
  t,
5381
5500
  config: elementConfig,
5382
5501
  filters: currentPage?.filters,
@@ -5399,7 +5518,7 @@ const DataSourceInnerContainer = react.memo(({ config, elementConfig, feature, m
5399
5518
  valueElement?.attributeName
5400
5519
  ]);
5401
5520
  const { hasAnyFilter, isFiltered, onFilter } = useFeatureFilters(type, filterName, data);
5402
- const render = react.useMemo(() => getRenderElement({
5521
+ const render = React.useMemo(() => getRenderElement({
5403
5522
  config,
5404
5523
  elementConfig,
5405
5524
  attributes,
@@ -5530,7 +5649,7 @@ const ProgressTotal = styled(uilibGl.Flex) `
5530
5649
  margin-top: 1rem;
5531
5650
  `;
5532
5651
 
5533
- const DataSourceProgressContainer = react.memo(({ config, elementConfig, type, innerComponent, renderElement }) => {
5652
+ const DataSourceProgressContainer = React.memo(({ config, elementConfig, type, innerComponent, renderElement }) => {
5534
5653
  const { t } = useGlobalContext();
5535
5654
  const { dataSources, expandedContainers } = useWidgetContext(type);
5536
5655
  const { dataSource, layerInfo } = useRelatedDataSourceAttributes({
@@ -5544,17 +5663,17 @@ const DataSourceProgressContainer = react.memo(({ config, elementConfig, type, i
5544
5663
  const valueElement = children?.find(item => item.id === "value");
5545
5664
  const unitsElement = children?.find(item => item.id === "units");
5546
5665
  const { sliceItems, checkIsSliced, showMore, onShowMore } = useShownOtherItems(options);
5547
- const totalUnits = react.useMemo(() => unitsElement?.type === "attributeUnits"
5666
+ const totalUnits = React.useMemo(() => unitsElement?.type === "attributeUnits"
5548
5667
  ? attributes?.find(({ attributeName }) => attributeName === unitsElement.attributeName)?.stringFormat
5549
5668
  ?.unitsLabel
5550
5669
  : dataSource?.features?.[0]?.attributes[unitsElement?.attributeName], [attributes, dataSource?.features, unitsElement?.attributeName, unitsElement?.type]);
5551
- const totalValue = react.useMemo(() => {
5670
+ const totalValue = React.useMemo(() => {
5552
5671
  const attribute = attributes?.find(({ attributeName }) => attributeName === valueElement?.attributeName);
5553
5672
  const { type: attributeType, stringFormat } = attribute || {};
5554
5673
  const result = dataSource?.features?.reduce((total, feature) => total + feature.attributes[valueElement?.attributeName], 0);
5555
5674
  return formatAttributeValue({ t, type: attributeType, value: result, stringFormat, noUnits: !!unitsElement?.type });
5556
5675
  }, [attributes, dataSource?.features, unitsElement?.type, valueElement?.attributeName]);
5557
- const currentMaxValue = react.useMemo(() => {
5676
+ const currentMaxValue = React.useMemo(() => {
5558
5677
  if (typeof maxValue === "string") {
5559
5678
  return dataSource?.features?.[0]?.attributes[maxValue];
5560
5679
  }
@@ -5570,7 +5689,7 @@ const DataSourceProgressContainer = react.memo(({ config, elementConfig, type, i
5570
5689
  return (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [jsxRuntime.jsx(ExpandableTitle, { elementConfig: elementConfig, type: type, renderElement: renderElement }), !isVisible ? null : dataSource ? (jsxRuntime.jsxs(DataSourceProgressContainerWrapper, { children: [sliceItems(dataSource?.features)?.map((feature, index) => (jsxRuntime.jsx(DataSourceInnerContainer, { type: type, index: index, feature: feature, config: config, elementConfig: elementConfig, maxValue: currentMaxValue, innerComponent: innerComponent }, index))), checkIsSliced(dataSource?.features) && (jsxRuntime.jsx(ContainerToggler, { toggled: showMore, onClick: onShowMore, children: showMore ? t("hide", { ns: "dashboard" }) : t("showAll", { ns: "dashboard" }) })), showTotal && (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [jsxRuntime.jsx(uilibGl.Divider, {}), jsxRuntime.jsxs(ProgressTotal, { children: [jsxRuntime.jsx(ProgressTotalTitle, { children: t("total", { ns: "dashboard" }) }), jsxRuntime.jsxs(ProgressValue, { children: [totalValue, jsxRuntime.jsx(ProgressUnits, { children: totalUnits })] })] })] }))] })) : (jsxRuntime.jsx(ContainerLoading, {})), jsxRuntime.jsx(HiddenTitleItems, { elementConfig: elementConfig, config: config, type: type })] }));
5571
5690
  });
5572
5691
 
5573
- const ProgressContainer = react.memo(({ type, elementConfig, feature, maxValue, index, renderElement }) => {
5692
+ const ProgressContainer = React.memo(({ type, elementConfig, feature, maxValue, index, renderElement }) => {
5574
5693
  const { t } = useGlobalContext();
5575
5694
  const { dataSources } = useWidgetContext(type);
5576
5695
  const { attributes } = useRelatedDataSourceAttributes({
@@ -5588,22 +5707,22 @@ const ProgressContainer = react.memo(({ type, elementConfig, feature, maxValue,
5588
5707
  const value = !lodash.isNil(valueAttribute?.value) ? +valueAttribute.value : +valueElement.value || 0;
5589
5708
  const currentMaxValue = (maxValue || optionMaxValue);
5590
5709
  const width = `${Math.min(100, (value / currentMaxValue) * 100).toFixed(2)}%`;
5591
- const renderTooltipAlias = react.useMemo(() => renderElement({
5710
+ const renderTooltipAlias = React.useMemo(() => renderElement({
5592
5711
  id: "alias",
5593
5712
  wrap: false
5594
5713
  }), [renderElement]);
5595
- const renderTooltipValue = react.useMemo(() => renderElement({
5714
+ const renderTooltipValue = React.useMemo(() => renderElement({
5596
5715
  id: "value",
5597
5716
  wrap: false
5598
5717
  }), [renderElement]);
5599
- const renderAlias = react.useMemo(() => renderElement({ id: "alias" }), [renderElement]);
5600
- const renderValue = react.useMemo(() => renderElement({ id: "value" }), [renderElement]);
5601
- const renderIcon = react.useMemo(() => renderElement({
5718
+ const renderAlias = React.useMemo(() => renderElement({ id: "alias" }), [renderElement]);
5719
+ const renderValue = React.useMemo(() => renderElement({ id: "value" }), [renderElement]);
5720
+ const renderIcon = React.useMemo(() => renderElement({
5602
5721
  id: "icon",
5603
5722
  wrap: false
5604
5723
  }), [renderElement]);
5605
- const renderTooltip = react.useMemo(() => (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [jsxRuntime.jsx(ProgressTooltipAlias, { children: renderTooltipAlias }), jsxRuntime.jsxs(ProgressTooltipValueContainer, { children: [jsxRuntime.jsx(ProgressTooltipValue, { children: renderTooltipValue }), jsxRuntime.jsx(ProgressTooltipValueOf, { children: t("of", { ns: "dashboard" }) }), jsxRuntime.jsx("div", { children: formatAttributeValue({ t, type: valueType, value: currentMaxValue, stringFormat }) })] })] })), [currentMaxValue, renderTooltipAlias, renderTooltipValue, stringFormat, t, valueType]);
5606
- const color = react.useMemo(() => colorAttribute
5724
+ const renderTooltip = React.useMemo(() => (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [jsxRuntime.jsx(ProgressTooltipAlias, { children: renderTooltipAlias }), jsxRuntime.jsxs(ProgressTooltipValueContainer, { children: [jsxRuntime.jsx(ProgressTooltipValue, { children: renderTooltipValue }), jsxRuntime.jsx(ProgressTooltipValueOf, { children: t("of", { ns: "dashboard" }) }), jsxRuntime.jsx("div", { children: formatAttributeValue({ t, type: valueType, value: currentMaxValue, stringFormat }) })] })] })), [currentMaxValue, renderTooltipAlias, renderTooltipValue, stringFormat, t, valueType]);
5725
+ const color = React.useMemo(() => colorAttribute
5607
5726
  ? attributes?.find(({ name }) => name === colorAttribute)?.value
5608
5727
  : colors?.[index] || bgColor, [attributes, bgColor, colorAttribute, colors, index]);
5609
5728
  return (jsxRuntime.jsx(uilibGl.Tooltip, { content: renderTooltip, placement: "top", arrow: true, children: ref => (jsxRuntime.jsxs(ProgressContainerWrapper, { ref: ref, style: innerTemplateStyle || style, children: [renderIcon && jsxRuntime.jsx(ProgressIcon, { children: renderIcon }), jsxRuntime.jsxs(ProgressContent, { children: [!hideTitle && (jsxRuntime.jsxs(ProgressAlias, { children: [jsxRuntime.jsx("div", { children: renderAlias }), jsxRuntime.jsxs(ProgressValue, { children: [renderValue, unitsElement && jsxRuntime.jsx(ProgressUnits, { children: renderElement({ id: "units" }) })] })] })), jsxRuntime.jsxs(ProgressBarContainer, { innerValue: innerValue, children: [jsxRuntime.jsx(ProgressBarWrapper, { children: jsxRuntime.jsx(ProgressBar, { "$width": width, "$color": color }) }), !!(hideTitle || innerValue) && jsxRuntime.jsx(ProgressInnerValue, { children: renderValue })] })] })] })) }));
@@ -5702,7 +5821,6 @@ const AnyChartWrapper = styled.div `
5702
5821
  `;
5703
5822
  const BarChartWrapper = styled(AnyChartWrapper) `
5704
5823
  width: 100%;
5705
- max-width: ${MAX_CHART_WIDTH}px;
5706
5824
  margin: 0 auto ${BAR_CHART_FOOTER_MARGIN}px;
5707
5825
 
5708
5826
  :hover {
@@ -5787,29 +5905,29 @@ const TooltipContainer = styled.div `
5787
5905
  }
5788
5906
  `;
5789
5907
 
5790
- const FiltersContainer = react.memo(({ elementConfig, config, type, renderElement }) => {
5908
+ const FiltersContainer = React.memo(({ elementConfig, config, type, renderElement }) => {
5791
5909
  const { filters, dataSources, expandedContainers } = useWidgetContext(type);
5792
5910
  const { currentPage } = useWidgetPage(type);
5793
5911
  const { filters: configFilters } = currentPage;
5794
5912
  const { id, style, options } = elementConfig || {};
5795
5913
  const { padding, bgColor, fontColor, fontSize, expandable, expanded } = options || {};
5796
- const isLoading = react.useMemo(() => checkIsLoading(dataSources, config, configFilters), [configFilters, config, dataSources]);
5797
- const filterItems = react.useMemo(() => elementConfig?.children?.filter(child => child.options?.filterName), [elementConfig?.children]);
5798
- const renderFilter = react.useCallback((filter, index) => {
5914
+ const isLoading = React.useMemo(() => checkIsLoading(dataSources, config, configFilters), [configFilters, config, dataSources]);
5915
+ const filterItems = React.useMemo(() => elementConfig?.children?.filter(child => child.options?.filterName), [elementConfig?.children]);
5916
+ const renderFilter = React.useCallback((filter, index) => {
5799
5917
  const FilterComponent = getFilterComponent(filter.type);
5800
5918
  return jsxRuntime.jsx(FilterComponent, { type: type, filter: filter, config: config, elementConfig: filter }, index);
5801
5919
  }, [config, type]);
5802
5920
  const isVisible = isVisibleContainer(id, expandable, expanded, expandedContainers);
5803
- const selectedItems = react.useMemo(() => getFilterSelectedItems(filterItems, filters, configFilters), [configFilters, filters, filterItems]);
5921
+ const selectedItems = React.useMemo(() => getFilterSelectedItems(filterItems, filters, configFilters), [configFilters, filters, filterItems]);
5804
5922
  return (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [jsxRuntime.jsx(uilibGl.Flex, { mb: !isVisible && selectedItems.length ? "2rem" : 0, children: jsxRuntime.jsx(ExpandableTitle, { elementConfig: elementConfig, type: type, renderElement: renderElement }) }), isLoading && jsxRuntime.jsx(ContainerLoading, {}), !isLoading && isVisible && (jsxRuntime.jsx(FiltersContainerWrapper, { style: style, "$padding": padding, "$bgColor": bgColor, "$fontSize": fontSize, "$fontColor": fontColor, children: filterItems?.map(renderFilter) })), jsxRuntime.jsx(HiddenTitleItems, { elementConfig: elementConfig, config: config, type: type, filter: filterItems[0].options?.filterName })] }));
5805
5923
  });
5806
5924
 
5807
- const DefaultAttributesContainer = react.memo(({ type, renderElement }) => {
5925
+ const DefaultAttributesContainer = React.memo(({ type, renderElement }) => {
5808
5926
  const { attributes } = useWidgetContext(type);
5809
5927
  return (jsxRuntime.jsx(jsxRuntime.Fragment, { children: attributes?.map(({ name }) => (jsxRuntime.jsx(Container, { children: renderElement({ id: name }) }, name))) }));
5810
5928
  });
5811
5929
 
5812
- const ChartContainer = react.memo(({ elementConfig, isVisible, type, renderElement }) => {
5930
+ const ChartContainer = React.memo(({ elementConfig, isVisible, type, renderElement }) => {
5813
5931
  const { options, children } = elementConfig || {};
5814
5932
  const { twoColumns, hideEmpty } = options || {};
5815
5933
  const aliasElement = children.find(({ id }) => id === "alias");
@@ -5827,17 +5945,17 @@ const ChartContainer = react.memo(({ elementConfig, isVisible, type, renderEleme
5827
5945
  return (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [jsxRuntime.jsx(ExpandableTitle, { elementConfig: elementConfig, type: type, renderElement: renderElement }), isVisible && (jsxRuntime.jsxs(Container, { isColumn: true, children: [aliasElement && jsxRuntime.jsx(ContainerAlias, { hasBottomMargin: true, children: renderElement({ id: "alias" }) }), jsxRuntime.jsx(ContainerValue, { column: !twoColumns, alignItems: "center", children: hasItems ? (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [jsxRuntime.jsx(ContainerChart, { children: renderElement({ id: "chart" }) }), jsxRuntime.jsx(ContainerLegend, { children: renderElement({ id: "legend" }) })] })) : (jsxRuntime.jsx(jsxRuntime.Fragment, { children: "\u2014" })) })] }))] }));
5828
5946
  });
5829
5947
 
5830
- const PagesContainer = react.memo(({ type }) => {
5948
+ const PagesContainer = React.memo(({ type }) => {
5831
5949
  const { config } = useWidgetConfig(type);
5832
5950
  const { pageIndex, currentPage } = useWidgetPage(type);
5833
5951
  const { selectedTabId, setSelectedTabId, expandedContainers, attributes } = useWidgetContext(type);
5834
5952
  const { options } = config || {};
5835
- const { column } = options || {};
5953
+ const { column, width } = options || {};
5836
5954
  const isColumn = column === undefined || column;
5837
- const filteredChildren = react.useMemo(() => !selectedTabId
5955
+ const filteredChildren = React.useMemo(() => !selectedTabId
5838
5956
  ? currentPage.children
5839
5957
  : currentPage.children.filter(item => !item.options?.tabId || item.options.tabId === selectedTabId), [currentPage.children, selectedTabId]);
5840
- const renderElement = react.useMemo(() => getRenderElement({
5958
+ const renderElement = React.useMemo(() => getRenderElement({
5841
5959
  type,
5842
5960
  config,
5843
5961
  elementConfig: currentPage,
@@ -5857,14 +5975,14 @@ const PagesContainer = react.memo(({ type }) => {
5857
5975
  setSelectedTabId,
5858
5976
  type,
5859
5977
  ]);
5860
- return (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [jsxRuntime.jsx(ExpandableTitle, { elementConfig: config, type: type, renderElement: renderElement }), jsxRuntime.jsx(Container, { isColumn: isColumn, isMain: true, children: jsxRuntime.jsx(ContainerChildren, { items: filteredChildren, isMain: true, renderElement: renderElement }) })] }));
5978
+ return (jsxRuntime.jsxs("div", { style: { width }, children: [jsxRuntime.jsx(ExpandableTitle, { elementConfig: config, type: type, renderElement: renderElement }), jsxRuntime.jsx(Container, { isColumn: isColumn, isMain: true, children: jsxRuntime.jsx(ContainerChildren, { items: filteredChildren, isMain: true, renderElement: renderElement }) })] }));
5861
5979
  });
5862
5980
 
5863
- const TwoColumnsInnerContainer = react.memo(({ renderElement }) => {
5981
+ const TwoColumnsInnerContainer = React.memo(({ renderElement }) => {
5864
5982
  return (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [jsxRuntime.jsxs(TwoColumnContainerWrapper, { children: [jsxRuntime.jsxs(Container, { children: [jsxRuntime.jsx(ContainerAlias, { children: renderElement({ id: "firstAlias" }) }), jsxRuntime.jsx(ContainerValue, { big: true, children: renderElement({ id: "firstValue" }) })] }), jsxRuntime.jsxs(Container, { children: [jsxRuntime.jsx(ContainerAlias, { children: renderElement({ id: "secondAlias" }) }), jsxRuntime.jsx(ContainerValue, { big: true, children: renderElement({ id: "secondValue" }) })] })] }), renderElement({ id: "thirdContainer" })] }));
5865
5983
  });
5866
5984
 
5867
- const ImageContainerBg = styled.div `
5985
+ const ImageContainerBg$1 = styled.div `
5868
5986
  position: absolute;
5869
5987
  top: 0;
5870
5988
  bottom: 0;
@@ -5924,9 +6042,9 @@ const ImageContainerWrapper = styled(uilibGl.Flex) `
5924
6042
  }
5925
6043
  `;
5926
6044
 
5927
- const ImageContainer = react.memo(({ elementConfig, renderElement }) => {
6045
+ const ImageContainer = React.memo(({ elementConfig, renderElement }) => {
5928
6046
  const { style } = elementConfig || {};
5929
- return (jsxRuntime.jsxs(ImageContainerWrapper, { style: style, children: [jsxRuntime.jsx(ImageContainerTitle, { children: renderElement({ id: "alias" }) }), jsxRuntime.jsx(ImageContainerText, { children: renderElement({ id: "text" }) }), renderElement({ id: "button" }), jsxRuntime.jsx(ImageContainerBg, { children: renderElement({ id: "image" }) })] }));
6047
+ return (jsxRuntime.jsxs(ImageContainerWrapper, { style: style, children: [jsxRuntime.jsx(ImageContainerTitle, { children: renderElement({ id: "alias" }) }), jsxRuntime.jsx(ImageContainerText, { children: renderElement({ id: "text" }) }), renderElement({ id: "button" }), jsxRuntime.jsx(ImageContainerBg$1, { children: renderElement({ id: "image" }) })] }));
5930
6048
  });
5931
6049
 
5932
6050
  const IconContainerWrapper = styled(uilibGl.Flex) `
@@ -5977,16 +6095,16 @@ const IconContainerText = styled.div `
5977
6095
  color: ${({ theme: { palette } }) => palette.textSecondary};
5978
6096
  `;
5979
6097
 
5980
- const IconContainer = react.memo(({ elementConfig, renderElement }) => {
6098
+ const IconContainer = React.memo(({ elementConfig, renderElement }) => {
5981
6099
  return (jsxRuntime.jsxs(IconContainerWrapper, { style: elementConfig?.style, children: [jsxRuntime.jsxs(IconContainerHeaderWrapper, { children: [jsxRuntime.jsxs(IconContainerHeader, { children: [renderElement({ id: "icon" }), jsxRuntime.jsx(IconContainerTitle, { children: renderElement({ id: "alias" }) })] }), renderElement({ id: "link" })] }), jsxRuntime.jsx(IconContainerText, { children: renderElement({ id: "text" }) })] }));
5982
6100
  });
5983
6101
 
5984
- const DataSourceContainer = react.memo(({ config, elementConfig, type, innerComponent, renderElement }) => {
6102
+ const DataSourceContainer = React.memo(({ config, elementConfig, type, innerComponent, renderElement }) => {
5985
6103
  const { dataSources, expandedContainers } = useWidgetContext(type);
5986
6104
  const { id, style, options } = elementConfig || {};
5987
6105
  const { column = true, relatedDataSource, expandable, expanded } = options || {};
5988
6106
  const isVisible = isVisibleContainer(id, expandable, expanded, expandedContainers);
5989
- const dataSource = react.useMemo(() => dataSources?.find(({ name }) => name === relatedDataSource), [dataSources, relatedDataSource]);
6107
+ const dataSource = React.useMemo(() => dataSources?.find(({ name }) => name === relatedDataSource), [dataSources, relatedDataSource]);
5990
6108
  if (!relatedDataSource)
5991
6109
  return null;
5992
6110
  if (dataSource && !dataSource.features) {
@@ -6075,26 +6193,26 @@ const ContainerTitle = styled(uilibGl.Flex) `
6075
6193
  `}
6076
6194
  `;
6077
6195
 
6078
- const TitleContainer = react.memo(({ containerId, templateName, layerNames, fontColor, expandable, expanded, isVisible, elementConfig, renderElement, type, }) => {
6196
+ const TitleContainer = React.memo(({ containerId, templateName, layerNames, fontColor, expandable, expanded, isVisible, elementConfig, renderElement, type, }) => {
6079
6197
  const { expandContainer, expandedContainers, setDashboardLayer } = useWidgetContext(type);
6080
6198
  const [layersVisibility, toggleVisibility] = useToggle(true);
6081
6199
  const { palette } = styled.useTheme();
6082
6200
  const { style, options } = elementConfig || {};
6083
6201
  const { simple } = options || {};
6084
6202
  const isLayers = templateName === exports.ContainerTemplate.Layers;
6085
- const onClick = react.useCallback(() => {
6203
+ const onClick = React.useCallback(() => {
6086
6204
  if (!expandable)
6087
6205
  return;
6088
6206
  expandContainer(containerId, expanded);
6089
6207
  }, [containerId, expandContainer, expandable, expanded]);
6090
- const onToggleVisibility = react.useCallback(() => {
6208
+ const onToggleVisibility = React.useCallback(() => {
6091
6209
  toggleVisibility();
6092
6210
  layerNames?.forEach(layerName => {
6093
6211
  setDashboardLayer({ name: layerName, isVisible: !layersVisibility });
6094
6212
  });
6095
6213
  }, [layerNames, layersVisibility, setDashboardLayer, toggleVisibility]);
6096
- const renderVisibility = react.useMemo(() => isLayers && (jsxRuntime.jsx(uilibGl.FlexSpan, { mr: "-1rem", children: jsxRuntime.jsx(uilibGl.IconToggle, { kind: layersVisibility ? "password_show" : "password_hide", className: "feature-visible", isSelected: layersVisibility, onClick: onToggleVisibility }) })), [isLayers, layersVisibility, onToggleVisibility]);
6097
- const renderToggler = react.useMemo(() => !!containerId &&
6214
+ const renderVisibility = React.useMemo(() => isLayers && (jsxRuntime.jsx(uilibGl.FlexSpan, { mr: "-1rem", children: jsxRuntime.jsx(uilibGl.IconToggle, { kind: layersVisibility ? "password_show" : "password_hide", className: "feature-visible", isSelected: layersVisibility, onClick: onToggleVisibility }) })), [isLayers, layersVisibility, onToggleVisibility]);
6215
+ const renderToggler = React.useMemo(() => !!containerId &&
6098
6216
  expandable && (jsxRuntime.jsx("div", { children: jsxRuntime.jsx(uilibGl.LegendToggler, { color: palette.icon, toggled: expandedContainers?.[containerId] !== undefined ? expandedContainers[containerId] : expanded, onClick: onClick }) })), [containerId, expandable, expanded, expandedContainers, onClick, palette.icon]);
6099
6217
  return (jsxRuntime.jsx(Container, { isTitle: isVisible, style: style, children: jsxRuntime.jsxs(ContainerTitle, { simple: simple, children: [jsxRuntime.jsxs(ContainerIconTitle, { fontColor: fontColor, children: [renderElement({ id: "titleIcon", wrap: false }), renderElement({ id: "title" }), isLayers && renderToggler] }), !isLayers && renderToggler, renderVisibility] }) }));
6100
6218
  });
@@ -6104,11 +6222,11 @@ const ContainerDivider = styled(uilibGl.Divider) `
6104
6222
  border-color: ${({ theme: { palette }, $bgColor }) => $bgColor || palette.elementDeep};
6105
6223
  `;
6106
6224
 
6107
- const DividerContainer = react.memo(({ elementConfig, config }) => {
6225
+ const DividerContainer = React.memo(({ elementConfig, config }) => {
6108
6226
  return (jsxRuntime.jsx(Container, { style: elementConfig?.style, children: jsxRuntime.jsx(ContainerDivider, { "$bgColor": config?.options?.bgColor }) }));
6109
6227
  });
6110
6228
 
6111
- const SlideshowContainer = react.memo(({ config, elementConfig, type }) => {
6229
+ const SlideshowContainer = React.memo(({ config, elementConfig, type }) => {
6112
6230
  const { pageIndex, expandedContainers, selectedTabId, setSelectedTabId, dataSources } = useWidgetContext(type);
6113
6231
  const { attributes, layerInfo } = useRelatedDataSourceAttributes({
6114
6232
  type,
@@ -6118,7 +6236,7 @@ const SlideshowContainer = react.memo(({ config, elementConfig, type }) => {
6118
6236
  const { id, options } = elementConfig || {};
6119
6237
  const { expandable, expanded } = options || {};
6120
6238
  const isVisible = isVisibleContainer(id, expandable, expanded, expandedContainers);
6121
- const render = react.useMemo(() => getRenderElement({
6239
+ const render = React.useMemo(() => getRenderElement({
6122
6240
  config,
6123
6241
  elementConfig,
6124
6242
  attributes,
@@ -6146,7 +6264,7 @@ const SlideshowContainer = react.memo(({ config, elementConfig, type }) => {
6146
6264
  }) })] }))] }));
6147
6265
  });
6148
6266
 
6149
- const CameraContainer = react.memo(({ elementConfig, type, renderElement }) => {
6267
+ const CameraContainer = React.memo(({ elementConfig, type, renderElement }) => {
6150
6268
  const { expandedContainers } = useWidgetContext(type);
6151
6269
  const { id, options, style } = elementConfig || {};
6152
6270
  const { expandable, expanded } = options || {};
@@ -6232,30 +6350,31 @@ const SwiperContainer = styled.div `
6232
6350
  }
6233
6351
  `;
6234
6352
 
6235
- const TabsContainer = react.memo(({ elementConfig, type }) => {
6353
+ const TabsContainer = React.memo(({ elementConfig, type }) => {
6236
6354
  const { palette } = styled.useTheme();
6237
6355
  const { selectedTabId, setSelectedTabId } = useWidgetContext(type);
6238
6356
  const { options, style, children: tabs } = elementConfig || {};
6239
6357
  const { radius, column, bgColor, noBg, onlyIcon, shownItems, maxLength = 12 } = options || {};
6240
- const renderIcon = react.useCallback((icon, active) => {
6358
+ const renderIcon = React.useCallback((icon, active) => {
6241
6359
  if (!icon)
6242
6360
  return null;
6243
6361
  if (!icon.includes("."))
6244
6362
  return jsxRuntime.jsx(uilibGl.Icon, { kind: icon });
6245
6363
  return icon.endsWith(".svg") ? (jsxRuntime.jsx(SvgImage, { url: icon, width: 16, fontColor: active ? palette.textContrast : palette.textSecondary })) : (jsxRuntime.jsx("img", { src: icon, alt: "" }));
6246
6364
  }, [palette.textContrast, palette.textSecondary]);
6247
- const onClick = react.useCallback((id) => {
6365
+ const onClick = React.useCallback((id) => {
6248
6366
  setSelectedTabId(id);
6249
6367
  window.location.hash = `#${id}`;
6250
6368
  }, [setSelectedTabId]);
6251
- react.useEffect(() => {
6369
+ React.useEffect(() => {
6252
6370
  if (!selectedTabId) {
6253
6371
  setSelectedTabId(tabs[0].id);
6254
6372
  }
6255
6373
  }, []);
6256
- return (jsxRuntime.jsx(SwiperContainer, { style: style, children: jsxRuntime.jsx(react$1.Swiper, { spaceBetween: 0, slidesPerView: shownItems || 2, children: tabs.map(({ id, value, options: tabOptions }) => (jsxRuntime.jsxs(react$1.SwiperSlide, { children: [jsxRuntime.jsxs(TabContainer, { href: `#${id}`, active: selectedTabId === id, column: column, bgColor: bgColor, noBg: noBg, radius: radius, onlyIcon: onlyIcon, hasIcon: !!tabOptions?.icon, onClick: () => onClick(id), children: [renderIcon(tabOptions?.icon, selectedTabId === id), !onlyIcon && (jsxRuntime.jsx(TabValue, { children: jsxRuntime.jsx(TextTrim, { maxLength: maxLength, children: value }) }))] }), jsxRuntime.jsx(TabAnchor, { id: id })] }, id))) }) }));
6374
+ return (jsxRuntime.jsx(SwiperContainer, { style: style, children: jsxRuntime.jsx(react.Swiper, { spaceBetween: 0, slidesPerView: shownItems || 2, children: tabs.map(({ id, value, options: tabOptions }) => (jsxRuntime.jsxs(react.SwiperSlide, { children: [jsxRuntime.jsxs(TabContainer, { href: `#${id}`, active: selectedTabId === id, column: column, bgColor: bgColor, noBg: noBg, radius: radius, onlyIcon: onlyIcon, hasIcon: !!tabOptions?.icon, onClick: () => onClick(id), children: [renderIcon(tabOptions?.icon, selectedTabId === id), !onlyIcon && (jsxRuntime.jsx(TabValue, { children: jsxRuntime.jsx(TextTrim, { maxLength: maxLength, children: value }) }))] }), jsxRuntime.jsx(TabAnchor, { id: id })] }, id))) }) }));
6257
6375
  });
6258
6376
 
6377
+ const ContainerIconValue = styled(uilibGl.Flex) ``;
6259
6378
  const RoundedBackgroundContainerWrapper = styled(uilibGl.Flex) `
6260
6379
  position: relative;
6261
6380
  flex-direction: ${({ $bigIcon }) => ($bigIcon ? "row" : "column")};
@@ -6273,13 +6392,10 @@ const RoundedBackgroundContainerWrapper = styled(uilibGl.Flex) `
6273
6392
  styled.css `
6274
6393
  align-items: center;
6275
6394
 
6276
- * {
6395
+ > * {
6277
6396
  display: flex;
6278
6397
  justify-content: center;
6279
6398
  text-align: center;
6280
- }
6281
-
6282
- > * {
6283
6399
  width: 100%;
6284
6400
  }
6285
6401
  `};
@@ -6312,8 +6428,32 @@ const RoundedBackgroundContainerWrapper = styled(uilibGl.Flex) `
6312
6428
  }
6313
6429
  `};
6314
6430
 
6431
+ ${ContainerIconValue} {
6432
+ align-items: center;
6433
+ flex-direction: column;
6434
+
6435
+ ${({ $big }) => $big &&
6436
+ styled.css `
6437
+ flex-direction: row;
6438
+ margin-bottom: 0.5rem;
6439
+
6440
+ > * {
6441
+ text-align: left;
6442
+ }
6443
+
6444
+ span[kind] {
6445
+ margin-right: 0.5rem;
6446
+ }
6447
+
6448
+ ${ContainerValue} {
6449
+ width: auto;
6450
+ }
6451
+ `};
6452
+ }
6453
+
6315
6454
  ${ContainerValue} {
6316
6455
  flex-direction: ${({ $inlineUnits }) => ($inlineUnits ? "row" : "column")};
6456
+ justify-content: ${({ $big }) => ($big ? "flex-start" : "flex-end")};
6317
6457
  align-items: ${({ $inlineUnits }) => ($inlineUnits ? "center" : "flex-start")};
6318
6458
  line-height: 1;
6319
6459
  }
@@ -6328,7 +6468,7 @@ const RoundedBackgroundContainerWrapper = styled(uilibGl.Flex) `
6328
6468
  `;
6329
6469
 
6330
6470
  const ALIAS_DEFAULT_MAX_LENGTH = 28;
6331
- const RoundedBackgroundContainer = react.memo(({ type, elementConfig, feature, renderElement }) => {
6471
+ const RoundedBackgroundContainer = React.memo(({ type, elementConfig, feature, renderElement }) => {
6332
6472
  const { dataSources } = useWidgetContext(type);
6333
6473
  const { attributes } = useRelatedDataSourceAttributes({
6334
6474
  type,
@@ -6337,18 +6477,20 @@ const RoundedBackgroundContainer = react.memo(({ type, elementConfig, feature, r
6337
6477
  feature
6338
6478
  });
6339
6479
  const { options, style, children } = elementConfig || {};
6340
- const { maxLength, center, fontColor, innerTemplateStyle, inlineUnits, bigIcon, hideEmpty, colorAttribute } = options || {};
6480
+ const { maxLength, center, fontColor, innerTemplateStyle, inlineUnits, big, bigIcon, hideEmpty, colorAttribute } = options || {};
6341
6481
  const aliasElement = children?.find(({ id }) => id === "alias");
6342
6482
  const unitsElement = children?.find(({ id }) => id === "units");
6343
6483
  const valueElement = children?.find(({ id }) => id === "value");
6344
6484
  const value = renderElement({ id: "value" });
6345
- const color = react.useMemo(() => attributes?.find(({ name }) => name === colorAttribute)?.value || fontColor, [attributes, colorAttribute, fontColor]);
6485
+ const color = React.useMemo(() => attributes?.find(({ name }) => name === colorAttribute)?.value || fontColor, [attributes, colorAttribute, fontColor]);
6486
+ const renderAlias = React.useMemo(() => (jsxRuntime.jsx(ContainerAlias, { style: aliasElement?.style, children: jsxRuntime.jsx(TextTrim, { maxLength: maxLength || ALIAS_DEFAULT_MAX_LENGTH, children: renderElement({ id: "alias", wrap: false }) }) })), [aliasElement?.style, maxLength, renderElement]);
6487
+ const renderValue = React.useMemo(() => (jsxRuntime.jsxs(ContainerValue, { style: valueElement?.style, big: true, children: [value, !!unitsElement && (jsxRuntime.jsx(ContainerUnits, { style: unitsElement?.style, children: renderElement({ id: "units" }) }))] })), [valueElement?.style, value, unitsElement, renderElement]);
6346
6488
  if (!value && hideEmpty)
6347
6489
  return null;
6348
- return (jsxRuntime.jsxs(RoundedBackgroundContainerWrapper, { style: innerTemplateStyle || style, "$center": center, "$color": color, "$inlineUnits": inlineUnits, "$bigIcon": bigIcon, children: [renderElement({ id: "icon", wrap: false }), jsxRuntime.jsxs(ContainerValue, { style: valueElement?.style, big: true, children: [value, !!unitsElement && (jsxRuntime.jsx(ContainerUnits, { style: unitsElement?.style, children: renderElement({ id: "units" }) }))] }), jsxRuntime.jsx(ContainerAlias, { style: aliasElement?.style, children: jsxRuntime.jsx(TextTrim, { maxLength: maxLength || ALIAS_DEFAULT_MAX_LENGTH, children: renderElement({ id: "alias", wrap: false }) }) })] }));
6490
+ return (jsxRuntime.jsxs(RoundedBackgroundContainerWrapper, { style: innerTemplateStyle || style, "$center": center, "$color": color, "$inlineUnits": inlineUnits, "$big": big, "$bigIcon": bigIcon, children: [jsxRuntime.jsxs(ContainerIconValue, { children: [renderElement({ id: "icon", wrap: false }), big ? renderAlias : renderValue] }), big ? renderValue : renderAlias] }));
6349
6491
  });
6350
6492
 
6351
- const AddFeatureContainer = react.memo(({ elementConfig }) => {
6493
+ const AddFeatureContainer = React.memo(({ elementConfig }) => {
6352
6494
  const { children, style } = elementConfig || {};
6353
6495
  return (jsxRuntime.jsx(Container, { style: style, children: children
6354
6496
  .filter(({ type }) => type === "button")
@@ -6371,21 +6513,20 @@ const LayersContainerWrapper = styled(Container) `
6371
6513
  }
6372
6514
  `;
6373
6515
 
6374
- const LayersContainer = react.memo(({ type, elementConfig, renderElement }) => {
6375
- const { LayerItemComponent } = useGlobalContext();
6516
+ const LayersContainer = React.memo(({ type, elementConfig, renderElement }) => {
6376
6517
  const { expandedContainers } = useWidgetContext(type);
6377
6518
  const { currentPage } = useWidgetPage(type);
6378
6519
  const { id, options, style } = elementConfig || {};
6379
6520
  const { layerNames, expandable, expanded } = options || {};
6380
6521
  const isVisible = isVisibleContainer(id, expandable, expanded, expandedContainers);
6381
- const layers = react.useMemo(() => {
6522
+ const layers = React.useMemo(() => {
6382
6523
  if (!currentPage?.layers)
6383
6524
  return [];
6384
6525
  if (!layerNames?.length)
6385
6526
  return currentPage.layers;
6386
6527
  return currentPage.layers.filter(({ name }) => layerNames.includes(name));
6387
6528
  }, [currentPage?.layers, layerNames]);
6388
- return (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [jsxRuntime.jsx(ExpandableTitle, { elementConfig: elementConfig, type: type, renderElement: renderElement }), isVisible && (jsxRuntime.jsx(LayersContainerWrapper, { style: style, children: jsxRuntime.jsx(LayerTree, { layers: layers, LayerItemComponent: LayerItemComponent, onlyMainTools: true }) }))] }));
6529
+ return (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [jsxRuntime.jsx(ExpandableTitle, { elementConfig: elementConfig, type: type, renderElement: renderElement }), isVisible && (jsxRuntime.jsx(LayersContainerWrapper, { style: style, children: jsxRuntime.jsx(LayerTree, { layers: layers, onlyMainTools: true }) }))] }));
6389
6530
  });
6390
6531
 
6391
6532
  const containerComponents = {
@@ -6414,157 +6555,480 @@ const containerComponents = {
6414
6555
 
6415
6556
  const getContainerComponent = (innerTemplateName) => innerTemplateName ? containerComponents[innerTemplateName] || containerComponents.default : null;
6416
6557
 
6417
- const getDataFromAttributes = (t, config, attributes) => {
6418
- const colors = config?.options?.colors || FEATURE_CARD_DEFAULT_COLORS;
6419
- const data = config?.children?.map(({ attributeName }, index) => {
6420
- const attribute = attributes?.find(({ name }) => name === attributeName);
6421
- return {
6422
- name: attribute?.name || "",
6423
- value: attribute ? Number(attribute.value) : 0,
6424
- color: colors[index] || FEATURE_CARD_OTHER_COLOR
6425
- };
6426
- });
6427
- if (config?.options?.orderByValue) {
6428
- data.sort((a, b) => b.value - a.value);
6429
- }
6430
- const isOtherSliced = config?.options?.otherItems && config.options.otherItems < data.length;
6431
- if (!isOtherSliced)
6432
- return data;
6433
- const otherValue = data
6434
- .slice(0, config?.options?.otherItems)
6435
- .reduce((prev, { value }) => prev + value, 0);
6436
- return [
6437
- ...data?.slice(0, config.options?.otherItems),
6438
- {
6439
- name: t("other", { ns: "dashboard" }),
6440
- color: FEATURE_CARD_OTHER_COLOR,
6441
- value: Number(otherValue)
6442
- }
6443
- ];
6444
- };
6445
-
6446
- const getDataFromRelatedFeatures = ({ t, config, filters, relatedConfig, dataSource, layerInfo }) => {
6447
- if (!dataSource || !dataSource?.features?.length) {
6448
- return [];
6449
- }
6450
- const { colors } = config.options;
6451
- const layerDefinition = getLayerDefinition(layerInfo);
6452
- let data = JSON.parse(JSON.stringify(dataSource?.features));
6453
- const sortAttribute = config?.options?.orderByValue
6454
- ? relatedConfig.attributeName
6455
- : config?.options?.orderByTitle
6456
- ? relatedConfig.attributeTitle
6457
- : null;
6458
- if (sortAttribute) {
6459
- data.sort((a, b) => b.attributes[sortAttribute] - a.attributes[sortAttribute]);
6460
- }
6461
- const isOtherSliced = config?.options?.otherItems && config.options.otherItems < data.length;
6462
- const otherValue = isOtherSliced
6463
- ? data
6464
- .slice(config.options.otherItems)
6465
- .reduce((prev, { attributes }) => prev + attributes[relatedConfig.attributeName], 0)
6466
- : null;
6467
- if (isOtherSliced) {
6468
- data = data.slice(0, config?.options?.otherItems);
6469
- }
6470
- const gradientArray = relatedConfig.chartAxis && colors?.length < data.length
6471
- ? new Gradient()
6472
- .setColorGradient(colors[0], colors[colors.length - 1])
6473
- .setMidpoint(data.length)
6474
- .getColors()
6475
- : colors;
6476
- const filter = getConfigFilter(relatedConfig?.filterName, filters);
6477
- const result = data.reduce((acc, feature, index) => {
6478
- if (relatedConfig?.filterName && !filter) {
6479
- return acc;
6480
- }
6481
- const attributeName = feature.attributes[relatedConfig.attributeName];
6482
- const attributeTitle = feature.attributes[relatedConfig.attributeTitle];
6483
- const attributeColor = feature.attributes[relatedConfig.attributeColor];
6484
- return [
6485
- ...acc,
6486
- {
6487
- name: formatAttributeValue({
6488
- t,
6489
- type: layerDefinition.attributes[relatedConfig.attributeTitle]?.type,
6490
- value: attributeTitle,
6491
- stringFormat: layerDefinition.attributes[relatedConfig.attributeTitle]?.stringFormat
6492
- }),
6493
- value: attributeName === null ? null : Number(attributeName),
6494
- color: attributeColor || gradientArray?.[index] || FEATURE_CARD_OTHER_COLOR,
6495
- min: filter?.attributeMin ? feature.attributes[filter.attributeMin] : null,
6496
- max: filter?.attributeMin ? feature.attributes[filter.attributeMax] : null
6497
- }
6498
- ];
6499
- }, []);
6500
- if (isOtherSliced) {
6501
- result.push({
6502
- name: t("other", { ns: "dashboard" }),
6503
- color: FEATURE_CARD_OTHER_COLOR,
6504
- value: Number(otherValue)
6505
- });
6506
- }
6507
- return result;
6508
- };
6509
-
6510
- const getDefaultConfig = ({ title, defaultTitle, items, baseMapName, position, resolution, srid, }) => {
6511
- const dashboardConfiguration = JSON.parse(JSON.stringify(DEFAULT_DASHBOARD_CONFIG));
6512
- const defaultPages = JSON.parse(JSON.stringify(DEFAULT_PAGES_CONFIG));
6513
- const defaultPage = createConfigPage({ title, baseMapName, position, resolution, defaultTitle });
6514
- if (items?.length) {
6515
- const contentItems = items.map(item => createConfigLayer(item.name));
6516
- defaultPage.layers.push(...contentItems);
6517
- }
6518
- defaultPages.children.push(defaultPage);
6519
- dashboardConfiguration.children.push(defaultPages);
6520
- return dashboardConfiguration;
6521
- };
6522
-
6523
- const ElementImage = react.memo(({ type, elementConfig }) => {
6524
- const { attributes } = useWidgetContext(type);
6525
- const { value, attributeName, options } = elementConfig || {};
6526
- const { width } = options || {};
6527
- const firstImage = react.useMemo(() => {
6528
- if (value) {
6529
- return value.toString();
6530
- }
6531
- if (!attributeName || Array.isArray(attributeName)) {
6532
- return null;
6533
- }
6534
- const attribute = attributes?.find(item => item.name === attributeName);
6535
- return attribute?.value?.split(";")?.[0];
6536
- }, [attributeName, attributes, value]);
6537
- return firstImage ? jsxRuntime.jsx("img", { src: firstImage, alt: firstImage, width: width }) : null;
6538
- });
6539
-
6540
- const SmallPreviewContainer$1 = styled.div `
6541
- width: 100%;
6542
- height: 100%;
6543
- `;
6558
+ var img$3 = "";
6544
6559
 
6545
- const AttributeGalleryContainer = styled.div `
6546
- && {
6547
- width: calc(100% + 3rem);
6548
- }
6560
+ const getMaskColor = (isDark) => (isDark ? "0, 0, 0" : "255, 255, 255");
6561
+ const DefaultHeaderContainer = styled(uilibGl.Flex) `
6562
+ flex-direction: column;
6563
+ position: relative;
6564
+ flex-shrink: 0;
6565
+ min-height: 8.375rem;
6566
+ margin-bottom: -1.5rem;
6567
+ padding: 1.5rem 1.5rem 0;
6568
+ border-top-left-radius: 0.5rem;
6569
+ border-top-right-radius: 0.5rem;
6570
+ overflow: hidden;
6549
6571
 
6550
- min-height: 12.625rem;
6551
- background-color: ${({ theme: { palette } }) => palette.element};
6572
+ > * {
6573
+ z-index: 1;
6574
+ }
6552
6575
 
6553
- img {
6576
+ &::before {
6577
+ content: "";
6578
+ position: absolute;
6579
+ top: 0;
6580
+ left: 0;
6554
6581
  width: 100%;
6582
+ height: 100%;
6583
+
6584
+ ${({ image, isDark }) => image
6585
+ ? styled.css `
6586
+ background: url(${image}) 0 0 no-repeat;
6587
+ background-size: cover;
6588
+ `
6589
+ : styled.css `
6590
+ background: url(${img$3}) 50% 0 no-repeat;
6591
+ opacity: ${isDark ? 1 : 0.5};
6592
+ `}
6555
6593
  }
6556
- `;
6557
- const LinearProgressContainer = styled(uilibGl.Flex) `
6558
- align-items: center;
6559
- justify-content: center;
6560
- min-height: inherit;
6561
6594
 
6595
+ ${({ image, isDark }) => image &&
6596
+ styled.css `
6597
+ &::before {
6598
+ -webkit-mask-image: linear-gradient(
6599
+ to bottom,
6600
+ rgba(${getMaskColor(isDark)}, 1),
6601
+ rgba(${getMaskColor(isDark)}, 0)
6602
+ );
6603
+ mask-image: linear-gradient(to bottom, rgba(${getMaskColor(isDark)}, 1), rgba(${getMaskColor(isDark)}, 0));
6604
+ }
6605
+ `}
6562
6606
  ${uilibGl.LinearProgress} {
6563
- max-width: 4rem;
6607
+ position: absolute;
6608
+ top: 0;
6609
+ left: 0;
6564
6610
  }
6565
6611
  `;
6566
- const NoLiveSnapshotContainer = styled(uilibGl.Flex) `
6567
- flex-direction: column;
6612
+ const TopContainer = styled(uilibGl.Flex) `
6613
+ z-index: 1;
6614
+ position: relative;
6615
+ justify-content: space-between;
6616
+ flex-wrap: nowrap;
6617
+ width: 100%;
6618
+ align-items: flex-start;
6619
+ `;
6620
+ const TopContainerButtons = styled(uilibGl.Flex) `
6621
+ align-items: center;
6622
+ width: auto;
6623
+ margin-right: -0.5rem;
6624
+
6625
+ button {
6626
+ width: auto;
6627
+ height: 1rem;
6628
+ padding: 0 0.5rem;
6629
+ }
6630
+ `;
6631
+ const LogoContainer = styled(uilibGl.Flex) `
6632
+ max-width: calc(100% - 1.4rem);
6633
+ flex-grow: 1;
6634
+ font-size: 0;
6635
+
6636
+ & > span::after {
6637
+ font-size: 2rem;
6638
+ }
6639
+
6640
+ img {
6641
+ max-height: 1.875rem;
6642
+ }
6643
+ `;
6644
+ const PageTitle = styled(uilibGl.H2) `
6645
+ display: -webkit-box;
6646
+ -webkit-line-clamp: 3;
6647
+ -webkit-box-orient: vertical;
6648
+ overflow: hidden;
6649
+ margin: 0;
6650
+ font-size: 1.25rem;
6651
+ font-weight: 600;
6652
+ pointer-events: initial;
6653
+ font-family: "Nunito Sans", serif;
6654
+ `;
6655
+
6656
+ exports.ThemeName = void 0;
6657
+ (function (ThemeName) {
6658
+ ThemeName["Light"] = "light";
6659
+ ThemeName["Dark"] = "dark";
6660
+ })(exports.ThemeName || (exports.ThemeName = {}));
6661
+
6662
+ exports.TmsType = void 0;
6663
+ (function (TmsType) {
6664
+ TmsType["WMS"] = "WMS";
6665
+ TmsType["TMS"] = "TMS";
6666
+ TmsType["ArcGIS"] = "ArcGIS";
6667
+ })(exports.TmsType || (exports.TmsType = {}));
6668
+ exports.EditGeometryType = void 0;
6669
+ (function (EditGeometryType) {
6670
+ EditGeometryType["Raster"] = "raster";
6671
+ })(exports.EditGeometryType || (exports.EditGeometryType = {}));
6672
+
6673
+ const DashboardDefaultHeader = React.memo(() => {
6674
+ const { components: { ProjectCatalogMenu, ProjectPanelMenu, ProjectPagesMenu }, } = useWidgetContext();
6675
+ const { pageId, image, icon, tooltip, themeName } = useDashboardHeader();
6676
+ return (jsxRuntime.jsxs(DefaultHeaderContainer, { image: image, isDark: themeName === exports.ThemeName.Dark, children: [!pageId && jsxRuntime.jsx(uilibGl.LinearProgress, {}), jsxRuntime.jsxs(uilibGl.Flex, { column: true, gap: "1rem", children: [jsxRuntime.jsx(uilibGl.FlexSpan, { children: jsxRuntime.jsxs(TopContainer, { children: [jsxRuntime.jsx(LogoContainer, { children: icon }), jsxRuntime.jsxs(TopContainerButtons, { children: [jsxRuntime.jsx(ProjectCatalogMenu, {}), jsxRuntime.jsx(ProjectPanelMenu, {})] })] }) }), jsxRuntime.jsx(uilibGl.FlexSpan, { children: jsxRuntime.jsx(uilibGl.Flex, { column: true, gap: "0.25rem", children: jsxRuntime.jsx(uilibGl.FlexSpan, { children: jsxRuntime.jsxs(uilibGl.Flex, { alignItems: "center", children: [jsxRuntime.jsx(uilibGl.FlexSpan, { flexGrow: 1, children: jsxRuntime.jsx(uilibGl.Tooltip, { arrow: true, content: tooltip, children: ref => (jsxRuntime.jsx(PageTitle, { ref: ref, children: jsxRuntime.jsx(ProjectPagesMenu, {}) })) }) }), jsxRuntime.jsx(uilibGl.FlexSpan, { children: jsxRuntime.jsx(Pagination, {}) })] }) }) }) })] })] }));
6677
+ });
6678
+
6679
+ const HeaderFrontView = styled(uilibGl.Flex) `
6680
+ z-index: 10;
6681
+ position: relative;
6682
+ justify-content: space-between;
6683
+ align-items: ${({ isDefault }) => (isDefault ? "center" : "flex-start")};
6684
+ width: 100%;
6685
+ font: ${({ theme: { fonts } }) => fonts.subtitle};
6686
+ `;
6687
+ const HeaderContainer = styled(uilibGl.FlexSpan) `
6688
+ display: flex;
6689
+ flex-grow: 1;
6690
+ flex-wrap: nowrap;
6691
+ width: calc(100% - 48px);
6692
+ `;
6693
+ const FeatureTitleContainer = styled.div `
6694
+ display: -webkit-box;
6695
+ max-width: 100%;
6696
+ width: 100%;
6697
+ margin: 0.5rem 0;
6698
+ -webkit-line-clamp: 2;
6699
+ -webkit-box-orient: vertical;
6700
+ overflow: hidden;
6701
+ text-overflow: ellipsis;
6702
+ color: ${({ theme: { palette } }) => palette.textPrimary};
6703
+
6704
+ & > ${uilibGl.FlexSpan} {
6705
+ cursor: ${({ clickable }) => clickable && "pointer"};
6706
+
6707
+ &:hover {
6708
+ color: ${({ clickable, theme: { palette } }) => clickable && palette.primary};
6709
+ }
6710
+ }
6711
+ `;
6712
+ const LayerDescription = styled(uilibGl.Description) `
6713
+ width: calc(100% - 4rem);
6714
+ display: -webkit-box;
6715
+ -webkit-line-clamp: 2;
6716
+ -webkit-box-orient: vertical;
6717
+ overflow: hidden;
6718
+ text-overflow: ellipsis;
6719
+ `;
6720
+ const HeaderTitleContainer = styled(uilibGl.Flex) `
6721
+ flex-direction: column;
6722
+ width: 100%;
6723
+ `;
6724
+ const RowHeaderMixin = styled.css `
6725
+ &&& {
6726
+ min-height: auto;
6727
+
6728
+ ${FeatureTitleContainer}, ${LayerDescription} {
6729
+ text-align: left;
6730
+ }
6731
+ }
6732
+
6733
+ ${HeaderContainer} {
6734
+ flex-direction: row;
6735
+ }
6736
+
6737
+ ${FeatureTitleContainer} {
6738
+ max-width: calc(100% - 3.8rem);
6739
+ }
6740
+ `;
6741
+ const Header = styled(uilibGl.Flex) `
6742
+ z-index: 1;
6743
+ position: relative;
6744
+ top: 0;
6745
+ flex-shrink: 0;
6746
+ overflow: hidden;
6747
+ padding: 0.5rem;
6748
+
6749
+ ${HeaderContainer} {
6750
+ flex-direction: column;
6751
+ }
6752
+
6753
+ ${({ $isRow }) => $isRow && RowHeaderMixin};
6754
+ `;
6755
+
6756
+ const HeaderTitle = ({ noFeature }) => {
6757
+ const { t } = useGlobalContext();
6758
+ const { attributes, layerInfo, feature } = useWidgetContext(exports.WidgetType.FeatureCard);
6759
+ const zoomToFeatures = useZoomToFeatures();
6760
+ const { alias, name, layerDefinition } = layerInfo || {};
6761
+ const resultDescription = alias || name || "";
6762
+ const resultTitle = React.useMemo(() => {
6763
+ const layerDefinitionAttribute = layerDefinition && attributes?.length
6764
+ ? attributes.find(item => item.name === layerDefinition.titleAttribute)
6765
+ : null;
6766
+ const layerDefinitionTitle = layerDefinition?.titleAttribute && layerDefinitionAttribute?.value
6767
+ ? layerDefinitionAttribute.value
6768
+ : null;
6769
+ return ((!!layerDefinitionTitle &&
6770
+ formatAttributeValue({
6771
+ t,
6772
+ type: layerDefinitionAttribute.type,
6773
+ value: layerDefinitionTitle,
6774
+ stringFormat: layerDefinitionAttribute.stringFormat,
6775
+ })) ||
6776
+ feature?.id);
6777
+ }, [attributes, feature, layerDefinition]);
6778
+ return (jsxRuntime.jsxs(HeaderTitleContainer, { children: [noFeature ? (jsxRuntime.jsx(FeatureTitleContainer, { children: t("noObjectFound", { ns: "dashboard" }) })) : (jsxRuntime.jsx(FeatureTitleContainer, { clickable: true, children: jsxRuntime.jsx(uilibGl.Tooltip, { arrow: true, placement: "top", content: t("zoomToFeature", { ns: "dashboard" }), delay: [600, 0], children: ref => (jsxRuntime.jsx(uilibGl.FlexSpan, { ref: ref, onClick: () => zoomToFeatures([feature]), children: resultTitle })) }) })), jsxRuntime.jsx(LayerDescription, { title: resultDescription, children: resultDescription })] }));
6779
+ };
6780
+
6781
+ const LayerIconContainer = styled.div `
6782
+ display: flex;
6783
+ align-items: center;
6784
+ margin-right: 0.75rem;
6785
+ `;
6786
+ const AlertIconContainer = styled(uilibGl.Flex) `
6787
+ align-items: center;
6788
+ justify-content: center;
6789
+ width: 2rem;
6790
+ height: 2rem;
6791
+
6792
+ ${uilibGl.Icon} {
6793
+ :after {
6794
+ color: ${({ theme: { palette } }) => palette.error};
6795
+ }
6796
+ }
6797
+ `;
6798
+
6799
+ var img$2 = "data:image/svg+xml,%3csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3e %3crect width='32' height='32' fill='transparent'/%3e %3cpath d='M20.248 9.67787C18.5318 8.04035 15.9473 7.54977 13.7024 8.43543C11.4576 9.32109 9.99566 11.4081 10 13.721C10 18.2894 16 24 16 24C16 24 22 18.2894 22 13.721C22.0028 12.2054 21.3726 10.7509 20.248 9.67787Z' fill='url(%23paint0_linear_6459_10338)'/%3e %3ccircle cx='16' cy='14' r='2' fill='white'/%3e %3cdefs%3e %3clinearGradient id='paint0_linear_6459_10338' x1='10' y1='8' x2='25.36' y2='19.52' gradientUnits='userSpaceOnUse'%3e %3cstop stop-color='%230084D6'/%3e %3cstop offset='0.489583' stop-color='%230084D6'/%3e %3cstop offset='0.489683' stop-color='%2305A9FF'/%3e %3cstop offset='0.921875' stop-color='%2305A9FF'/%3e %3c/linearGradient%3e %3c/defs%3e%3c/svg%3e";
6800
+
6801
+ var img$1 = "data:image/svg+xml,%3csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3e %3crect width='32' height='32' fill='transparent'/%3e %3cpath d='M4 17.9574H10.1538L13.5385 12L18.4615 20L22 13.8723H28' stroke='url(%23paint0_linear_6459_10369)' stroke-width='2' stroke-linejoin='round'/%3e %3cdefs%3e %3clinearGradient id='paint0_linear_6459_10369' x1='4' y1='9.54167' x2='28.7229' y2='20.1484' gradientUnits='userSpaceOnUse'%3e %3cstop offset='0.525942' stop-color='%2306AAFF'/%3e %3cstop offset='0.526042' stop-color='%230184D7'/%3e %3c/linearGradient%3e %3c/defs%3e%3c/svg%3e";
6802
+
6803
+ var img = "data:image/svg+xml,%3csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3e %3crect width='32' height='32' fill='transparent'/%3e %3crect x='8' y='8' width='16' height='16' rx='2' fill='url(%23paint0_linear_6459_10399)'/%3e %3cdefs%3e %3clinearGradient id='paint0_linear_6459_10399' x1='8' y1='8' x2='24' y2='24' gradientUnits='userSpaceOnUse'%3e %3cstop stop-color='%230084D6'/%3e %3cstop offset='0.489583' stop-color='%230084D6'/%3e %3cstop offset='0.489683' stop-color='%2305A9FF'/%3e %3cstop offset='0.921875' stop-color='%2305A9FF'/%3e %3c/linearGradient%3e %3c/defs%3e%3c/svg%3e";
6804
+
6805
+ const LayerIcon = ({ layerInfo }) => {
6806
+ const renderSymbol = React.useMemo(() => {
6807
+ if (!layerInfo.geometryType) {
6808
+ return (jsxRuntime.jsx(AlertIconContainer, { children: jsxRuntime.jsx(uilibGl.Icon, { kind: "warning" }) }));
6809
+ }
6810
+ switch (layerInfo.geometryType) {
6811
+ case api.GeometryType.Polyline:
6812
+ return jsxRuntime.jsx("img", { src: img$1, alt: "" });
6813
+ case api.GeometryType.Polygon:
6814
+ return jsxRuntime.jsx("img", { src: img, alt: "" });
6815
+ case api.GeometryType.Point:
6816
+ default:
6817
+ return jsxRuntime.jsx("img", { src: img$2, alt: "" });
6818
+ }
6819
+ }, [layerInfo.geometryType]);
6820
+ return jsxRuntime.jsx(LayerIconContainer, { children: renderSymbol });
6821
+ };
6822
+
6823
+ const FeatureCardDefaultHeader = ({ noFeature }) => {
6824
+ const { layerInfo } = useWidgetContext(exports.WidgetType.FeatureCard);
6825
+ return (jsxRuntime.jsx(Header, { "$isRow": true, children: jsxRuntime.jsxs(HeaderFrontView, { isDefault: true, children: [jsxRuntime.jsxs(HeaderContainer, { children: [!!layerInfo?.name && jsxRuntime.jsx(LayerIcon, { layerInfo: layerInfo }), jsxRuntime.jsx(HeaderTitle, { noFeature: noFeature })] }), jsxRuntime.jsx(FeatureCardButtons, {})] }) }));
6826
+ };
6827
+
6828
+ const HeaderFontColorMixin$1 = styled.css `
6829
+ ${HeaderTitleContainer}, ${LayerDescription} {
6830
+ color: ${({ $fontColor }) => $fontColor};
6831
+ }
6832
+ `;
6833
+ const HeaderWrapperMixin$1 = styled.css `
6834
+ padding: 0.5rem 0.5rem 0;
6835
+
6836
+ ${Header} {
6837
+ min-height: 5.25rem;
6838
+ }
6839
+
6840
+ ${HeaderContainer} {
6841
+ max-width: 100%;
6842
+ width: 100%;
6843
+ }
6844
+
6845
+ ${FeatureControls} {
6846
+ max-width: calc(100% - 2rem);
6847
+ width: calc(100% - 2rem);
6848
+ margin-top: -0.5rem;
6849
+ padding-top: 1rem;
6850
+ border-radius: ${({ theme: { borderRadius } }) => borderRadius.medium};
6851
+ }
6852
+
6853
+ ${({ $fontColor }) => !!$fontColor && HeaderFontColorMixin$1};
6854
+ `;
6855
+ const GradientHeaderWrapper = styled.div `
6856
+ ${Header} {
6857
+ background: ${({ $bgColor }) => $bgColor || "radial-gradient(129.21% 133.22% at 51.94% 0%, #e8fffe 9.48%, #5fcaff 100%)"};
6858
+ }
6859
+
6860
+ ${HeaderContainer} {
6861
+ align-items: center;
6862
+ }
6863
+
6864
+ ${HeaderTitleContainer} {
6865
+ margin-left: 0;
6866
+ text-align: center;
6867
+ }
6868
+
6869
+ ${FeatureTitleContainer} {
6870
+ text-align: center;
6871
+ }
6872
+
6873
+ ${LayerDescription} {
6874
+ text-align: center;
6875
+ }
6876
+
6877
+ ${HeaderWrapperMixin$1};
6878
+ `;
6879
+
6880
+ const FeatureCardGradientHeader = ({ isRow }) => {
6881
+ const { layerInfo } = useWidgetContext(exports.WidgetType.FeatureCard);
6882
+ const { config } = useWidgetConfig(exports.WidgetType.FeatureCard);
6883
+ const { header, options } = config || {};
6884
+ const renderElement = useHeaderRender(header);
6885
+ return (jsxRuntime.jsx(GradientHeaderWrapper, { "$fontColor": options?.fontColor, "$bgColor": options?.bgColor, children: jsxRuntime.jsx(uilibGl.ThemeProvider, { theme: uilibGl.defaultTheme, children: jsxRuntime.jsx(Header, { "$isRow": isRow, children: jsxRuntime.jsxs(HeaderFrontView, { children: [jsxRuntime.jsxs(HeaderContainer, { children: [jsxRuntime.jsx(LayerIcon, { layerInfo: layerInfo }), jsxRuntime.jsx(FeatureCardTitle, { title: renderElement({
6886
+ id: "title",
6887
+ wrap: false,
6888
+ }), description: renderElement({
6889
+ id: "description",
6890
+ wrap: false,
6891
+ }) })] }), jsxRuntime.jsx(FeatureCardButtons, {})] }) }) }) }));
6892
+ };
6893
+
6894
+ const HeaderFontColorMixin = styled.css `
6895
+ ${HeaderTitleContainer}, ${LayerDescription} {
6896
+ color: ${({ $fontColor }) => $fontColor};
6897
+ }
6898
+ `;
6899
+ const HeaderWrapperMixin = styled.css `
6900
+ padding: 0.5rem 0.5rem 0;
6901
+
6902
+ ${Header} {
6903
+ min-height: 5.25rem;
6904
+ }
6905
+
6906
+ ${HeaderContainer} {
6907
+ max-width: 100%;
6908
+ width: 100%;
6909
+ }
6910
+
6911
+ ${FeatureControls} {
6912
+ max-width: calc(100% - 2rem);
6913
+ width: calc(100% - 2rem);
6914
+ margin-top: -0.5rem;
6915
+ padding-top: 1rem;
6916
+ border-radius: ${({ theme: { borderRadius } }) => borderRadius.medium};
6917
+ }
6918
+
6919
+ ${({ $fontColor }) => !!$fontColor && HeaderFontColorMixin};
6920
+ `;
6921
+ const HeaderIcon = styled(uilibGl.Flex) `
6922
+ position: absolute;
6923
+ top: 0;
6924
+ right: 0;
6925
+ justify-content: flex-end;
6926
+ align-items: center;
6927
+ min-width: 7.5rem;
6928
+ height: 100%;
6929
+
6930
+ span[kind]:after {
6931
+ font-size: 7.5rem;
6932
+ }
6933
+
6934
+ span[kind]:after,
6935
+ path,
6936
+ line,
6937
+ circle {
6938
+ fill: rgba(255, 255, 255, 0.36);
6939
+ }
6940
+
6941
+ && > * {
6942
+ display: flex;
6943
+ align-items: center;
6944
+ height: 100%;
6945
+ }
6946
+ `;
6947
+ const BigIconHeaderMixin = styled.css `
6948
+ ${HeaderIcon} {
6949
+ min-width: 14rem;
6950
+ right: -3rem;
6951
+
6952
+ span[kind]:after {
6953
+ font-size: 14rem;
6954
+ }
6955
+ }
6956
+ `;
6957
+ const IconHeaderWrapper = styled.div `
6958
+ ${Header} {
6959
+ width: calc(100% + 3rem);
6960
+ margin: -1.5rem -1.5rem 0 -1.5rem;
6961
+ padding: 1.5rem;
6962
+ border-top-left-radius: 0.5rem;
6963
+ border-top-right-radius: 0.5rem;
6964
+ background: ${({ $bgColor }) => $bgColor || "linear-gradient(96.55deg, #FFFCD3 0%, #B4DC47 100%)"};
6965
+ }
6966
+
6967
+ ${HeaderWrapperMixin};
6968
+
6969
+ ${({ $bigIcon }) => $bigIcon && BigIconHeaderMixin};
6970
+ `;
6971
+
6972
+ const FeatureCardIconHeader = ({ isRow }) => {
6973
+ const { layerInfo } = useWidgetContext(exports.WidgetType.FeatureCard);
6974
+ const { config } = useWidgetConfig(exports.WidgetType.FeatureCard);
6975
+ const { header, options } = config || {};
6976
+ const renderElement = useHeaderRender(header);
6977
+ return (jsxRuntime.jsx(IconHeaderWrapper, { "$fontColor": options?.fontColor, "$bgColor": options?.bgColor, "$bigIcon": options?.bigIcon, children: jsxRuntime.jsx(uilibGl.ThemeProvider, { theme: uilibGl.defaultTheme, children: jsxRuntime.jsxs(Header, { "$isRow": isRow, children: [jsxRuntime.jsxs(HeaderFrontView, { children: [jsxRuntime.jsxs(HeaderContainer, { children: [jsxRuntime.jsx(LayerIcon, { layerInfo: layerInfo }), jsxRuntime.jsx(FeatureCardTitle, { title: renderElement({
6978
+ id: "title",
6979
+ wrap: false,
6980
+ }), description: renderElement({
6981
+ id: "description",
6982
+ wrap: false,
6983
+ }) })] }), jsxRuntime.jsx(FeatureCardButtons, {})] }), jsxRuntime.jsx(HeaderIcon, { children: renderElement({
6984
+ id: "icon",
6985
+ wrap: false,
6986
+ }) })] }) }) }));
6987
+ };
6988
+
6989
+ const ImageContainerButton = styled(uilibGl.FlatButton) `
6990
+ min-height: 1.5rem;
6991
+ border-radius: ${({ theme: { borderRadius } }) => borderRadius.large};
6992
+ background-color: ${({ theme: { palette } }) => palette.primary};
6993
+ text-transform: none;
6994
+
6995
+ :hover {
6996
+ background-color: ${({ theme: { palette } }) => palette.primaryDeep};
6997
+ }
6998
+ `;
6999
+
7000
+ const ElementButton = React.memo(({ type, elementConfig }) => {
7001
+ const { attributes } = useWidgetContext(type);
7002
+ const attribute = getAttributeByName(elementConfig?.attributeName, attributes);
7003
+ const link = attribute?.value;
7004
+ if (!link || typeof link !== "string")
7005
+ return null;
7006
+ return (jsxRuntime.jsx(ImageContainerButton, { onClick: () => window.open(attribute?.value), children: elementConfig.value || "" }));
7007
+ });
7008
+
7009
+ const AttributeGalleryContainer = styled.div `
7010
+ && {
7011
+ width: calc(100% + 3rem);
7012
+ }
7013
+
7014
+ min-height: 12.625rem;
7015
+ background-color: ${({ theme: { palette } }) => palette.element};
7016
+
7017
+ img {
7018
+ width: 100%;
7019
+ }
7020
+ `;
7021
+ const LinearProgressContainer = styled(uilibGl.Flex) `
7022
+ align-items: center;
7023
+ justify-content: center;
7024
+ min-height: inherit;
7025
+
7026
+ ${uilibGl.LinearProgress} {
7027
+ max-width: 4rem;
7028
+ }
7029
+ `;
7030
+ const NoLiveSnapshotContainer = styled(uilibGl.Flex) `
7031
+ flex-direction: column;
6568
7032
  align-items: center;
6569
7033
 
6570
7034
  span[kind="alert"] {
@@ -6651,7 +7115,7 @@ const imgSlideShowMixin = styled.css `
6651
7115
  }
6652
7116
  }
6653
7117
  `;
6654
- const SmallPreviewContainer = styled.div `
7118
+ const SmallPreviewContainer$1 = styled.div `
6655
7119
  cursor: ${({ onClick }) => (onClick ? "pointer" : "default")};
6656
7120
  position: relative;
6657
7121
  width: 100%;
@@ -6696,19 +7160,23 @@ const SmallPreviewContainer = styled.div `
6696
7160
  `;
6697
7161
  const SmallPreviewImages = styled.div ``;
6698
7162
 
7163
+ const NoLiveSnapshot = () => {
7164
+ return (jsxRuntime.jsxs(NoLiveSnapshotContainer, { children: [jsxRuntime.jsx(uilibGl.Icon, { kind: "alert" }), jsxRuntime.jsx(uilibGl.Description, { children: "\u0421\u043D\u0438\u043C\u043E\u043A \u043D\u0435 \u0434\u043E\u0441\u0442\u0443\u043F\u0435\u043D" })] }));
7165
+ };
7166
+
6699
7167
  const SmallPreview = ({ images, totalCount, currentIndex, isLoading, setCurrentIndex, onClick, onNextPage }) => {
6700
7168
  const { t } = useGlobalContext();
6701
- const [prevIndex, setPrevIndex] = react.useState(currentIndex + 1);
7169
+ const [prevIndex, setPrevIndex] = React.useState(currentIndex + 1);
6702
7170
  const isPrevDisabled = currentIndex === 0;
6703
7171
  const total = totalCount || images?.length || 0;
6704
7172
  const isNextDisabled = currentIndex === total - 1;
6705
- const prev = react.useCallback(() => {
7173
+ const prev = React.useCallback(() => {
6706
7174
  if (isPrevDisabled)
6707
7175
  return;
6708
7176
  setPrevIndex(currentIndex);
6709
7177
  setCurrentIndex(currentIndex - 1);
6710
7178
  }, [currentIndex, isPrevDisabled, setCurrentIndex]);
6711
- const next = react.useCallback(() => {
7179
+ const next = React.useCallback(() => {
6712
7180
  if (isNextDisabled)
6713
7181
  return;
6714
7182
  setPrevIndex(currentIndex);
@@ -6717,40 +7185,151 @@ const SmallPreview = ({ images, totalCount, currentIndex, isLoading, setCurrentI
6717
7185
  onNextPage();
6718
7186
  }
6719
7187
  }, [currentIndex, images, isNextDisabled, onNextPage, setCurrentIndex]);
6720
- return (jsxRuntime.jsxs(SmallPreviewContainer, { currentIndex: currentIndex + 1, prevIndex: prevIndex + 1, children: [isLoading && jsxRuntime.jsx(uilibGl.LinearProgress, {}), images.length > 1 && (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [jsxRuntime.jsx(SmallPreviewLeft, { "$isDisabled": isPrevDisabled, onClick: prev }), jsxRuntime.jsx(SmallPreviewRight, { "$isDisabled": isNextDisabled, onClick: next }), jsxRuntime.jsx(SmallPreviewCounter, { children: jsxRuntime.jsxs("div", { children: [currentIndex + 1, " ", t("of", { ns: "dashboard" }), " ", total] }) })] })), jsxRuntime.jsx(SmallPreviewImages, { onClick: onClick, children: images.map(image => (jsxRuntime.jsx("img", { src: image.src, alt: "" }, image.src))) })] }));
7188
+ return (jsxRuntime.jsxs(SmallPreviewContainer$1, { currentIndex: currentIndex + 1, prevIndex: prevIndex + 1, children: [isLoading && jsxRuntime.jsx(uilibGl.LinearProgress, {}), images.length > 1 && (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [jsxRuntime.jsx(SmallPreviewLeft, { "$isDisabled": isPrevDisabled, onClick: prev }), jsxRuntime.jsx(SmallPreviewRight, { "$isDisabled": isNextDisabled, onClick: next }), jsxRuntime.jsx(SmallPreviewCounter, { children: jsxRuntime.jsxs("div", { children: [currentIndex + 1, " ", t("of", { ns: "dashboard" }), " ", total] }) })] })), jsxRuntime.jsx(SmallPreviewImages, { onClick: onClick, children: images.map(image => (jsxRuntime.jsx("img", { src: image.src, alt: "" }, image.src))) })] }));
6721
7189
  };
6722
7190
 
6723
- const ElementSlideshow = ({ elementConfig, type, renderElement }) => {
6724
- const { expandedContainers, dataSources, attributes } = useWidgetContext(type);
6725
- const [currentIndex, setCurrentIndex] = react.useState(0);
7191
+ const DEFAULT_ARCHIVED_PERIOD = "#24h";
7192
+ const SNAPSHOTS_PER_PAGE = 10;
7193
+ const getDuration = (archivePeriod) => {
7194
+ const lastLetter = archivePeriod[archivePeriod.length - 1];
7195
+ const value = archivePeriod.slice(1, -1);
7196
+ const prop = lastLetter === "d"
7197
+ ? "days"
7198
+ : lastLetter === "h"
7199
+ ? "hours"
7200
+ : lastLetter === "m"
7201
+ ? "minutes"
7202
+ : lastLetter === "s"
7203
+ ? "seconds"
7204
+ : null;
7205
+ if (!lastLetter || !value || !prop)
7206
+ return {};
7207
+ return { [prop]: Number(value) };
7208
+ };
7209
+ const useCameraAttribute = (cameraId) => {
7210
+ const { api } = useGlobalContext();
7211
+ const [currentIndex, setCurrentIndex] = React.useState(0);
7212
+ const [currentPage, setCurrentPage] = React.useState(0);
7213
+ const [timeline, setTimeline] = React.useState([]);
7214
+ const [isLoadingSnapshot, setLoadingSnapshot] = React.useState(false);
7215
+ const [isLoadingTimeline, setLoadingTimeline] = React.useState(false);
6726
7216
  const [isOpenGallery, toggleGallery] = useToggle();
6727
- const { id, options, attributeName } = elementConfig || {};
6728
- const { expandable, expanded, relatedDataSource } = options || {};
6729
- const isVisible = isVisibleContainer(id, expandable, expanded, expandedContainers);
6730
- const images = react.useMemo(() => {
6731
- const dataSource = relatedDataSource ? dataSources?.find(({ name }) => name === relatedDataSource) : null;
6732
- const array = dataSource
6733
- ? dataSource.features.map(feature => feature.attributes[attributeName])
6734
- : getSlideshowImages({
6735
- element: elementConfig,
6736
- attribute: attributes?.find(({ name }) => name === attributeName),
7217
+ const [galleryImages, setGalleryImages] = React.useState([]);
7218
+ const imagesRef = React.useRef([]);
7219
+ // Todo get archivePeriod from project configuration
7220
+ const archivePeriod = /* appConfig?.settings?.cameraAttribute?.archivePeriod ||*/ DEFAULT_ARCHIVED_PERIOD;
7221
+ const onNextPage = React.useCallback(() => {
7222
+ setCurrentPage(currentPage + 1);
7223
+ }, [currentPage]);
7224
+ React.useEffect(() => {
7225
+ (async () => {
7226
+ if (!cameraId)
7227
+ return;
7228
+ setLoadingSnapshot(true);
7229
+ const snapshot = null; // await api.cameras.getLiveSnapshot(cameraId);
7230
+ const url = URL.createObjectURL(snapshot);
7231
+ setLoadingSnapshot(false);
7232
+ setLoadingTimeline(true);
7233
+ imagesRef.current = [
7234
+ {
7235
+ src: url,
7236
+ startTime: new Date(),
7237
+ },
7238
+ ];
7239
+ setGalleryImages([
7240
+ {
7241
+ src: url,
7242
+ fileName: formatDate$1(new Date()),
7243
+ },
7244
+ ]);
7245
+ const formatDateOptions = { dateFormat: exports.DateFormat.UTC };
7246
+ const startTime = formatDate$1(dateFns.sub(new Date(), getDuration(archivePeriod)), formatDateOptions);
7247
+ const endTime = formatDate$1(new Date(), formatDateOptions);
7248
+ const timelineResponse = await api.cameras.getArchiveTimeline({ cameraId, startTime, endTime });
7249
+ if (!timelineResponse.success || !timelineResponse.timeline.length)
7250
+ return;
7251
+ timelineResponse.timeline.reverse();
7252
+ setTimeline(timelineResponse.timeline);
7253
+ })();
7254
+ }, [cameraId]);
7255
+ React.useEffect(() => {
7256
+ if (!timeline.length)
7257
+ return;
7258
+ const timelinePage = timeline.slice(currentPage * SNAPSHOTS_PER_PAGE, (currentPage + 1) * SNAPSHOTS_PER_PAGE);
7259
+ for (const item of timelinePage) {
7260
+ if (!item)
7261
+ continue;
7262
+ api.cameras
7263
+ .getArchiveSnapshot({
7264
+ cameraId,
7265
+ time: item.startTime,
7266
+ })
7267
+ .then(response => {
7268
+ const newImages = [
7269
+ ...imagesRef.current,
7270
+ { src: URL.createObjectURL(response), startTime: new Date(item.startTime) },
7271
+ ];
7272
+ newImages.sort((a, b) => (a.startTime < b.startTime ? 1 : a.startTime > b.startTime ? -1 : 0));
7273
+ imagesRef.current = newImages;
7274
+ setGalleryImages(newImages.map(newImage => ({
7275
+ src: newImage.src,
7276
+ fileName: formatDate$1(newImage.startTime, { dateFormat: exports.DateFormat.DateTime }),
7277
+ })));
7278
+ if (newImages.length === timelinePage.length) {
7279
+ setLoadingTimeline(false);
7280
+ }
6737
7281
  });
6738
- return (array?.map(image => ({
6739
- src: image,
6740
- fileName: image,
6741
- })) || []);
6742
- }, [attributeName, attributes, dataSources, elementConfig, relatedDataSource]);
6743
- return (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [jsxRuntime.jsx(ExpandableTitle, { elementConfig: elementConfig, type: type, renderElement: renderElement }), isVisible && (jsxRuntime.jsxs(SmallPreviewContainer$1, { children: [jsxRuntime.jsx(SmallPreview, { images: images, currentIndex: currentIndex, setCurrentIndex: setCurrentIndex, onClick: toggleGallery }), isOpenGallery && (jsxRuntime.jsx(uilibGl.Preview, { images: images,
7282
+ }
7283
+ }, [timeline, currentPage]);
7284
+ return [
7285
+ {
7286
+ currentIndex,
7287
+ currentPage,
7288
+ totalCount: timeline.length,
7289
+ galleryImages,
7290
+ isLoadingSnapshot,
7291
+ isLoadingTimeline,
7292
+ isOpenGallery,
7293
+ },
7294
+ {
7295
+ toggleGallery,
7296
+ setGalleryImages,
7297
+ setCurrentIndex,
7298
+ onNextPage,
7299
+ },
7300
+ ];
7301
+ };
7302
+
7303
+ const ElementCamera = ({ type, renderElement, elementConfig, field }) => {
7304
+ const { expandedContainers, attributes } = useWidgetContext(type);
7305
+ const attribute = getAttributeByName(elementConfig?.attributeName, attributes);
7306
+ const value = attribute?.value;
7307
+ const { id, options } = elementConfig || {};
7308
+ const { expandable, expanded } = options || {};
7309
+ const isVisible = isVisibleContainer(id, expandable, expanded, expandedContainers);
7310
+ const [{ currentIndex /* , currentPage*/, totalCount, galleryImages, isLoadingSnapshot, isLoadingTimeline, isOpenGallery }, { setCurrentIndex, toggleGallery, onNextPage },] = useCameraAttribute(value);
7311
+ return (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [jsxRuntime.jsx(ExpandableTitle, { elementConfig: elementConfig, type: type, renderElement: renderElement }), isVisible && (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [!!field && jsxRuntime.jsx(AttributeLabel, { children: field }), jsxRuntime.jsx(AttributeGalleryContainer, { children: isLoadingSnapshot ? (jsxRuntime.jsx(LinearProgressContainer, { children: jsxRuntime.jsx(uilibGl.LinearProgress, {}) })) : !galleryImages?.length && !isLoadingTimeline ? (jsxRuntime.jsx(NoLiveSnapshot, {})) : (jsxRuntime.jsx(SmallPreview, { images: galleryImages, totalCount: totalCount, currentIndex: currentIndex, isLoading: isLoadingTimeline, setCurrentIndex: setCurrentIndex, onClick: toggleGallery, onNextPage: onNextPage })) }), isOpenGallery && (jsxRuntime.jsx(uilibGl.Preview, { images: galleryImages,
6744
7312
  // currentIndex={currentIndex}
6745
- // totalCount={images.length}
6746
- isOpen: isOpenGallery, onClose: toggleGallery }))] }))] }));
7313
+ // currentPage={currentPage}
7314
+ // totalCount={totalCount}
7315
+ isOpen: isOpenGallery,
7316
+ // isLoading={isLoadingTimeline}
7317
+ // hasSidebar
7318
+ onClose: toggleGallery }))] }))] }));
6747
7319
  };
6748
7320
 
6749
- const ElementSvg = react.memo(({ type, elementConfig }) => {
6750
- const { layerInfo, attributes } = useWidgetContext(type);
6751
- const { options } = elementConfig || {};
6752
- const { width, height, fontColor } = options || {};
6753
- return (jsxRuntime.jsx(SvgImage, { url: getSvgUrl({ elementConfig, layerInfo, attributes }), width: width, height: height, fontColor: fontColor }));
7321
+ const ElementChart = React.memo(({ type, elementConfig, renderElement }) => {
7322
+ const { config } = useWidgetConfig(type);
7323
+ return (jsxRuntime.jsx(Chart, { type: type, config: config, elementConfig: elementConfig, element: elementConfig, renderElement: renderElement }));
7324
+ });
7325
+
7326
+ const ElementChips = React.memo(({ type, elementConfig }) => {
7327
+ const { attributes } = useWidgetContext(type);
7328
+ const { attributeName, options, style } = elementConfig || {};
7329
+ const { separator, bgColor, fontColor, fontSize } = options || {};
7330
+ const attribute = getAttributeByName(attributeName, attributes);
7331
+ const tags = attribute?.value?.split(separator || ",");
7332
+ return (jsxRuntime.jsx(DashboardChipsContainer, { style: style, children: tags?.map(tag => (jsxRuntime.jsx(DashboardChip$1, { text: tag, "$bgColor": bgColor, "$fontColor": fontColor, "$fontSize": fontSize, "$isDefault": true }, tag))) }));
6754
7333
  });
6755
7334
 
6756
7335
  const StyledIconFontSizeMixin = styled.css `
@@ -6770,35 +7349,53 @@ const StyledIcon = styled(uilibGl.Icon) `
6770
7349
  ${({ fontColor }) => !!fontColor && StyledIconFontColorMixin};
6771
7350
  `;
6772
7351
 
6773
- const ElementIcon = react.memo(({ type, elementConfig }) => {
7352
+ const ElementIcon = React.memo(({ type, elementConfig }) => {
6774
7353
  const { attributes } = useWidgetContext(type);
6775
7354
  const { value, attributeName, options } = elementConfig || {};
6776
7355
  const { fontSize, fontColor } = options || {};
6777
- const iconValue = react.useMemo(() => (attributeName ? attributes?.find(item => item.name === attributeName)?.value : value), [attributeName, attributes, value]);
7356
+ const iconValue = React.useMemo(() => (attributeName ? attributes?.find(item => item.name === attributeName)?.value : value), [attributeName, attributes, value]);
6778
7357
  return jsxRuntime.jsx(StyledIcon, { kind: iconValue, fontSize: fontSize, fontColor: fontColor });
6779
7358
  });
6780
7359
 
6781
- const TooltipIcon = styled(uilibGl.Icon) `
6782
- &&& {
6783
- :after {
6784
- font-size: 0.75rem;
6785
- color: ${({ theme: { palette } }) => palette.iconDisabled};
6786
- transition: color ${uilibGl.transition.hover};
6787
- }
6788
-
6789
- :hover:after {
6790
- color: ${({ theme: { palette } }) => palette.icon};
6791
- }
6792
- }
6793
- `;
7360
+ const ElementImage = React.memo(({ type, elementConfig }) => {
7361
+ const { attributes } = useWidgetContext(type);
7362
+ const { value, attributeName, options } = elementConfig || {};
7363
+ const { width } = options || {};
7364
+ const firstImage = React.useMemo(() => {
7365
+ if (value) {
7366
+ return value.toString();
7367
+ }
7368
+ if (!attributeName || Array.isArray(attributeName)) {
7369
+ return null;
7370
+ }
7371
+ const attribute = attributes?.find(item => item.name === attributeName);
7372
+ return attribute?.value?.split(";")?.[0];
7373
+ }, [attributeName, attributes, value]);
7374
+ return firstImage ? jsxRuntime.jsx("img", { src: firstImage, alt: firstImage, width: width }) : null;
7375
+ });
6794
7376
 
6795
- const ElementTooltip = react.memo(({ type, elementConfig }) => {
7377
+ const ElementLegend = React.memo(({ type, element, elementConfig, expandedContainers }) => {
6796
7378
  const { attributes } = useWidgetContext(type);
6797
- const { attributeName, options } = elementConfig || {};
6798
- const { icon } = options || {};
6799
- const attribute = getAttributeByName(attributeName, attributes);
6800
- const text = (attribute?.value || elementConfig?.value);
6801
- return text ? (jsxRuntime.jsx(uilibGl.Tooltip, { placement: "top", arrow: true, content: text, children: ref => jsxRuntime.jsx(TooltipIcon, { kind: icon || "question", ref: ref }) })) : null;
7379
+ const { options } = elementConfig || {};
7380
+ const { twoColumns, chartId, relatedDataSources } = options || {};
7381
+ const chartElement = React.useMemo(() => findAnd.returnFound(element?.children, { id: chartId }), [chartId, element?.children]);
7382
+ const { data, loading } = useChartData({
7383
+ element: chartElement,
7384
+ type,
7385
+ });
7386
+ const legendData = React.useMemo(() => {
7387
+ if (!data[0]?.items?.length)
7388
+ return [];
7389
+ const relatedAttributes = relatedDataSources || [];
7390
+ const isRelated = !!relatedAttributes?.length;
7391
+ return isRelated
7392
+ ? data[0].items
7393
+ : data[0].items.map(item => {
7394
+ const attribute = attributes?.find(({ name }) => name === item.name);
7395
+ return { ...item, name: attribute?.alias || item.name };
7396
+ });
7397
+ }, [attributes, data, relatedDataSources]);
7398
+ return !chartElement?.options?.expanded || expandedContainers?.[chartElement.id] ? (jsxRuntime.jsx(ChartLegend, { data: legendData, loading: loading, chartElement: chartElement, twoColumns: twoColumns, type: type })) : null;
6802
7399
  });
6803
7400
 
6804
7401
  const ExternalLink = styled(uilibGl.IconButton).attrs(() => ({
@@ -6850,17 +7447,17 @@ const LocalLinkCopy = styled(uilibGl.Flex) `
6850
7447
  justify-content: center;
6851
7448
  `;
6852
7449
 
6853
- const LocalLink = react.memo(({ link }) => {
7450
+ const LocalLink = React.memo(({ link }) => {
6854
7451
  const { t } = useGlobalContext();
6855
7452
  const [isOpen, toggleOpen] = useToggle();
6856
- const onCopy = react.useCallback(() => {
7453
+ const onCopy = React.useCallback(() => {
6857
7454
  navigator.clipboard.writeText(link);
6858
7455
  toggleOpen();
6859
7456
  }, [link, toggleOpen]);
6860
7457
  return (jsxRuntime.jsx(uilibGl.Popover, { open: isOpen, zIndex: 10, onRequestClose: toggleOpen, anchorOrigin: "bottom-right", targetOrigin: "top-right", anchor: jsxRuntime.jsx(LocalLinkButton, { onClick: toggleOpen }), children: jsxRuntime.jsxs(LocalLinkBlank, { children: [jsxRuntime.jsx("a", { href: link, target: "_blank", rel: "noreferrer", children: link }), jsxRuntime.jsx(LocalLinkCopy, { children: jsxRuntime.jsx(uilibGl.IconButton, { kind: "copy", onClick: onCopy, children: t("copy", { ns: "dashboard" }) }) })] }) }));
6861
7458
  });
6862
7459
 
6863
- const ElementLink = react.memo(({ type, elementConfig }) => {
7460
+ const ElementLink = React.memo(({ type, elementConfig }) => {
6864
7461
  const { attributes } = useWidgetContext(type);
6865
7462
  const attribute = getAttributeByName(elementConfig?.attributeName, attributes);
6866
7463
  const link = attribute?.value;
@@ -6869,198 +7466,273 @@ const ElementLink = react.memo(({ type, elementConfig }) => {
6869
7466
  return link.startsWith("http") ? jsxRuntime.jsx(ExternalLink, { onClick: () => window.open(link) }) : jsxRuntime.jsx(LocalLink, { link: link });
6870
7467
  });
6871
7468
 
6872
- const ImageContainerButton = styled(uilibGl.FlatButton) `
6873
- min-height: 1.5rem;
6874
- border-radius: ${({ theme: { borderRadius } }) => borderRadius.large};
6875
- background-color: ${({ theme: { palette } }) => palette.primary};
6876
- text-transform: none;
6877
-
6878
- :hover {
6879
- background-color: ${({ theme: { palette } }) => palette.primaryDeep};
6880
- }
7469
+ const SmallPreviewContainer = styled.div `
7470
+ width: 100%;
7471
+ height: 100%;
6881
7472
  `;
6882
7473
 
6883
- const ElementButton = react.memo(({ type, elementConfig }) => {
6884
- const { attributes } = useWidgetContext(type);
6885
- const attribute = getAttributeByName(elementConfig?.attributeName, attributes);
6886
- const link = attribute?.value;
6887
- if (!link || typeof link !== "string")
6888
- return null;
6889
- return (jsxRuntime.jsx(ImageContainerButton, { onClick: () => window.open(attribute?.value), children: elementConfig.value || "" }));
6890
- });
6891
-
6892
- const NoLiveSnapshot = () => {
6893
- return (jsxRuntime.jsxs(NoLiveSnapshotContainer, { children: [jsxRuntime.jsx(uilibGl.Icon, { kind: "alert" }), jsxRuntime.jsx(uilibGl.Description, { children: "\u0421\u043D\u0438\u043C\u043E\u043A \u043D\u0435 \u0434\u043E\u0441\u0442\u0443\u043F\u0435\u043D" })] }));
6894
- };
6895
-
6896
- const DEFAULT_ARCHIVED_PERIOD = "#24h";
6897
- const SNAPSHOTS_PER_PAGE = 10;
6898
- const getDuration = (archivePeriod) => {
6899
- const lastLetter = archivePeriod[archivePeriod.length - 1];
6900
- const value = archivePeriod.slice(1, -1);
6901
- const prop = lastLetter === "d"
6902
- ? "days"
6903
- : lastLetter === "h"
6904
- ? "hours"
6905
- : lastLetter === "m"
6906
- ? "minutes"
6907
- : lastLetter === "s"
6908
- ? "seconds"
6909
- : null;
6910
- if (!lastLetter || !value || !prop)
6911
- return {};
6912
- return { [prop]: Number(value) };
6913
- };
6914
- const useCameraAttribute = (cameraId) => {
6915
- const { api } = useGlobalContext();
6916
- const [currentIndex, setCurrentIndex] = react.useState(0);
6917
- const [currentPage, setCurrentPage] = react.useState(0);
6918
- const [timeline, setTimeline] = react.useState([]);
6919
- const [isLoadingSnapshot, setLoadingSnapshot] = react.useState(false);
6920
- const [isLoadingTimeline, setLoadingTimeline] = react.useState(false);
7474
+ const ElementSlideshow = ({ elementConfig, type, renderElement }) => {
7475
+ const { expandedContainers, dataSources, attributes } = useWidgetContext(type);
7476
+ const [currentIndex, setCurrentIndex] = React.useState(0);
6921
7477
  const [isOpenGallery, toggleGallery] = useToggle();
6922
- const [galleryImages, setGalleryImages] = react.useState([]);
6923
- const imagesRef = react.useRef([]);
6924
- // Todo get archivePeriod from project configuration
6925
- const archivePeriod = /* appConfig?.settings?.cameraAttribute?.archivePeriod ||*/ DEFAULT_ARCHIVED_PERIOD;
6926
- const onNextPage = react.useCallback(() => {
6927
- setCurrentPage(currentPage + 1);
6928
- }, [currentPage]);
6929
- react.useEffect(() => {
6930
- (async () => {
6931
- if (!cameraId)
6932
- return;
6933
- setLoadingSnapshot(true);
6934
- const snapshot = null; // await api.cameras.getLiveSnapshot(cameraId);
6935
- const url = URL.createObjectURL(snapshot);
6936
- setLoadingSnapshot(false);
6937
- setLoadingTimeline(true);
6938
- imagesRef.current = [
6939
- {
6940
- src: url,
6941
- startTime: new Date(),
6942
- },
6943
- ];
6944
- setGalleryImages([
6945
- {
6946
- src: url,
6947
- fileName: formatDate$1(new Date()),
6948
- },
6949
- ]);
6950
- const formatDateOptions = { dateFormat: exports.DateFormat.UTC };
6951
- const startTime = formatDate$1(dateFns.sub(new Date(), getDuration(archivePeriod)), formatDateOptions);
6952
- const endTime = formatDate$1(new Date(), formatDateOptions);
6953
- const timelineResponse = await api.cameras.getArchiveTimeline({ cameraId, startTime, endTime });
6954
- if (!timelineResponse.success || !timelineResponse.timeline.length)
6955
- return;
6956
- timelineResponse.timeline.reverse();
6957
- setTimeline(timelineResponse.timeline);
6958
- })();
6959
- }, [cameraId]);
6960
- react.useEffect(() => {
6961
- if (!timeline.length)
6962
- return;
6963
- const timelinePage = timeline.slice(currentPage * SNAPSHOTS_PER_PAGE, (currentPage + 1) * SNAPSHOTS_PER_PAGE);
6964
- for (const item of timelinePage) {
6965
- if (!item)
6966
- continue;
6967
- api.cameras
6968
- .getArchiveSnapshot({
6969
- cameraId,
6970
- time: item.startTime,
6971
- })
6972
- .then(response => {
6973
- const newImages = [
6974
- ...imagesRef.current,
6975
- { src: URL.createObjectURL(response), startTime: new Date(item.startTime) },
6976
- ];
6977
- newImages.sort((a, b) => (a.startTime < b.startTime ? 1 : a.startTime > b.startTime ? -1 : 0));
6978
- imagesRef.current = newImages;
6979
- setGalleryImages(newImages.map(newImage => ({
6980
- src: newImage.src,
6981
- fileName: formatDate$1(newImage.startTime, { dateFormat: exports.DateFormat.DateTime }),
6982
- })));
6983
- if (newImages.length === timelinePage.length) {
6984
- setLoadingTimeline(false);
6985
- }
6986
- });
6987
- }
6988
- }, [timeline, currentPage]);
6989
- return [
6990
- {
6991
- currentIndex,
6992
- currentPage,
6993
- totalCount: timeline.length,
6994
- galleryImages,
6995
- isLoadingSnapshot,
6996
- isLoadingTimeline,
6997
- isOpenGallery,
6998
- },
6999
- {
7000
- toggleGallery,
7001
- setGalleryImages,
7002
- setCurrentIndex,
7003
- onNextPage,
7004
- },
7005
- ];
7006
- };
7007
-
7008
- const ElementCamera = ({ type, renderElement, elementConfig, field }) => {
7009
- const { expandedContainers, attributes } = useWidgetContext(type);
7010
- const attribute = getAttributeByName(elementConfig?.attributeName, attributes);
7011
- const value = attribute?.value;
7012
- const { id, options } = elementConfig || {};
7013
- const { expandable, expanded } = options || {};
7478
+ const { id, options, attributeName } = elementConfig || {};
7479
+ const { expandable, expanded, relatedDataSource } = options || {};
7014
7480
  const isVisible = isVisibleContainer(id, expandable, expanded, expandedContainers);
7015
- const [{ currentIndex /* , currentPage*/, totalCount, galleryImages, isLoadingSnapshot, isLoadingTimeline, isOpenGallery }, { setCurrentIndex, toggleGallery, onNextPage },] = useCameraAttribute(value);
7016
- return (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [jsxRuntime.jsx(ExpandableTitle, { elementConfig: elementConfig, type: type, renderElement: renderElement }), isVisible && (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [!!field && jsxRuntime.jsx(AttributeLabel, { children: field }), jsxRuntime.jsx(AttributeGalleryContainer, { children: isLoadingSnapshot ? (jsxRuntime.jsx(LinearProgressContainer, { children: jsxRuntime.jsx(uilibGl.LinearProgress, {}) })) : !galleryImages?.length && !isLoadingTimeline ? (jsxRuntime.jsx(NoLiveSnapshot, {})) : (jsxRuntime.jsx(SmallPreview, { images: galleryImages, totalCount: totalCount, currentIndex: currentIndex, isLoading: isLoadingTimeline, setCurrentIndex: setCurrentIndex, onClick: toggleGallery, onNextPage: onNextPage })) }), isOpenGallery && (jsxRuntime.jsx(uilibGl.Preview, { images: galleryImages,
7481
+ const images = React.useMemo(() => {
7482
+ const dataSource = relatedDataSource ? dataSources?.find(({ name }) => name === relatedDataSource) : null;
7483
+ const array = dataSource
7484
+ ? dataSource.features.map(feature => feature.attributes[attributeName])
7485
+ : getSlideshowImages({
7486
+ element: elementConfig,
7487
+ attribute: attributes?.find(({ name }) => name === attributeName),
7488
+ });
7489
+ return (array?.map(image => ({
7490
+ src: image,
7491
+ fileName: image,
7492
+ })) || []);
7493
+ }, [attributeName, attributes, dataSources, elementConfig, relatedDataSource]);
7494
+ return (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [jsxRuntime.jsx(ExpandableTitle, { elementConfig: elementConfig, type: type, renderElement: renderElement }), isVisible && (jsxRuntime.jsxs(SmallPreviewContainer, { children: [jsxRuntime.jsx(SmallPreview, { images: images, currentIndex: currentIndex, setCurrentIndex: setCurrentIndex, onClick: toggleGallery }), isOpenGallery && (jsxRuntime.jsx(uilibGl.Preview, { images: images,
7017
7495
  // currentIndex={currentIndex}
7018
- // currentPage={currentPage}
7019
- // totalCount={totalCount}
7020
- isOpen: isOpenGallery,
7021
- // isLoading={isLoadingTimeline}
7022
- // hasSidebar
7023
- onClose: toggleGallery }))] }))] }));
7496
+ // totalCount={images.length}
7497
+ isOpen: isOpenGallery, onClose: toggleGallery }))] }))] }));
7024
7498
  };
7025
7499
 
7026
- const ElementChart = react.memo(({ type, elementConfig, renderElement }) => {
7027
- const { config } = useWidgetConfig(type);
7028
- return (jsxRuntime.jsx(Chart, { type: type, config: config, elementConfig: elementConfig, element: elementConfig, renderElement: renderElement }));
7029
- });
7030
-
7031
- const ElementLegend = react.memo(({ type, element, elementConfig, expandedContainers }) => {
7032
- const { attributes } = useWidgetContext(type);
7500
+ const ElementSvg = React.memo(({ type, elementConfig }) => {
7501
+ const { layerInfo, attributes } = useWidgetContext(type);
7033
7502
  const { options } = elementConfig || {};
7034
- const { twoColumns, chartId, relatedDataSources } = options || {};
7035
- const chartElement = react.useMemo(() => findAnd.returnFound(element?.children, { id: chartId }), [chartId, element?.children]);
7036
- const { data, loading } = useChartData({
7037
- element: chartElement,
7038
- type,
7039
- });
7040
- const legendData = react.useMemo(() => {
7041
- if (!data[0]?.items?.length)
7042
- return [];
7043
- const relatedAttributes = relatedDataSources || [];
7044
- const isRelated = !!relatedAttributes?.length;
7045
- return isRelated
7046
- ? data[0].items
7047
- : data[0].items.map(item => {
7048
- const attribute = attributes?.find(({ name }) => name === item.name);
7049
- return { ...item, name: attribute?.alias || item.name };
7050
- });
7051
- }, [attributes, data, relatedDataSources]);
7052
- return !chartElement?.options?.expanded || expandedContainers?.[chartElement.id] ? (jsxRuntime.jsx(ChartLegend, { data: legendData, loading: loading, chartElement: chartElement, twoColumns: twoColumns, type: type })) : null;
7503
+ const { width, height, fontColor } = options || {};
7504
+ return (jsxRuntime.jsx(SvgImage, { url: getSvgUrl({ elementConfig, layerInfo, attributes }), width: width, height: height, fontColor: fontColor }));
7053
7505
  });
7054
7506
 
7055
- const ElementChips = react.memo(({ type, elementConfig }) => {
7507
+ const TooltipIcon = styled(uilibGl.Icon) `
7508
+ &&& {
7509
+ :after {
7510
+ font-size: 0.75rem;
7511
+ color: ${({ theme: { palette } }) => palette.iconDisabled};
7512
+ transition: color ${uilibGl.transition.hover};
7513
+ }
7514
+
7515
+ :hover:after {
7516
+ color: ${({ theme: { palette } }) => palette.icon};
7517
+ }
7518
+ }
7519
+ `;
7520
+
7521
+ const ElementTooltip = React.memo(({ type, elementConfig }) => {
7056
7522
  const { attributes } = useWidgetContext(type);
7057
- const { attributeName, options, style } = elementConfig || {};
7058
- const { separator, bgColor, fontColor, fontSize } = options || {};
7523
+ const { attributeName, options } = elementConfig || {};
7524
+ const { icon } = options || {};
7059
7525
  const attribute = getAttributeByName(attributeName, attributes);
7060
- const tags = attribute?.value?.split(separator || ",");
7061
- return (jsxRuntime.jsx(DashboardChipsContainer, { style: style, children: tags?.map(tag => (jsxRuntime.jsx(DashboardChip$1, { text: tag, "$bgColor": bgColor, "$fontColor": fontColor, "$fontSize": fontSize, "$isDefault": true }, tag))) }));
7526
+ const text = (attribute?.value || elementConfig?.value);
7527
+ return text ? (jsxRuntime.jsx(uilibGl.Tooltip, { placement: "top", arrow: true, content: text, children: ref => jsxRuntime.jsx(TooltipIcon, { kind: icon || "question", ref: ref }) })) : null;
7062
7528
  });
7063
7529
 
7530
+ const SlideshowHeaderWrapper = styled.div `
7531
+ padding: ${({ withPadding }) => (withPadding ? "0.5rem 0.5rem 0" : 0)};
7532
+
7533
+ ${Header} {
7534
+ align-items: flex-start;
7535
+ width: calc(100% + 2rem);
7536
+ height: ${({ big }) => (big ? "15.5rem" : "auto")};
7537
+ padding: 1.5rem;
7538
+ margin: -1rem -1rem 0 -1rem;
7539
+ border-radius: ${({ theme: { borderRadius } }) => borderRadius.medium};
7540
+
7541
+ ${SmallPreviewCounter} {
7542
+ bottom: ${({ withPadding }) => (withPadding ? 0.625 : 1.125)}rem;
7543
+ }
7544
+
7545
+ :before,
7546
+ :after {
7547
+ content: "";
7548
+ z-index: 2;
7549
+ position: absolute;
7550
+ top: 0;
7551
+ left: 0;
7552
+ width: 100%;
7553
+ }
7554
+
7555
+ :before {
7556
+ height: 100%;
7557
+ background: rgba(32, 46, 53, 0.25);
7558
+ }
7559
+
7560
+ :after {
7561
+ height: 4.5rem;
7562
+ background: linear-gradient(180deg, #000000 0%, rgba(0, 0, 0, 0) 100%);
7563
+ }
7564
+
7565
+ :hover {
7566
+ ${SmallPreviewControl}, ${SmallPreviewCounter} {
7567
+ opacity: 1;
7568
+ }
7569
+ }
7570
+ }
7571
+ `;
7572
+ const ImageContainerBg = styled.div `
7573
+ position: absolute;
7574
+ top: 0;
7575
+ bottom: 0;
7576
+ left: 0;
7577
+ right: 0;
7578
+
7579
+ img {
7580
+ width: 100%;
7581
+ height: 100%;
7582
+ object-position: center;
7583
+ object-fit: cover;
7584
+ }
7585
+ `;
7586
+ const HeaderSlideshow = styled.div `
7587
+ position: absolute;
7588
+ top: 0;
7589
+ bottom: ${({ height }) => (height ? `${height}px` : 0)};
7590
+ left: 0;
7591
+ right: 0;
7592
+
7593
+ img {
7594
+ width: 100%;
7595
+ height: 100%;
7596
+ min-height: inherit;
7597
+ object-position: center;
7598
+ object-fit: cover;
7599
+ }
7600
+ `;
7601
+
7602
+ const FeatureCardSlideshowHeader = ({ isRow }) => {
7603
+ const { layerInfo } = useWidgetContext(exports.WidgetType.FeatureCard);
7604
+ const { config } = useWidgetConfig(exports.WidgetType.FeatureCard);
7605
+ const { header, options } = config || {};
7606
+ const renderElement = useHeaderRender(header);
7607
+ return (jsxRuntime.jsx(SlideshowHeaderWrapper, { big: true, withPadding: options?.withPadding, children: jsxRuntime.jsx(uilibGl.ThemeProvider, { theme: uilibGl.darkTheme, children: jsxRuntime.jsxs(Header, { "$isRow": isRow, children: [jsxRuntime.jsxs(HeaderFrontView, { children: [jsxRuntime.jsxs(HeaderContainer, { children: [jsxRuntime.jsx(LayerIcon, { layerInfo: layerInfo }), jsxRuntime.jsx(FeatureCardTitle, { title: renderElement({
7608
+ id: "title",
7609
+ wrap: false,
7610
+ }), description: renderElement({
7611
+ id: "description",
7612
+ wrap: false,
7613
+ }) })] }), jsxRuntime.jsx(FeatureCardButtons, {})] }), jsxRuntime.jsx(ImageContainerBg, { children: renderElement({
7614
+ id: "bgImage",
7615
+ wrap: false,
7616
+ }) }), jsxRuntime.jsx(HeaderSlideshow, { height: options?.height, children: renderElement({
7617
+ id: "slideshow",
7618
+ wrap: false,
7619
+ }) })] }) }) }));
7620
+ };
7621
+
7622
+ const getDashboardHeader = (templateName) => {
7623
+ switch (templateName) {
7624
+ case exports.HeaderTemplate.Default:
7625
+ default:
7626
+ return DashboardDefaultHeader;
7627
+ }
7628
+ };
7629
+
7630
+ const getDataFromAttributes = (t, config, attributes) => {
7631
+ const colors = config?.options?.colors || FEATURE_CARD_DEFAULT_COLORS;
7632
+ const data = config?.children?.map(({ attributeName }, index) => {
7633
+ const attribute = attributes?.find(({ name }) => name === attributeName);
7634
+ return {
7635
+ name: attribute?.name || "",
7636
+ value: attribute ? Number(attribute.value) : 0,
7637
+ color: colors[index] || FEATURE_CARD_OTHER_COLOR
7638
+ };
7639
+ });
7640
+ if (config?.options?.orderByValue) {
7641
+ data.sort((a, b) => b.value - a.value);
7642
+ }
7643
+ const isOtherSliced = config?.options?.otherItems && config.options.otherItems < data.length;
7644
+ if (!isOtherSliced)
7645
+ return data;
7646
+ const otherValue = data
7647
+ .slice(0, config?.options?.otherItems)
7648
+ .reduce((prev, { value }) => prev + value, 0);
7649
+ return [
7650
+ ...data?.slice(0, config.options?.otherItems),
7651
+ {
7652
+ name: t("other", { ns: "dashboard" }),
7653
+ color: FEATURE_CARD_OTHER_COLOR,
7654
+ value: Number(otherValue)
7655
+ }
7656
+ ];
7657
+ };
7658
+
7659
+ const getDataFromRelatedFeatures = ({ t, config, filters, relatedConfig, dataSource, layerInfo }) => {
7660
+ if (!dataSource || !dataSource?.features?.length) {
7661
+ return [];
7662
+ }
7663
+ const { colors } = config.options;
7664
+ const layerDefinition = getLayerDefinition(layerInfo);
7665
+ let data = JSON.parse(JSON.stringify(dataSource?.features));
7666
+ const sortAttribute = config?.options?.orderByValue
7667
+ ? relatedConfig.attributeName
7668
+ : config?.options?.orderByTitle
7669
+ ? relatedConfig.attributeTitle
7670
+ : null;
7671
+ if (sortAttribute) {
7672
+ data.sort((a, b) => b.attributes[sortAttribute] - a.attributes[sortAttribute]);
7673
+ }
7674
+ const isOtherSliced = config?.options?.otherItems && config.options.otherItems < data.length;
7675
+ const otherValue = isOtherSliced
7676
+ ? data
7677
+ .slice(config.options.otherItems)
7678
+ .reduce((prev, { attributes }) => prev + attributes[relatedConfig.attributeName], 0)
7679
+ : null;
7680
+ if (isOtherSliced) {
7681
+ data = data.slice(0, config?.options?.otherItems);
7682
+ }
7683
+ const gradientArray = relatedConfig.chartAxis && colors?.length < data.length
7684
+ ? new Gradient()
7685
+ .setColorGradient(colors[0], colors[colors.length - 1])
7686
+ .setMidpoint(data.length)
7687
+ .getColors()
7688
+ : colors;
7689
+ const filter = getConfigFilter(relatedConfig?.filterName, filters);
7690
+ const result = data.reduce((acc, feature, index) => {
7691
+ if (relatedConfig?.filterName && !filter) {
7692
+ return acc;
7693
+ }
7694
+ const attributeName = feature.attributes[relatedConfig.attributeName];
7695
+ const attributeTitle = feature.attributes[relatedConfig.attributeTitle];
7696
+ const attributeColor = feature.attributes[relatedConfig.attributeColor];
7697
+ return [
7698
+ ...acc,
7699
+ {
7700
+ name: formatAttributeValue({
7701
+ t,
7702
+ type: layerDefinition.attributes[relatedConfig.attributeTitle]?.type,
7703
+ value: attributeTitle,
7704
+ stringFormat: layerDefinition.attributes[relatedConfig.attributeTitle]?.stringFormat
7705
+ }),
7706
+ value: attributeName === null ? null : Number(attributeName),
7707
+ color: attributeColor || gradientArray?.[index] || FEATURE_CARD_OTHER_COLOR,
7708
+ min: filter?.attributeMin ? feature.attributes[filter.attributeMin] : null,
7709
+ max: filter?.attributeMin ? feature.attributes[filter.attributeMax] : null
7710
+ }
7711
+ ];
7712
+ }, []);
7713
+ if (isOtherSliced) {
7714
+ result.push({
7715
+ name: t("other", { ns: "dashboard" }),
7716
+ color: FEATURE_CARD_OTHER_COLOR,
7717
+ value: Number(otherValue)
7718
+ });
7719
+ }
7720
+ return result;
7721
+ };
7722
+
7723
+ const getDefaultConfig = ({ title, defaultTitle, items, baseMapName, position, resolution, srid, }) => {
7724
+ const dashboardConfiguration = JSON.parse(JSON.stringify(DEFAULT_DASHBOARD_CONFIG));
7725
+ const defaultPages = JSON.parse(JSON.stringify(DEFAULT_PAGES_CONFIG));
7726
+ const defaultPage = createConfigPage({ title, baseMapName, position, resolution, defaultTitle });
7727
+ if (items?.length) {
7728
+ const contentItems = items.map(item => createConfigLayer(item.name));
7729
+ defaultPage.layers.push(...contentItems);
7730
+ }
7731
+ defaultPages.children.push(defaultPage);
7732
+ dashboardConfiguration.children.push(defaultPages);
7733
+ return dashboardConfiguration;
7734
+ };
7735
+
7064
7736
  const elementComponents = {
7065
7737
  image: ElementImage,
7066
7738
  slideshow: ElementSlideshow,
@@ -7152,6 +7824,20 @@ function getFeatureAttributes(feature = {}, layer, dataSource) {
7152
7824
  });
7153
7825
  }
7154
7826
 
7827
+ const getFeatureCardHeader = (templateName) => {
7828
+ switch (templateName) {
7829
+ case exports.HeaderTemplate.Slideshow:
7830
+ return FeatureCardSlideshowHeader;
7831
+ case exports.HeaderTemplate.Gradient:
7832
+ return FeatureCardGradientHeader;
7833
+ case exports.HeaderTemplate.Icon:
7834
+ return FeatureCardIconHeader;
7835
+ case exports.HeaderTemplate.Default:
7836
+ default:
7837
+ return FeatureCardDefaultHeader;
7838
+ }
7839
+ };
7840
+
7155
7841
  const RangeNumberFilter = ({ type, filter }) => {
7156
7842
  const { t } = useGlobalContext();
7157
7843
  const { filters, changeFilters } = useWidgetContext(type);
@@ -7160,8 +7846,8 @@ const RangeNumberFilter = ({ type, filter }) => {
7160
7846
  const { filters: configFilters } = currentPage;
7161
7847
  const { filterName, width, step, label, minValue = RANGE_MIN_VALUE, maxValue = RANGE_MAX_VALUE } = filter.options;
7162
7848
  const filterValue = [filters[filterName]?.min || minValue, filters[filterName]?.max || maxValue];
7163
- const [value, setValue] = react.useState(filterValue);
7164
- react.useEffect(() => {
7849
+ const [value, setValue] = React.useState(filterValue);
7850
+ React.useEffect(() => {
7165
7851
  updateDebounce(() => {
7166
7852
  changeFilters({
7167
7853
  [filterName]: {
@@ -7180,8 +7866,8 @@ const RangeNumberFilter = ({ type, filter }) => {
7180
7866
 
7181
7867
  const TextFilter = ({ type, filter, config }) => {
7182
7868
  const { filters, changeFilters, dataSources } = useWidgetContext(type);
7183
- const suggestRef = react.useRef(null);
7184
- const [totalCount, setTotalCount] = react.useState(0);
7869
+ const suggestRef = React.useRef(null);
7870
+ const [totalCount, setTotalCount] = React.useState(0);
7185
7871
  const { attributes, layerInfo } = useRelatedDataSourceAttributes({
7186
7872
  type,
7187
7873
  elementConfig: filter,
@@ -7191,10 +7877,10 @@ const TextFilter = ({ type, filter, config }) => {
7191
7877
  const { filterName, searchFilterName, placeholder, width, height, multiSelect, variants } = filter.options;
7192
7878
  const { eqlParameters } = (layerInfo?.configuration || {});
7193
7879
  const layerParams = useLayerParams(layerInfo);
7194
- const [value, setValue] = react.useState("");
7195
- const configFilter = react.useMemo(() => getConfigFilter(filterName, configFilters), [configFilters, filterName]);
7196
- const searchFilter = react.useMemo(() => getConfigFilter(searchFilterName, configFilters), [configFilters, searchFilterName]);
7197
- const searchDataSource = react.useMemo(() => !!searchFilter?.relatedDataSource &&
7880
+ const [value, setValue] = React.useState("");
7881
+ const configFilter = React.useMemo(() => getConfigFilter(filterName, configFilters), [configFilters, filterName]);
7882
+ const searchFilter = React.useMemo(() => getConfigFilter(searchFilterName, configFilters), [configFilters, searchFilterName]);
7883
+ const searchDataSource = React.useMemo(() => !!searchFilter?.relatedDataSource &&
7198
7884
  configDataSources.find(({ name }) => name === searchFilter.relatedDataSource), [configDataSources, searchFilter?.relatedDataSource]);
7199
7885
  const updateDataSource = useUpdateDataSource({
7200
7886
  dataSource: searchDataSource,
@@ -7204,15 +7890,15 @@ const TextFilter = ({ type, filter, config }) => {
7204
7890
  layerParams,
7205
7891
  eqlParameters,
7206
7892
  });
7207
- const [options, setOptions] = react.useState([]);
7208
- const setFilterValue = react.useCallback(newValue => {
7893
+ const [options, setOptions] = React.useState([]);
7894
+ const setFilterValue = React.useCallback(newValue => {
7209
7895
  changeFilters({
7210
7896
  [filterName]: {
7211
7897
  value: newValue,
7212
7898
  },
7213
7899
  }, configFilter?.resetFilters);
7214
7900
  }, [changeFilters, configFilter?.resetFilters, filterName]);
7215
- const onSelect = react.useCallback((newSelected) => {
7901
+ const onSelect = React.useCallback((newSelected) => {
7216
7902
  const lastSelected = newSelected[newSelected.length - 1];
7217
7903
  setFilterValue(multiSelect
7218
7904
  ? newSelected.map(item => item.value)
@@ -7221,7 +7907,7 @@ const TextFilter = ({ type, filter, config }) => {
7221
7907
  : []);
7222
7908
  setValue(multiSelect ? "" : lastSelected?.text || "");
7223
7909
  }, [multiSelect, setFilterValue]);
7224
- const onBlur = react.useCallback(() => {
7910
+ const onBlur = React.useCallback(() => {
7225
7911
  if (variants) {
7226
7912
  setFilterValue([value]);
7227
7913
  }
@@ -7229,7 +7915,7 @@ const TextFilter = ({ type, filter, config }) => {
7229
7915
  setOptions([]);
7230
7916
  }
7231
7917
  }, [variants, setFilterValue, value]);
7232
- const onKeyUp = react.useCallback(e => {
7918
+ const onKeyUp = React.useCallback(e => {
7233
7919
  if (e.key === "Enter") {
7234
7920
  setFilterValue([value]);
7235
7921
  }
@@ -7248,7 +7934,7 @@ const TextFilter = ({ type, filter, config }) => {
7248
7934
  }));
7249
7935
  },
7250
7936
  });
7251
- const search = react.useCallback((searchValue, isLoadNext = false) => {
7937
+ const search = React.useCallback((searchValue, isLoadNext = false) => {
7252
7938
  if (variants || !searchFilter?.attributeAlias || !searchFilter?.attributeValue)
7253
7939
  return;
7254
7940
  const newFilter = {
@@ -7270,11 +7956,11 @@ const TextFilter = ({ type, filter, config }) => {
7270
7956
  updateDataSource,
7271
7957
  options.length,
7272
7958
  ]);
7273
- const infiniteScrollProps = react.useMemo(() => ({
7959
+ const infiniteScrollProps = React.useMemo(() => ({
7274
7960
  loadNext: () => loadNext(() => search(value, true)),
7275
7961
  hasMore: totalCount > options.length,
7276
7962
  }), [totalCount, options.length, loadNext, search, value]);
7277
- const selected = react.useMemo(() => {
7963
+ const selected = React.useMemo(() => {
7278
7964
  if (!filters[filterName]?.value)
7279
7965
  return [];
7280
7966
  return Array.isArray(filters[filterName].value)
@@ -7306,7 +7992,7 @@ const DropdownFilter = ({ type, filter, elementConfig }) => {
7306
7992
  const { filterName, placeholder, width, noEmptyOption } = filter.options;
7307
7993
  const { sliceItems } = useShownOtherItems(options);
7308
7994
  const configFilter = getConfigFilter(filterName, configFilters);
7309
- const filterOptions = react.useMemo(() => {
7995
+ const filterOptions = React.useMemo(() => {
7310
7996
  const dataSource = getDataSource(configFilter.relatedDataSource, dataSources);
7311
7997
  const listOptions = getListOptions(sliceItems(dataSource?.features), filterName, configFilters);
7312
7998
  if (noEmptyOption === undefined || noEmptyOption === false) {
@@ -7314,12 +8000,12 @@ const DropdownFilter = ({ type, filter, elementConfig }) => {
7314
8000
  }
7315
8001
  return listOptions;
7316
8002
  }, [configFilter.relatedDataSource, configFilters, dataSources, filterName, noEmptyOption, placeholder, sliceItems]);
7317
- const value = react.useMemo(() => filters?.[filterName] !== undefined
8003
+ const value = React.useMemo(() => filters?.[filterName] !== undefined
7318
8004
  ? Array.isArray(filters[filterName].value)
7319
8005
  ? null
7320
8006
  : filters[filterName].value
7321
8007
  : getConfigFilter(filterName, configFilters).defaultValue, [configFilters, filterName, filters]);
7322
- const onChange = react.useCallback((items) => {
8008
+ const onChange = React.useCallback((items) => {
7323
8009
  changeFilters({ [filterName]: items[0] });
7324
8010
  }, [changeFilters, filterName]);
7325
8011
  if (!configFilter)
@@ -7362,7 +8048,7 @@ const ChartLoading = ({ column }) => (jsxRuntime.jsx(uilibGl.Flex, { position: "
7362
8048
 
7363
8049
  const ContainerLoading = () => (jsxRuntime.jsx(uilibGl.Flex, { alignContent: "center", justifyContent: "center", width: "100%", children: jsxRuntime.jsx(uilibGl.CircularProgress, { diameter: 1.5, mono: true }) }));
7364
8050
 
7365
- const ChartWrapper = react.memo(({ width, height, column, loading, children }) => {
8051
+ const ChartWrapper = React.memo(({ width, height, column, loading, children }) => {
7366
8052
  return (jsxRuntime.jsxs(ChartWrapperContainer, { column: column, style: { width, height }, children: [loading && jsxRuntime.jsx(ChartLoading, { column: column }), jsxRuntime.jsx(uilibGl.Flex, { opacity: loading ? FILTERED_VALUE_OPACITY / 100 : 1, children: children })] }));
7367
8053
  });
7368
8054
 
@@ -7411,16 +8097,16 @@ const useBarChartFilter = ({ type, filter, elementConfig }) => {
7411
8097
  const { t } = useGlobalContext();
7412
8098
  const { isLoading, dataSources, filters, changeFilters } = useWidgetContext(type);
7413
8099
  const { currentPage } = useWidgetPage(type);
7414
- const [selectedPixels, setSelectedPixels] = react.useState(null);
8100
+ const [selectedPixels, setSelectedPixels] = React.useState(null);
7415
8101
  const { filterName } = filter.options;
7416
8102
  const { options } = elementConfig;
7417
8103
  const { barHeight, barWidth = DEFAULT_BAR_WIDTH, padding = DEFAULT_BAR_PADDING, height = DEFAULT_BAR_CHART_HEIGHT, radius = DEFAULT_BARCHART_RADIUS, width: configWidth, markers: configMarkers, colors: configColors, colorAttribute, defaultColor, primaryColor, drawMinMax } = options || {};
7418
8104
  const barFullWidth = barWidth + padding;
7419
8105
  const { filters: configFilters } = currentPage;
7420
- const configFilter = react.useMemo(() => getConfigFilter(filterName, configFilters), [configFilters, filterName]);
8106
+ const configFilter = React.useMemo(() => getConfigFilter(filterName, configFilters), [configFilters, filterName]);
7421
8107
  const { relatedDataSource, attributeValue, defaultValue } = configFilter || {};
7422
- const dataSource = react.useMemo(() => getDataSource(relatedDataSource, dataSources), [dataSources, relatedDataSource]);
7423
- const filterItems = react.useMemo(() => getDataFromRelatedFeatures({
8108
+ const dataSource = React.useMemo(() => getDataSource(relatedDataSource, dataSources), [dataSources, relatedDataSource]);
8109
+ const filterItems = React.useMemo(() => getDataFromRelatedFeatures({
7424
8110
  t,
7425
8111
  dataSource,
7426
8112
  config: elementConfig,
@@ -7442,32 +8128,32 @@ const useBarChartFilter = ({ type, filter, elementConfig }) => {
7442
8128
  elementConfig,
7443
8129
  t
7444
8130
  ]);
7445
- const amount = react.useMemo(() => filterItems?.length || 0, [filterItems?.length]);
7446
- const width = react.useMemo(() => configWidth || amount * barFullWidth, [amount, barFullWidth, configWidth]);
7447
- const data = react.useMemo(() => getDataFromFilterItems(filterItems), [filterItems]);
8131
+ const amount = React.useMemo(() => filterItems?.length || 0, [filterItems?.length]);
8132
+ const width = React.useMemo(() => configWidth || amount * barFullWidth, [amount, barFullWidth, configWidth]);
8133
+ const data = React.useMemo(() => getDataFromFilterItems(filterItems), [filterItems]);
7448
8134
  const loading = isLoading && !amount;
7449
8135
  const { formatFilterColor, onFilter } = useFeatureFilters(type, filterName, filterItems);
7450
- const colors = react.useMemo(() => getColorsFromFilterItems(filterItems, defaultColor, formatFilterColor), [defaultColor, filterItems, formatFilterColor]);
7451
- const selectedRange = react.useMemo(() => getIndexRange(selectedPixels, amount, barFullWidth), [amount, barFullWidth, selectedPixels]);
7452
- const customXAxis = react.useCallback(xAxis => {
8136
+ const colors = React.useMemo(() => getColorsFromFilterItems(filterItems, defaultColor, formatFilterColor), [defaultColor, filterItems, formatFilterColor]);
8137
+ const selectedRange = React.useMemo(() => getIndexRange(selectedPixels, amount, barFullWidth), [amount, barFullWidth, selectedPixels]);
8138
+ const customXAxis = React.useCallback(xAxis => {
7453
8139
  xAxis.select(".tick:first-of-type").attr("transform", "translate(0, 0)");
7454
8140
  xAxis.select(".tick:last-of-type").attr("transform", `translate(${width}, 0)`);
7455
8141
  }, [width]);
7456
- const customXAxisBottom = react.useCallback(xAxisBottom => {
8142
+ const customXAxisBottom = React.useCallback(xAxisBottom => {
7457
8143
  xAxisBottom.tickFormat((index) => (drawMinMax === undefined || drawMinMax) && (index === 0 || index === data.length - 1)
7458
8144
  ? data[index].groupName
7459
8145
  : "");
7460
8146
  }, [data, drawMinMax]);
7461
- const customYAxis = react.useCallback(yAxis => {
8147
+ const customYAxis = React.useCallback(yAxis => {
7462
8148
  yAxis.remove();
7463
8149
  }, []);
7464
- const customBars = react.useCallback(({ bars }) => {
8150
+ const customBars = React.useCallback(({ bars }) => {
7465
8151
  bars.attr("rx", radius).attr("ry", radius);
7466
8152
  }, [radius]);
7467
- const tooltipY = react.useMemo(() => (barHeight ? height - barHeight - DEFAULT_BAR_CHART_MARGINS.bottom : undefined), [barHeight, height]);
7468
- const markers = react.useMemo(() => getChartMarkers(filterItems, configMarkers), [configMarkers, filterItems]);
7469
- const hasDefinedColors = react.useMemo(() => configColors?.length || colorAttribute, [configColors, colorAttribute]);
7470
- const marshalledMap = react.useCallback(items => {
8153
+ const tooltipY = React.useMemo(() => (barHeight ? height - barHeight - DEFAULT_BAR_CHART_MARGINS.bottom : undefined), [barHeight, height]);
8154
+ const markers = React.useMemo(() => getChartMarkers(filterItems, configMarkers), [configMarkers, filterItems]);
8155
+ const hasDefinedColors = React.useMemo(() => configColors?.length || colorAttribute, [configColors, colorAttribute]);
8156
+ const marshalledMap = React.useCallback(items => {
7471
8157
  if (hasDefinedColors)
7472
8158
  return undefined;
7473
8159
  return items.map(([item], index) => {
@@ -7481,22 +8167,22 @@ const useBarChartFilter = ({ type, filter, elementConfig }) => {
7481
8167
  ];
7482
8168
  });
7483
8169
  }, [defaultColor, hasDefinedColors, primaryColor, selectedRange, theme.palette.elementDeep, theme.palette.primary]);
7484
- const minValue = react.useMemo(() => getMinValueFromFilterItems(filterItems), [filterItems]);
7485
- const onBarClick = react.useCallback((group) => {
8170
+ const minValue = React.useMemo(() => getMinValueFromFilterItems(filterItems), [filterItems]);
8171
+ const onBarClick = React.useCallback((group) => {
7486
8172
  if (loading)
7487
8173
  return;
7488
8174
  setSelectedPixels([group.groupIndex * barFullWidth, group.groupIndex * barFullWidth]);
7489
8175
  const feature = filterItems.find(({ value }) => value === group.value);
7490
8176
  onFilter(feature.name);
7491
8177
  }, [barFullWidth, filterItems, loading, onFilter]);
7492
- const getValueRange = react.useCallback((pixelRange) => {
8178
+ const getValueRange = React.useCallback((pixelRange) => {
7493
8179
  const currentRange = pixelRange || selectedPixels;
7494
8180
  const indexRange = getIndexRange(currentRange, filterItems?.length, barFullWidth);
7495
8181
  if (!indexRange || lodash.isNil(filterItems[indexRange[0]]) || lodash.isNil(filterItems[indexRange[1]]))
7496
8182
  return null;
7497
8183
  return [filterItems[indexRange[0]].name, filterItems[indexRange[1]].name];
7498
8184
  }, [barFullWidth, filterItems, selectedPixels]);
7499
- const onSelect = react.useCallback((pixelRange) => {
8185
+ const onSelect = React.useCallback((pixelRange) => {
7500
8186
  const valueRange = getValueRange(pixelRange);
7501
8187
  if (!valueRange)
7502
8188
  return;
@@ -7509,25 +8195,25 @@ const useBarChartFilter = ({ type, filter, elementConfig }) => {
7509
8195
  }
7510
8196
  }, configFilter?.resetFilters);
7511
8197
  }, [changeFilters, configFilter?.resetFilters, filterName, getValueRange]);
7512
- const prevDisabled = react.useMemo(() => loading || !selectedPixels || selectedPixels?.[0] - barFullWidth < 0, [barFullWidth, loading, selectedPixels]);
7513
- const nextDisabled = react.useMemo(() => loading || !selectedPixels || selectedPixels?.[1] + barFullWidth >= width, [barFullWidth, loading, selectedPixels, width]);
7514
- const onPrev = react.useCallback(() => {
8198
+ const prevDisabled = React.useMemo(() => loading || !selectedPixels || selectedPixels?.[0] - barFullWidth < 0, [barFullWidth, loading, selectedPixels]);
8199
+ const nextDisabled = React.useMemo(() => loading || !selectedPixels || selectedPixels?.[1] + barFullWidth >= width, [barFullWidth, loading, selectedPixels, width]);
8200
+ const onPrev = React.useCallback(() => {
7515
8201
  if (prevDisabled)
7516
8202
  return;
7517
8203
  onSelect([selectedPixels[0] - barFullWidth, selectedPixels[1] - barFullWidth]);
7518
8204
  }, [barFullWidth, prevDisabled, onSelect, selectedPixels]);
7519
- const onNext = react.useCallback(() => {
8205
+ const onNext = React.useCallback(() => {
7520
8206
  if (nextDisabled)
7521
8207
  return;
7522
8208
  onSelect([selectedPixels[0] + barFullWidth, selectedPixels[1] + barFullWidth]);
7523
8209
  }, [barFullWidth, nextDisabled, onSelect, selectedPixels]);
7524
- react.useEffect(() => {
8210
+ React.useEffect(() => {
7525
8211
  if (selectedRange || !defaultValue || !filterItems?.length)
7526
8212
  return;
7527
8213
  const pixelRange = getPixelRange((Array.isArray(defaultValue) ? defaultValue : [defaultValue, defaultValue]), filterItems, barFullWidth);
7528
8214
  onSelect(pixelRange);
7529
8215
  }, [amount]);
7530
- react.useEffect(() => {
8216
+ React.useEffect(() => {
7531
8217
  if (lodash.isEmpty(filters[filterName]?.value)) {
7532
8218
  if (!lodash.isEmpty(selectedPixels)) {
7533
8219
  setSelectedPixels(null);
@@ -7574,14 +8260,14 @@ const BarChartFilter = ({ type, config, elementConfig, filter }) => {
7574
8260
  elementConfig,
7575
8261
  filter
7576
8262
  });
7577
- const renderSelectedRange = react.useMemo(() => {
8263
+ const renderSelectedRange = React.useMemo(() => {
7578
8264
  if (!selectedRange)
7579
8265
  return null;
7580
8266
  return (jsxRuntime.jsx(BarChartFilterSelected, { children: selectedRange[0] === selectedRange[1]
7581
8267
  ? data[selectedRange[0]].groupName
7582
8268
  : `${data[selectedRange[0]].groupName} - ${data[selectedRange[1]].groupName}` }));
7583
8269
  }, [data, selectedRange]);
7584
- const renderTooltip = react.useCallback((groups) => {
8270
+ const renderTooltip = React.useCallback((groups) => {
7585
8271
  const group = groups[0];
7586
8272
  return (jsxRuntime.jsx(uilibGl.ThemeProvider, { children: jsxRuntime.jsx(TooltipContainer, { children: group.groupName === group.value.toString() ? group.value : [group.groupName, group.value].join(": ") }) }));
7587
8273
  }, []);
@@ -7594,25 +8280,25 @@ const RangeDateFilter = ({ type, filter }) => {
7594
8280
  const { t } = useGlobalContext();
7595
8281
  const { changeFilters, dataSources } = useWidgetContext(type);
7596
8282
  const { currentPage } = useWidgetPage(type);
7597
- const [value, setValue] = react.useState([undefined, undefined]);
7598
- const isInit = react.useRef(false);
8283
+ const [value, setValue] = React.useState([undefined, undefined]);
8284
+ const isInit = React.useRef(false);
7599
8285
  const updateDebounce = useDebouncedCallback(FILTER_UPDATE_DEBOUNCE);
7600
8286
  const { filters: configFilters } = currentPage;
7601
8287
  const { filterName, label, minValue, maxValue, withTime } = filter.options;
7602
- const configFilter = react.useMemo(() => getConfigFilter(filterName, configFilters), [configFilters, filterName]);
7603
- const dataSource = react.useMemo(() => getDataSource(configFilter?.relatedDataSource, dataSources), [configFilter?.relatedDataSource, dataSources]);
7604
- const { minFromData, maxFromData } = react.useMemo(() => dataSource?.features?.reduce((prev, { attributes }) => {
8288
+ const configFilter = React.useMemo(() => getConfigFilter(filterName, configFilters), [configFilters, filterName]);
8289
+ const dataSource = React.useMemo(() => getDataSource(configFilter?.relatedDataSource, dataSources), [configFilter?.relatedDataSource, dataSources]);
8290
+ const { minFromData, maxFromData } = React.useMemo(() => dataSource?.features?.reduce((prev, { attributes }) => {
7605
8291
  const date = getDate(attributes[configFilter?.attributeValue]);
7606
8292
  return {
7607
8293
  minFromData: !prev.minFromData ? date : date < prev.minFromData ? date : prev.minFromData,
7608
8294
  maxFromData: !prev.maxFromData ? date : date > prev.maxFromData ? date : prev.maxFromData
7609
8295
  };
7610
8296
  }, DEFAULT_FROM_DATA) || DEFAULT_FROM_DATA, [configFilter?.attributeValue, dataSource?.features]);
7611
- const minDate = react.useMemo(() => minValue || minFromData.toISOString(), [minFromData, minValue]);
7612
- const maxDate = react.useMemo(() => maxValue || maxFromData.toISOString(), [maxFromData, maxValue]);
7613
- const isEmpty = react.useMemo(() => lodash.isNil(value?.[0]) && lodash.isNil(value?.[1]), [value]);
7614
- const isDefault = react.useMemo(() => value?.[0] === minDate && value?.[1] === maxDate, [maxDate, minDate, value]);
7615
- const reset = react.useCallback(() => {
8297
+ const minDate = React.useMemo(() => minValue || minFromData.toISOString(), [minFromData, minValue]);
8298
+ const maxDate = React.useMemo(() => maxValue || maxFromData.toISOString(), [maxFromData, maxValue]);
8299
+ const isEmpty = React.useMemo(() => lodash.isNil(value?.[0]) && lodash.isNil(value?.[1]), [value]);
8300
+ const isDefault = React.useMemo(() => value?.[0] === minDate && value?.[1] === maxDate, [maxDate, minDate, value]);
8301
+ const reset = React.useCallback(() => {
7616
8302
  setValue([minDate, maxDate]);
7617
8303
  changeFilters({
7618
8304
  [filterName]: {
@@ -7622,15 +8308,15 @@ const RangeDateFilter = ({ type, filter }) => {
7622
8308
  }
7623
8309
  }, configFilter?.resetFilters);
7624
8310
  }, [changeFilters, configFilter?.resetFilters, filterName, maxDate, minDate]);
7625
- const onChange = react.useCallback((newValue) => {
8311
+ const onChange = React.useCallback((newValue) => {
7626
8312
  setValue([newValue[0].toISOString(), newValue[1].toISOString()]);
7627
8313
  }, []);
7628
- react.useEffect(() => {
8314
+ React.useEffect(() => {
7629
8315
  if (!lodash.isNil(minDate) && !lodash.isNil(maxDate)) {
7630
8316
  reset();
7631
8317
  }
7632
8318
  }, []); // eslint-disable-line
7633
- react.useEffect(() => {
8319
+ React.useEffect(() => {
7634
8320
  if (!isInit.current) {
7635
8321
  isInit.current = true;
7636
8322
  return;
@@ -7850,7 +8536,7 @@ const pieChartTooltipFromAttributes = (t, data, attributes) => {
7850
8536
  const pieChartTooltipFromRelatedFeatures = (t, data, relatedAttributes, layerInfo) => {
7851
8537
  if (!data?.length)
7852
8538
  return "";
7853
- return (jsxRuntime.jsx(uilibGl.ThemeProvider, { children: jsxRuntime.jsxs(ChartTooltip, { children: [jsxRuntime.jsx(ChartTooltipColor, { "$color": data[0].color }), jsxRuntime.jsxs("div", { children: [data[0].name, " \u00A0 ", formatChartRelatedValue(t, data[0].value, layerInfo, relatedAttributes)] })] }) }));
8539
+ return (jsxRuntime.jsx(uilibGl.ThemeProvider, { children: jsxRuntime.jsxs(ChartTooltip, { alignItems: "center", children: [jsxRuntime.jsx(ChartTooltipColor, { "$color": data[0].color }), jsxRuntime.jsxs("div", { children: [data[0].name, " \u00A0 ", formatChartRelatedValue(t, data[0].value, layerInfo, relatedAttributes)] })] }) }));
7854
8540
  };
7855
8541
 
7856
8542
  const removeDataSource = (dashboardConfiguration, name, pageIndex) => {
@@ -7908,9 +8594,9 @@ const tooltipValueFromRelatedFeatures = (t, value, relatedAttributes, layerInfo)
7908
8594
  };
7909
8595
 
7910
8596
  const useWidgetConfig = (type = exports.WidgetType.Dashboard) => {
7911
- const { config: configProp } = react.useContext(ConfigContext);
8597
+ const { config: configProp } = React.useContext(ConfigContext);
7912
8598
  const { projectInfo, layerInfo } = useWidgetContext(type);
7913
- const config = react.useMemo(() => {
8599
+ const config = React.useMemo(() => {
7914
8600
  if (configProp) {
7915
8601
  return configProp;
7916
8602
  }
@@ -7919,7 +8605,7 @@ const useWidgetConfig = (type = exports.WidgetType.Dashboard) => {
7919
8605
  }
7920
8606
  return (layerInfo?.configuration?.cardConfiguration || {});
7921
8607
  }, [configProp, layerInfo?.configuration?.cardConfiguration, projectInfo?.content?.dashboardConfiguration, type]);
7922
- const pages = react.useMemo(() => (getPagesFromConfig(config) || []), [config]);
8608
+ const pages = React.useMemo(() => (getPagesFromConfig(config) || []), [config]);
7923
8609
  return {
7924
8610
  config,
7925
8611
  pages,
@@ -7930,10 +8616,11 @@ const useWidgetPage = (type = exports.WidgetType.Dashboard) => {
7930
8616
  const { t } = useGlobalContext();
7931
8617
  const { pageIndex, changePage, projectInfo, updateProject } = useWidgetContext(type);
7932
8618
  const { config } = useWidgetConfig(type);
7933
- const currentPage = react.useMemo(() => {
8619
+ const currentPage = React.useMemo(() => {
7934
8620
  const page = getPagesFromConfig(config)?.[pageIndex - 1];
7935
8621
  return {
7936
8622
  ...(page || {}),
8623
+ header: page?.header || config.header,
7937
8624
  dataSources: [
7938
8625
  ...(page?.dataSources?.filter(({ name }) => !config.dataSources?.some(item => item.name === name)) || []),
7939
8626
  ...(config.dataSources || [])
@@ -7948,7 +8635,7 @@ const useWidgetPage = (type = exports.WidgetType.Dashboard) => {
7948
8635
  })) || []
7949
8636
  };
7950
8637
  }, [config, pageIndex]);
7951
- const updateConfigPage = react.useCallback((newConfig) => {
8638
+ const updateConfigPage = React.useCallback((newConfig) => {
7952
8639
  const newProjectInfo = JSON.parse(JSON.stringify(projectInfo));
7953
8640
  const pages = getPagesFromProjectInfo(newProjectInfo);
7954
8641
  if (!pages?.[pageIndex - 1])
@@ -7956,7 +8643,7 @@ const useWidgetPage = (type = exports.WidgetType.Dashboard) => {
7956
8643
  pages[pageIndex - 1] = { ...pages[pageIndex - 1], ...newConfig };
7957
8644
  updateProject(newProjectInfo);
7958
8645
  }, [projectInfo, pageIndex, updateProject]);
7959
- const addConfigPage = react.useCallback(() => {
8646
+ const addConfigPage = React.useCallback(() => {
7960
8647
  const newProjectInfo = JSON.parse(JSON.stringify(projectInfo));
7961
8648
  const pages = getPagesFromProjectInfo(newProjectInfo);
7962
8649
  if (!pages)
@@ -7965,7 +8652,7 @@ const useWidgetPage = (type = exports.WidgetType.Dashboard) => {
7965
8652
  updateProject(newProjectInfo);
7966
8653
  changePage(pages.length);
7967
8654
  }, [changePage, projectInfo, t, updateProject]);
7968
- const deleteConfigPage = react.useCallback((index) => {
8655
+ const deleteConfigPage = React.useCallback((index) => {
7969
8656
  const newProjectInfo = JSON.parse(JSON.stringify(projectInfo));
7970
8657
  const pages = getPagesFromProjectInfo(newProjectInfo);
7971
8658
  if (!pages?.[index] || pages.length === 1)
@@ -7983,10 +8670,10 @@ const useChartData = ({ element, type }) => {
7983
8670
  const { t } = useGlobalContext();
7984
8671
  const { dataSources, layerInfos, attributes } = useWidgetContext(type);
7985
8672
  const { currentPage } = useWidgetPage(type);
7986
- const relatedAttributes = react.useMemo(() => element?.options?.relatedDataSources || [], [element?.options?.relatedDataSources]);
7987
- const loading = react.useMemo(() => !!relatedAttributes?.length &&
8673
+ const relatedAttributes = React.useMemo(() => element?.options?.relatedDataSources || [], [element?.options?.relatedDataSources]);
8674
+ const loading = React.useMemo(() => !!relatedAttributes?.length &&
7988
8675
  !dataSources?.some(({ name }) => relatedAttributes.some(({ chartAxis, dataSourceName }) => chartAxis === "y" && dataSourceName === name)), [dataSources, relatedAttributes]);
7989
- const fetchedData = react.useMemo(() => {
8676
+ const fetchedData = React.useMemo(() => {
7990
8677
  const configDataSources = currentPage?.dataSources || [];
7991
8678
  const isRelated = !!relatedAttributes?.length;
7992
8679
  const filteredAttributes = relatedAttributes.filter(({ chartAxis }) => chartAxis === "y");
@@ -8039,8 +8726,8 @@ const useChartData = ({ element, type }) => {
8039
8726
  layerInfos,
8040
8727
  dataSources
8041
8728
  ]);
8042
- const [data, setData] = react.useState(fetchedData);
8043
- react.useEffect(() => {
8729
+ const [data, setData] = React.useState(fetchedData);
8730
+ React.useEffect(() => {
8044
8731
  if (!loading && fetchedData?.length) {
8045
8732
  setData(fetchedData);
8046
8733
  }
@@ -8048,17 +8735,34 @@ const useChartData = ({ element, type }) => {
8048
8735
  return { data, loading };
8049
8736
  };
8050
8737
 
8051
- const useDashboardHeader = (renderElement) => {
8738
+ const useHeaderRender = (elementConfig, type = exports.WidgetType.Dashboard) => {
8739
+ const { layerInfo, attributes, expandedContainers, selectedTabId, setSelectedTabId } = useWidgetContext(type);
8740
+ const { config } = useWidgetConfig(type);
8741
+ const { pageIndex } = useWidgetPage(type);
8742
+ return React.useMemo(() => getRenderElement({
8743
+ config,
8744
+ elementConfig,
8745
+ attributes,
8746
+ layerInfo,
8747
+ expandedContainers,
8748
+ selectedTabId,
8749
+ setSelectedTabId,
8750
+ pageIndex,
8751
+ }), [config, elementConfig, attributes, layerInfo, expandedContainers, selectedTabId, setSelectedTabId, pageIndex]);
8752
+ };
8753
+
8754
+ const useDashboardHeader = () => {
8052
8755
  const { themeName } = useGlobalContext();
8053
8756
  const { projectInfo } = useWidgetContext();
8054
8757
  const { currentPage } = useWidgetPage();
8055
8758
  const { alias, name } = projectInfo || {};
8056
8759
  const { id: pageId, header } = currentPage || {};
8057
- const image = react.useMemo(() => (header?.children?.some(({ id }) => id === "image") ? renderElement({ id: "image", wrap: false }) : null), [header?.children, renderElement]);
8058
- const icon = react.useMemo(() => header?.children?.some(({ id }) => id === "icon") ? (renderElement({ id: "icon", wrap: false })) : (jsxRuntime.jsx(uilibGl.Icon, { kind: "logo", style: { width: "2rem", height: "2rem" } })), [header?.children, renderElement]);
8059
- const title = react.useMemo(() => header?.children?.some(({ id }) => id === "title") ? renderElement({ id: "title" }) : currentPage?.options?.title, [header?.children, renderElement, currentPage?.options?.title]);
8060
- const tooltip = react.useMemo(() => (jsxRuntime.jsxs(uilibGl.Flex, { textAlign: "left", children: [jsxRuntime.jsx("div", { children: alias || name }), jsxRuntime.jsx(uilibGl.Flex, { opacity: 0.54, children: title })] })), [alias, name, title]);
8061
- const description = react.useMemo(() => (header?.children?.some(({ id }) => id === "description") ? renderElement({ id: "description" }) : ""), [header?.children, renderElement]);
8760
+ const renderElement = useHeaderRender(header);
8761
+ const image = React.useMemo(() => (header?.children?.some(({ id }) => id === "image") ? renderElement({ id: "image", wrap: false }) : null), [header?.children, renderElement]);
8762
+ const icon = React.useMemo(() => header?.children?.some(({ id }) => id === "icon") ? (renderElement({ id: "icon", wrap: false })) : (jsxRuntime.jsx(uilibGl.Icon, { kind: "logo", style: { width: "2rem", height: "2rem" } })), [header?.children, renderElement]);
8763
+ const title = React.useMemo(() => header?.children?.some(({ id }) => id === "title") ? renderElement({ id: "title" }) : currentPage?.options?.title, [header?.children, renderElement, currentPage?.options?.title]);
8764
+ const tooltip = React.useMemo(() => (jsxRuntime.jsxs(uilibGl.Flex, { textAlign: "left", children: [jsxRuntime.jsx("div", { children: alias || name }), jsxRuntime.jsx(uilibGl.Flex, { opacity: 0.54, children: title })] })), [alias, name, title]);
8765
+ const description = React.useMemo(() => (header?.children?.some(({ id }) => id === "description") ? renderElement({ id: "description" }) : ""), [header?.children, renderElement]);
8062
8766
  return {
8063
8767
  pageId,
8064
8768
  image,
@@ -8067,6 +8771,7 @@ const useDashboardHeader = (renderElement) => {
8067
8771
  tooltip,
8068
8772
  description,
8069
8773
  themeName,
8774
+ renderElement,
8070
8775
  };
8071
8776
  };
8072
8777
 
@@ -8076,8 +8781,8 @@ const useDataSources = ({ type: widgetType, config, attributes, filters, layerPa
8076
8781
  const { ewktGeometry, api } = useGlobalContext();
8077
8782
  const { dataSources } = useWidgetContext(widgetType);
8078
8783
  const { filters: configFilters, dataSources: configDataSources } = config || {};
8079
- const prevFilters = react.useRef({});
8080
- const getDataSourcePromises = react.useCallback(async ({ ds, query, parameters, layerName, limit, condition, url, resourceId, type }, newFilters, offset = 0) => {
8784
+ const prevFilters = React.useRef({});
8785
+ const getDataSourcePromises = React.useCallback(async ({ ds, query, parameters, layerName, limit, condition, url, resourceId, type }, newFilters, offset = 0) => {
8081
8786
  const selectedFilters = {
8082
8787
  ...(filters || {}),
8083
8788
  ...(newFilters || {}),
@@ -8139,7 +8844,7 @@ const useDataSources = ({ type: widgetType, config, attributes, filters, layerPa
8139
8844
  : [condition, query].filter(Boolean),
8140
8845
  });
8141
8846
  }, [filters, dataSources, configFilters, ewktGeometry, attributes, layerParams, eqlParameters]);
8142
- const getUpdatingDataSources = react.useCallback(() => {
8847
+ const getUpdatingDataSources = React.useCallback(() => {
8143
8848
  const diffFilterNames = filters
8144
8849
  ? Object.keys(filters).filter(key => Array.isArray(filters[key].value)
8145
8850
  ? filters[key].value.length !== prevFilters.current[key]?.value?.length ||
@@ -8156,7 +8861,7 @@ const useDataSources = ({ type: widgetType, config, attributes, filters, layerPa
8156
8861
  prevFilters.current = JSON.parse(JSON.stringify(filters));
8157
8862
  return filteredDataSources;
8158
8863
  }, [configDataSources, filters]);
8159
- const getUpdatedDataSources = react.useCallback((responses, currentDataSources, otherDataSources) => {
8864
+ const getUpdatedDataSources = React.useCallback((responses, currentDataSources, otherDataSources) => {
8160
8865
  const newDataSources = JSON.parse(JSON.stringify([...currentDataSources, ...otherDataSources]));
8161
8866
  responses.forEach((response, index) => {
8162
8867
  const isQueryDataSource = !lodash.isNil(currentDataSources[index].query) ||
@@ -8179,7 +8884,7 @@ const useDataSources = ({ type: widgetType, config, attributes, filters, layerPa
8179
8884
  });
8180
8885
  return newDataSources;
8181
8886
  }, []);
8182
- const zoomToLayersExtent = react.useCallback(async (layers) => {
8887
+ const zoomToLayersExtent = React.useCallback(async (layers) => {
8183
8888
  const newExtent = await api.layers.getBulkExtents({ srId: 4326 }, layers?.map(({ layerName, conditions, parameters }) => ({
8184
8889
  layerName,
8185
8890
  conditions,
@@ -8210,17 +8915,17 @@ const useDataSources = ({ type: widgetType, config, attributes, filters, layerPa
8210
8915
 
8211
8916
  const useDiffPage = (type) => {
8212
8917
  const { pageIndex } = useWidgetContext(type);
8213
- const prevPageIndex = react.useRef(pageIndex);
8918
+ const prevPageIndex = React.useRef(pageIndex);
8214
8919
  const isDiffPage = prevPageIndex.current !== pageIndex;
8215
- react.useEffect(() => {
8920
+ React.useEffect(() => {
8216
8921
  prevPageIndex.current = pageIndex;
8217
8922
  }, [pageIndex]);
8218
- return react.useMemo(() => isDiffPage, [isDiffPage]);
8923
+ return React.useMemo(() => isDiffPage, [isDiffPage]);
8219
8924
  };
8220
8925
 
8221
8926
  const useExpandableContainers = () => {
8222
- const [expandedContainers, setExpandedContainers] = react.useState({});
8223
- const expandContainer = react.useCallback((id, expanded) => {
8927
+ const [expandedContainers, setExpandedContainers] = React.useState({});
8928
+ const expandContainer = React.useCallback((id, expanded) => {
8224
8929
  setExpandedContainers(state => ({
8225
8930
  ...state,
8226
8931
  [id]: state?.[id] === undefined && expanded !== undefined ? !expanded : !state?.[id],
@@ -8244,11 +8949,11 @@ const useFeatureFilters = (type, filterName, items) => {
8244
8949
  const theme = styled.useTheme();
8245
8950
  const { filters, changeFilters } = useWidgetContext(type);
8246
8951
  const { currentPage } = useWidgetPage(type);
8247
- const configFilter = react.useMemo(() => getConfigFilter(filterName, currentPage.filters), [currentPage.filters, filterName]);
8952
+ const configFilter = React.useMemo(() => getConfigFilter(filterName, currentPage.filters), [currentPage.filters, filterName]);
8248
8953
  const hasAnyFilter = Array.isArray(filters?.[filterName]?.value)
8249
8954
  ? !!filters[filterName].value.length
8250
8955
  : !!filters?.[filterName]?.value;
8251
- const isFiltered = react.useCallback((name) => {
8956
+ const isFiltered = React.useCallback((name) => {
8252
8957
  const filterValue = filters[filterName]?.value || configFilter?.defaultValue;
8253
8958
  if (!Array.isArray(filterValue)) {
8254
8959
  return filterValue === name;
@@ -8258,7 +8963,7 @@ const useFeatureFilters = (type, filterName, items) => {
8258
8963
  ? filterValue.includes(name)
8259
8964
  : filterValue?.[0] <= name && filterValue?.[1] >= name));
8260
8965
  }, [configFilter?.defaultValue, configFilter?.valueType, filterName, filters]);
8261
- const formatFilterColor = react.useCallback((name, color, defaultColor = FEATURE_CARD_OTHER_COLOR) => {
8966
+ const formatFilterColor = React.useCallback((name, color, defaultColor = FEATURE_CARD_OTHER_COLOR) => {
8262
8967
  return isFiltered(name)
8263
8968
  ? color || theme.palette.primary
8264
8969
  : hasAnyFilter
@@ -8267,7 +8972,7 @@ const useFeatureFilters = (type, filterName, items) => {
8267
8972
  : defaultColor
8268
8973
  : color || theme.palette.primary;
8269
8974
  }, [hasAnyFilter, isFiltered, theme.palette.primary]);
8270
- const onFilter = react.useCallback((newValue) => {
8975
+ const onFilter = React.useCallback((newValue) => {
8271
8976
  const value = getFilterValue({
8272
8977
  selectedFilters: filters,
8273
8978
  configFilters: currentPage.filters,
@@ -8313,12 +9018,12 @@ const useFeatureFilters = (type, filterName, items) => {
8313
9018
 
8314
9019
  const useGetConfigLayer = () => {
8315
9020
  const { currentPage } = useWidgetPage();
8316
- return react.useCallback((layerName) => currentPage?.layers?.find(({ name }) => name === layerName), [currentPage?.layers]);
9021
+ return React.useCallback((layerName) => currentPage?.layers?.find(({ name }) => name === layerName), [currentPage?.layers]);
8317
9022
  };
8318
9023
 
8319
9024
  const useProjectDashboardInit = () => {
8320
9025
  const { projectInfo, updateProject } = useWidgetContext();
8321
- return react.useCallback(() => {
9026
+ return React.useCallback(() => {
8322
9027
  if (!lodash.isEmpty(projectInfo?.content?.dashboardConfiguration) || !projectInfo?.content?.items?.length) {
8323
9028
  return;
8324
9029
  }
@@ -8335,10 +9040,10 @@ const useRelatedDataSourceAttributes = ({ type = exports.WidgetType.Dashboard, e
8335
9040
  const { currentPage } = useWidgetPage(type);
8336
9041
  const { options } = elementConfig || {};
8337
9042
  const { relatedDataSource } = options || {};
8338
- const dataSource = react.useMemo(() => getDataSource(relatedDataSource, dataSources), [dataSources, relatedDataSource]);
8339
- const layerInfo = react.useMemo(() => getLayerInfoFromDataSources(layerInfos, currentPage?.dataSources, relatedDataSource) ||
9043
+ const dataSource = React.useMemo(() => getDataSource(relatedDataSource, dataSources), [dataSources, relatedDataSource]);
9044
+ const layerInfo = React.useMemo(() => getLayerInfoFromDataSources(layerInfos, currentPage?.dataSources, relatedDataSource) ||
8340
9045
  { layerDefinition: { attributes: dataSource?.attributeDefinition || {} } }, [currentPage?.dataSources, dataSource?.attributeDefinition, layerInfos, relatedDataSource]);
8341
- const attributes = react.useMemo(() => getFeatureAttributes(feature, layerInfo, dataSource), [dataSource, feature, layerInfo]);
9046
+ const attributes = React.useMemo(() => getFeatureAttributes(feature, layerInfo, dataSource), [dataSource, feature, layerInfo]);
8342
9047
  return { layerInfo, attributes, dataSource };
8343
9048
  };
8344
9049
 
@@ -8347,7 +9052,7 @@ const useRenderElement = (type = exports.WidgetType.Dashboard, elementConfig) =>
8347
9052
  const { selectedTabId, setSelectedTabId, expandedContainers, layerInfo, attributes } = useWidgetContext(type);
8348
9053
  const { config } = useWidgetConfig(type);
8349
9054
  const { pageIndex } = useWidgetPage(type);
8350
- return react.useMemo(() => getRenderElement({
9055
+ return React.useMemo(() => getRenderElement({
8351
9056
  t,
8352
9057
  type,
8353
9058
  config,
@@ -8392,7 +9097,7 @@ const useUpdateDataSource = ({ dataSource, config, filters, attributes, layerPar
8392
9097
  layerParams,
8393
9098
  eqlParameters,
8394
9099
  });
8395
- return react.useCallback(async (newFilters, offset = 0) => {
9100
+ return React.useCallback(async (newFilters, offset = 0) => {
8396
9101
  return getDataSourcePromises(dataSource, newFilters, offset).then((response) => {
8397
9102
  const responseDataSources = getUpdatedDataSources([response], [dataSource], dataSources);
8398
9103
  return {
@@ -8411,17 +9116,17 @@ const StackBar = ({ data, filterName, type, alias, options, renderElement, rende
8411
9116
  const { items, layerInfo, attributeName } = data?.[0] || {};
8412
9117
  const attribute = layerInfo?.layerDefinition.attributes[attributeName];
8413
9118
  const units = attribute?.stringFormat?.unitsLabel;
8414
- const total = react.useMemo(() => items?.reduce((result, { value }) => result + Number(value), 0) || 0, [items]);
8415
- const getWidth = react.useCallback(value => ((Number(value) / total) * 100).toFixed(2), [total]);
8416
- const renderGroupTooltip = react.useMemo(() => (jsxRuntime.jsx(uilibGl.ThemeProvider, { children: jsxRuntime.jsx(ChartTooltipTable, { cellPadding: 0, cellSpacing: 0, children: items?.map(({ name, value, color }, index) => (jsxRuntime.jsxs("tr", { children: [jsxRuntime.jsx("td", { children: jsxRuntime.jsxs(ChartTooltip, { children: [jsxRuntime.jsx(ChartTooltipColor, { "$color": color }), jsxRuntime.jsx(ChartTooltipName, { children: name })] }) }), jsxRuntime.jsx("td", { children: value })] }, index))) }) })), [items]);
8417
- const renderItem = react.useCallback(({ name, value, color }, ref) => (jsxRuntime.jsx(StackBarSection, { ref: ref, "$width": getWidth(value), "$height": height, "$color": color, cornerRadius: cornerRadius, hasAnyFilter: hasAnyFilter, isFiltered: isFiltered(name), onClick: filterName ? () => onFilter(name) : undefined })), [cornerRadius, filterName, getWidth, hasAnyFilter, height, isFiltered, onFilter]);
8418
- const renderItems = react.useMemo(() => (jsxRuntime.jsx(jsxRuntime.Fragment, { children: items?.map((item, index) => (jsxRuntime.jsx(react.Fragment, { children: groupTooltip ? (renderItem(item)) : (jsxRuntime.jsx(uilibGl.ThemeProvider, { children: jsxRuntime.jsx(uilibGl.Tooltip, { placement: "top", arrow: true, content: renderTooltip([item]), children: ref => renderItem(item, ref) }) })) }, index))) })), [groupTooltip, items, renderItem, renderTooltip]);
9119
+ const total = React.useMemo(() => items?.reduce((result, { value }) => result + Number(value), 0) || 0, [items]);
9120
+ const getWidth = React.useCallback(value => ((Number(value) / total) * 100).toFixed(2), [total]);
9121
+ const renderGroupTooltip = React.useMemo(() => (jsxRuntime.jsx(uilibGl.ThemeProvider, { children: jsxRuntime.jsx(ChartTooltipTable, { cellPadding: 0, cellSpacing: 0, children: items?.map(({ name, value, color }, index) => (jsxRuntime.jsxs("tr", { children: [jsxRuntime.jsx("td", { children: jsxRuntime.jsxs(ChartTooltip, { alignItems: "center", children: [jsxRuntime.jsx(ChartTooltipColor, { "$color": color }), jsxRuntime.jsx(ChartTooltipName, { children: name })] }) }), jsxRuntime.jsx("td", { children: value })] }, index))) }) })), [items]);
9122
+ const renderItem = React.useCallback(({ name, value, color }, ref) => (jsxRuntime.jsx(StackBarSection, { ref: ref, "$width": getWidth(value), "$height": height, "$color": color, cornerRadius: cornerRadius, hasAnyFilter: hasAnyFilter, isFiltered: isFiltered(name), onClick: filterName ? () => onFilter(name) : undefined })), [cornerRadius, filterName, getWidth, hasAnyFilter, height, isFiltered, onFilter]);
9123
+ const renderItems = React.useMemo(() => (jsxRuntime.jsx(jsxRuntime.Fragment, { children: items?.map((item, index) => (jsxRuntime.jsx(React.Fragment, { children: groupTooltip ? (renderItem(item)) : (jsxRuntime.jsx(uilibGl.ThemeProvider, { children: jsxRuntime.jsx(uilibGl.Tooltip, { placement: "top", arrow: true, content: renderTooltip([item]), children: ref => renderItem(item, ref) }) })) }, index))) })), [groupTooltip, items, renderItem, renderTooltip]);
8419
9124
  if (!total || !attribute)
8420
9125
  return null;
8421
9126
  return (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [(alias || showTotal) && (jsxRuntime.jsxs(StackBarHeader, { children: [jsxRuntime.jsx(StackBarAlias, { children: renderElement({ id: "alias" }) }), showTotal && (jsxRuntime.jsxs(StackBarTotal, { children: [jsxRuntime.jsx(StackBarValue, { children: formatAttributeValue({ t, type: attribute.type, value: total, stringFormat: attribute.stringFormat, noUnits: true }) }), !!units && jsxRuntime.jsx(StackBarUnits, { children: units })] }))] })), groupTooltip ? (jsxRuntime.jsx(uilibGl.Tooltip, { placement: "top", arrow: true, content: renderGroupTooltip, children: ref => jsxRuntime.jsx(StackBarContainer, { ref: ref, children: renderItems }) })) : (jsxRuntime.jsx(StackBarContainer, { children: renderItems }))] }));
8422
9127
  };
8423
9128
 
8424
- const Chart = react.memo(({ config, element, elementConfig, type, renderElement }) => {
9129
+ const Chart = React.memo(({ config, element, elementConfig, type, renderElement }) => {
8425
9130
  const theme = styled.useTheme();
8426
9131
  const { palette } = theme;
8427
9132
  const { t } = useGlobalContext();
@@ -8440,22 +9145,17 @@ const Chart = react.memo(({ config, element, elementConfig, type, renderElement
8440
9145
  const barWidth = options?.barWidth || DEFAULT_BAR_WIDTH;
8441
9146
  const cornerRadius = options?.cornerRadius || 0;
8442
9147
  const primaryColor = theme.palette.primary;
8443
- const relatedAttributes = react.useMemo(() => relatedDataSources || [], [relatedDataSources]);
8444
- const axes = react.useMemo(() => relatedAttributes.filter(({ chartAxis }) => chartAxis === "y"), [relatedAttributes]);
9148
+ const relatedAttributes = React.useMemo(() => relatedDataSources || [], [relatedDataSources]);
9149
+ const axes = React.useMemo(() => relatedAttributes.filter(({ chartAxis }) => chartAxis === "y"), [relatedAttributes]);
8445
9150
  const filterName = getChartFilterName(relatedDataSources);
8446
9151
  const isRelated = !!relatedAttributes?.length;
8447
9152
  const { data, loading } = useChartData({ element, type });
8448
9153
  const totalWidthFromBarWidth = (data[0]?.items?.length || 0) * (barWidth + padding);
8449
- const width = options?.width
8450
- ? options.width > MAX_CHART_WIDTH
8451
- ? MAX_CHART_WIDTH
8452
- : options.width
8453
- : isStackBar
9154
+ const width = options?.width ? options.width :
9155
+ isStackBar
8454
9156
  ? "100%"
8455
9157
  : isBarChart
8456
- ? totalWidthFromBarWidth < MAX_CHART_WIDTH
8457
- ? MAX_CHART_WIDTH
8458
- : totalWidthFromBarWidth
9158
+ ? totalWidthFromBarWidth || MAX_CHART_WIDTH
8459
9159
  : DEFAULT_CHART_WIDTH;
8460
9160
  const [customize] = useChartChange({
8461
9161
  dataSources: config.dataSources,
@@ -8467,37 +9167,37 @@ const Chart = react.memo(({ config, element, elementConfig, type, renderElement
8467
9167
  defaultColor: primaryColor
8468
9168
  });
8469
9169
  const { formatFilterColor, onFilter } = useFeatureFilters(type, filterName, data?.[0]?.items);
8470
- const formattedAttributes = react.useMemo(() => getFormattedAttributes(t, (data[0]?.items || []), attributes, element), [attributes, data, element, t]);
8471
- const totalSum = react.useMemo(() => isRelated
9170
+ const formattedAttributes = React.useMemo(() => getFormattedAttributes(t, (data[0]?.items || []), attributes, element), [attributes, data, element, t]);
9171
+ const totalSum = React.useMemo(() => isRelated
8472
9172
  ? getTotalFromRelatedFeatures((data[0]?.items || []))
8473
9173
  : getTotalFromAttributes(children, formattedAttributes), [data, children, formattedAttributes, isRelated]);
8474
- const totalWord = react.useMemo(() => options?.totalAttribute
9174
+ const totalWord = React.useMemo(() => options?.totalAttribute
8475
9175
  ? attributes?.find(({ name }) => name === options.totalAttribute)?.value
8476
9176
  : options?.totalWord, [attributes, options.totalAttribute, options?.totalWord]);
8477
- const renderPieChartTooltip = react.useCallback(item => isRelated
9177
+ const renderPieChartTooltip = React.useCallback(item => isRelated
8478
9178
  ? pieChartTooltipFromRelatedFeatures(t, item, relatedAttributes, data[0]?.layerInfo)
8479
9179
  : pieChartTooltipFromAttributes(t, item, formattedAttributes), [data, formattedAttributes, isRelated, relatedAttributes]);
8480
- const labels = react.useMemo(() => {
9180
+ const labels = React.useMemo(() => {
8481
9181
  if (!isLineChart)
8482
9182
  return [];
8483
9183
  const result = data?.map(item => item?.items?.map(({ name }) => name?.toString()))?.flat();
8484
9184
  return result?.filter((name, index) => result.indexOf(name) === index);
8485
9185
  }, [data, isLineChart]);
8486
- const renderLineChartTooltip = react.useCallback((items, { indexX }) => {
8487
- return (jsxRuntime.jsx(uilibGl.ThemeProvider, { children: jsxRuntime.jsxs(ChartTooltip, { children: [jsxRuntime.jsx("div", { children: labels[indexX] }), items
8488
- .filter(({ value }) => !isEmptyValue(value))
8489
- ?.map(({ layerName, attributeName, value, stroke }, index) => {
8490
- const layerInfo = layerInfos?.find(({ name }) => name === layerName);
8491
- const attribute = layerInfo?.layerDefinition.attributes[attributeName];
8492
- return (jsxRuntime.jsxs(ChartTooltipRow, { children: [jsxRuntime.jsx(ChartLegendColor$1, { "$color": stroke }), jsxRuntime.jsx(ChartLegendValue, { children: attribute ? formatAttributeValue({ t, type: attribute.type, value, stringFormat: attribute.stringFormat }) : value })] }, index));
8493
- })] }) }));
9186
+ const renderLineChartTooltip = React.useCallback((items, { indexX }) => {
9187
+ return (jsxRuntime.jsxs(Tooltip, { children: [jsxRuntime.jsx(ChartTooltipLabel, { children: labels[indexX] }), items
9188
+ .filter(({ value }) => !isEmptyValue(value))
9189
+ ?.map(({ layerName, attributeName, value, stroke }, index) => {
9190
+ const layerInfo = layerInfos?.find(({ name }) => name === layerName);
9191
+ const attribute = layerInfo?.layerDefinition.attributes[attributeName];
9192
+ return (jsxRuntime.jsxs(ChartTooltipRow, { children: [jsxRuntime.jsx(ChartLegendColor$1, { "$color": stroke }), jsxRuntime.jsx(ChartLegendValue, { children: attribute ? formatAttributeValue({ t, type: attribute.type, value, stringFormat: attribute.stringFormat }) : value })] }, index));
9193
+ })] }));
8494
9194
  }, [labels, layerInfos]);
8495
- const markers = react.useMemo(() => getChartMarkers(data[0]?.items, configMarkers), [data, configMarkers]);
8496
- const formatTooltipName = react.useCallback((name) => (isRelated ? name : tooltipNameFromAttributes(name, formattedAttributes)), [formattedAttributes, isRelated]);
8497
- const formatTooltipValue = react.useCallback((value, name) => isRelated
9195
+ const markers = React.useMemo(() => getChartMarkers(data[0]?.items, configMarkers), [data, configMarkers]);
9196
+ const formatTooltipName = React.useCallback((name) => (isRelated ? name : tooltipNameFromAttributes(name, formattedAttributes)), [formattedAttributes, isRelated]);
9197
+ const formatTooltipValue = React.useCallback((value, name) => isRelated
8498
9198
  ? tooltipValueFromRelatedFeatures(t, value, relatedAttributes, data[0]?.layerInfo)
8499
9199
  : tooltipValueFromAttributes(t, value, name, formattedAttributes), [data, formattedAttributes, isRelated, relatedAttributes]);
8500
- const renderChart = react.useMemo(() => {
9200
+ const renderChart = React.useMemo(() => {
8501
9201
  if (!element)
8502
9202
  return null;
8503
9203
  if (isLineChart) {
@@ -8514,16 +9214,16 @@ const Chart = react.memo(({ config, element, elementConfig, type, renderElement
8514
9214
  .flat()
8515
9215
  .map(({ values }) => values)
8516
9216
  .flat();
8517
- return (jsxRuntime.jsx(AnyChartWrapper, { height: height, children: jsxRuntime.jsx(charts.LineChart, { data: lineChartData, labels: labels, width: +width, height: height, xAxisPadding: 15, yAxisPadding: 0, min: Math.min(...chartValues), customYAxisSelection: yAxis => {
8518
- if (isHidedY) {
8519
- yAxis.remove();
8520
- }
8521
- }, customYAxis: yAxis => yAxis.ticks(4), renderTooltip: renderLineChartTooltip, customize: customize, dynamicTooltipEnable: true, stackedTooltip: true, drawGridX: !isHidedY, margin: {
8522
- top: 0,
8523
- right: 0,
8524
- bottom: 0,
8525
- left: 0
8526
- } }) }));
9217
+ return (jsxRuntime.jsxs(AnyChartWrapper, { height: height, children: [jsxRuntime.jsx(LineChartTooltipStyles, {}), jsxRuntime.jsx(charts.LineChart, { data: lineChartData, labels: labels, width: +width, height: height, xAxisPadding: 15, yAxisPadding: 0, min: Math.min(...chartValues), customYAxisSelection: yAxis => {
9218
+ if (isHidedY) {
9219
+ yAxis.remove();
9220
+ }
9221
+ }, customYAxis: yAxis => yAxis.ticks(4), renderTooltip: renderLineChartTooltip, customize: customize, dynamicTooltipEnable: true, stackedTooltip: true, tooltipClassName: "dashboardLineChartTooltip", drawGridX: !isHidedY, margin: {
9222
+ top: 0,
9223
+ right: 0,
9224
+ bottom: 0,
9225
+ left: 0
9226
+ } })] }));
8527
9227
  }
8528
9228
  if (isStackBar) {
8529
9229
  return (jsxRuntime.jsx(AnyChartWrapper, { height: height, children: jsxRuntime.jsx(StackBar, { data: data, filterName: filterName, type: type, alias: elementConfig?.children?.find(child => child.id === "alias"), options: options, renderTooltip: renderPieChartTooltip, renderElement: renderElement }) }));
@@ -8536,16 +9236,17 @@ const Chart = react.memo(({ config, element, elementConfig, type, renderElement
8536
9236
  color: formatFilterColor(item.name, item.color, defaultColor)
8537
9237
  })) || []), width: +width, height: +width, padAngle: angle, outerRadius: radius, cornerRadius: cornerRadius, renderTooltip: renderPieChartTooltip, withTooltip: true, onClick: filterName ? item => onFilter(item.name) : undefined, children: showTotal && jsxRuntime.jsx(PieChartCenter, { children: totalWord || roundTotalSum(+totalSum) }) }) }));
8538
9238
  }
9239
+ const markersMargin = options?.showMarkers ? 10 : 0;
8539
9240
  const margin = {
8540
- top: 0,
8541
- right: 0,
8542
- bottom: markers?.length ? 20 : 0,
8543
- left: 0
9241
+ top: markersMargin,
9242
+ right: markersMargin,
9243
+ bottom: markers?.length ? 20 : markersMargin,
9244
+ left: markersMargin
8544
9245
  };
8545
- return (jsxRuntime.jsx(BarChartWrapper, { height: height + margin.bottom, children: jsxRuntime.jsx(BarChartContainer, { children: jsxRuntime.jsx(charts.BarChart, { data: getDataFromFilterItems(data[0]?.items), colors: getColorsFromFilterItems(data[0]?.items, defaultColor, formatFilterColor), minValue: getMinValueFromFilterItems(data[0]?.items), markers: markers, width: +width, height: height, barWidth: barWidth, barPadding: padding, customXAxis: axis => axis.remove(), customYAxis: axis => axis.remove(), customBars: ({ bars }) => {
8546
- bars.attr("rx", radius);
8547
- bars.attr("ry", radius);
8548
- }, margin: margin, xAxisPadding: 0, yAxisPadding: 0, formatTooltipValue: formatTooltipValue, formatTooltipName: formatTooltipName, onBarClick: filterName ? item => onFilter(item.name) : undefined, hideTooltipGroupName: true, dynamicTooltipEnable: true, isBarTooltip: true }) }) }));
9246
+ return (jsxRuntime.jsx(BarChartWrapper, { height: height + margin.bottom, children: jsxRuntime.jsxs(BarChartContainer, { children: [jsxRuntime.jsx(BarChartStyles, {}), jsxRuntime.jsx(charts.BarChart, { data: getDataFromFilterItems(data[0]?.items), colors: getColorsFromFilterItems(data[0]?.items, defaultColor, formatFilterColor), minValue: getMinValueFromFilterItems(data[0]?.items), markers: markers, width: +width, height: height, barWidth: barWidth, barPadding: padding, className: "dashboardBarChart", customXAxis: axis => axis.remove(), customYAxis: axis => !options?.showMarkers && axis.remove(), customBars: ({ bars }) => {
9247
+ bars.attr("rx", radius);
9248
+ bars.attr("ry", radius);
9249
+ }, margin: margin, xAxisPadding: 0, yAxisPadding: 0, formatTooltipValue: formatTooltipValue, formatTooltipName: formatTooltipName, hideTooltipGroupName: true, dynamicTooltipEnable: true, isBarTooltip: true, onBarClick: filterName ? item => onFilter(item.name) : undefined })] }) }));
8549
9250
  }, [
8550
9251
  element,
8551
9252
  isLineChart,
@@ -8588,16 +9289,16 @@ const Chart = react.memo(({ config, element, elementConfig, type, renderElement
8588
9289
  const ChartLegend = ({ data, chartElement, type, twoColumns, loading }) => {
8589
9290
  const { t } = useGlobalContext();
8590
9291
  const { sliceItems, showMore, onShowMore } = useShownOtherItems(chartElement?.options);
8591
- const axes = react.useMemo(() => chartElement?.options?.relatedDataSources?.filter(({ chartAxis }) => chartAxis === "y"), [chartElement?.options?.relatedDataSources]);
9292
+ const axes = React.useMemo(() => chartElement?.options?.relatedDataSources?.filter(({ chartAxis }) => chartAxis === "y"), [chartElement?.options?.relatedDataSources]);
8592
9293
  const { isFiltered, hasAnyFilter, onFilter } = useFeatureFilters(type, axes?.[0]?.filterName);
8593
9294
  const { fontColor, shownItems, hideEmpty } = chartElement?.options || {};
8594
9295
  const showMoreBtn = !!shownItems && data.length > shownItems;
8595
9296
  if (!loading && !data?.length && !!hideEmpty)
8596
9297
  return null;
8597
- return (jsxRuntime.jsxs(uilibGl.Flex, { opacity: loading ? FILTERED_VALUE_OPACITY / 100 : 1, children: [jsxRuntime.jsx(ChartLegendContainer, { twoColumns: twoColumns, children: sliceItems(data)?.map(({ name, color }, index) => (jsxRuntime.jsxs(ChartLegendItem, { hasAnyFilter: hasAnyFilter, isFiltered: isFiltered(name), onClick: axes?.[0]?.filterName ? () => onFilter(name) : undefined, children: [jsxRuntime.jsx(ChartLegendColor, { color: color }), jsxRuntime.jsx(ChartLegendName, { "$fontColor": fontColor, children: name })] }, index))) }), showMoreBtn && (jsxRuntime.jsx(uilibGl.LegendToggler, { onClick: onShowMore, toggled: showMore, children: t("other", { ns: "dashboard" }) }))] }));
9298
+ return (jsxRuntime.jsxs(uilibGl.Flex, { flexDirection: "column", opacity: loading ? FILTERED_VALUE_OPACITY / 100 : 1, children: [jsxRuntime.jsx(ChartLegendContainer, { twoColumns: twoColumns, children: sliceItems(data)?.map(({ name, color }, index) => (jsxRuntime.jsxs(ChartLegendItem, { hasAnyFilter: hasAnyFilter, isFiltered: isFiltered(name), onClick: axes?.[0]?.filterName ? () => onFilter(name) : undefined, children: [jsxRuntime.jsx(ChartLegendColor, { color: color }), jsxRuntime.jsx(ChartLegendName, { "$fontColor": fontColor, children: name })] }, index))) }), showMoreBtn && (jsxRuntime.jsx(uilibGl.LegendToggler, { onClick: onShowMore, toggled: showMore, children: t("other", { ns: "dashboard" }) }))] }));
8598
9299
  };
8599
9300
 
8600
- const ContainerChildren = react.memo(({ items, isColumn, isMain, renderElement }) => {
9301
+ const ContainerChildren = React.memo(({ items, isColumn, isMain, renderElement }) => {
8601
9302
  return (jsxRuntime.jsx(jsxRuntime.Fragment, { children: items
8602
9303
  ?.filter(item => !["title", "icon", "titleIcon"].includes(item.id))
8603
9304
  ?.map((item, index) => {
@@ -8606,19 +9307,19 @@ const ContainerChildren = react.memo(({ items, isColumn, isMain, renderElement }
8606
9307
  index,
8607
9308
  wrap: isColumn || isMain,
8608
9309
  });
8609
- return isMain ? (jsxRuntime.jsx(ContainerWrapper, { "data-id": item.id, "data-templatename": item.templateName, children: jsxRuntime.jsx(DashboardWrapper, { children: render }) }, index)) : (jsxRuntime.jsx(react.Fragment, { children: render }, index));
9310
+ return isMain ? (jsxRuntime.jsx(ContainerWrapper, { "data-id": item.id, "data-templatename": item.templateName, children: jsxRuntime.jsx(DashboardWrapper, { children: render }) }, index)) : (jsxRuntime.jsx(React.Fragment, { children: render }, index));
8610
9311
  }) }));
8611
9312
  });
8612
9313
 
8613
- const DashboardLoading = react.memo(() => {
9314
+ const DashboardLoading = React.memo(() => {
8614
9315
  return (jsxRuntime.jsx(ContainerWrapper, { children: jsxRuntime.jsx(DashboardWrapper, { children: jsxRuntime.jsx(DashboardPlaceholderWrap, { children: jsxRuntime.jsxs(DashboardPlaceholder, { isLoading: true, children: [jsxRuntime.jsx(uilibGl.Icon, { kind: "dashboard_loading" }), jsxRuntime.jsx(uilibGl.Flex, { width: "8rem", children: jsxRuntime.jsx(uilibGl.LinearProgress, {}) })] }) }) }) }));
8615
9316
  });
8616
9317
 
8617
- const Dashboard = react.memo(({ type = exports.WidgetType.Dashboard, config }) => {
9318
+ const Dashboard = React.memo(({ type = exports.WidgetType.Dashboard, config }) => {
8618
9319
  const { dataSources, isLoading } = useWidgetContext(type);
8619
9320
  const { currentPage } = useWidgetPage();
8620
9321
  const isDiffPage = useDiffPage(type);
8621
- const dataSourceLoading = react.useMemo(() => currentPage?.dataSources?.length && !dataSources?.length && isLoading, [currentPage?.dataSources?.length, dataSources?.length, isLoading]);
9322
+ const dataSourceLoading = React.useMemo(() => currentPage?.dataSources?.length && !dataSources?.length && isLoading, [currentPage?.dataSources?.length, dataSources?.length, isLoading]);
8622
9323
  if (dataSourceLoading || isDiffPage) {
8623
9324
  return (jsxRuntime.jsx(DashboardLoading, {}));
8624
9325
  }
@@ -8635,12 +9336,18 @@ const DashboardCheckbox = ({ title, checked, onChange, }) => {
8635
9336
  return (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [jsxRuntime.jsx(AttributeLabel, { isEdit: !!onChange, forCheckbox: !!onChange, children: title }), onChange ? jsxRuntime.jsx(CardCheckbox, { checked: checked, label: label, onChange: onChange }) : label] }));
8636
9337
  };
8637
9338
 
9339
+ const DashboardHeader = React.memo(() => {
9340
+ const { currentPage } = useWidgetPage();
9341
+ const HeaderTemplate = React.useMemo(() => getDashboardHeader(currentPage?.header?.templateName), [currentPage?.header?.templateName]);
9342
+ return jsxRuntime.jsx(HeaderTemplate, {});
9343
+ });
9344
+
8638
9345
  const DataSourceError = ({ name }) => {
8639
9346
  const { t } = useGlobalContext();
8640
9347
  return (jsxRuntime.jsxs(DataSourceErrorContainer, { children: [jsxRuntime.jsx(uilibGl.Icon, { kind: "alert" }), jsxRuntime.jsxs("div", { children: [t("block", { ns: "dashboard" }), " \"", name, "\" ", t("isNotLoaded", { ns: "dashboard" })] })] }));
8641
9348
  };
8642
9349
 
8643
- const ExpandableTitle = react.memo(({ elementConfig, type, renderElement }) => {
9350
+ const ExpandableTitle = React.memo(({ elementConfig, type, renderElement }) => {
8644
9351
  const { expandedContainers } = useWidgetContext(type);
8645
9352
  const { id, options, templateName, children } = elementConfig || {};
8646
9353
  const { expandable, fontColor, expanded, layerNames } = options || {};
@@ -8651,6 +9358,50 @@ const ExpandableTitle = react.memo(({ elementConfig, type, renderElement }) => {
8651
9358
  return (jsxRuntime.jsx(TitleContainer, { containerId: id, elementConfig: titleElement, templateName: templateName, layerNames: layerNames, fontColor: fontColor, expandable: expandable, expanded: expanded, type: type, isVisible: isVisible, renderElement: renderElement }));
8652
9359
  });
8653
9360
 
9361
+ const FeatureCardButtons = () => {
9362
+ const { closeFeatureCard } = useWidgetContext(exports.WidgetType.FeatureCard);
9363
+ return (jsxRuntime.jsx(uilibGl.FlexSpan, { children: jsxRuntime.jsx(uilibGl.IconButton, { kind: "close", style: {
9364
+ padding: 0,
9365
+ width: "1rem",
9366
+ }, onClick: closeFeatureCard }) }));
9367
+ };
9368
+
9369
+ const FeatureCardHeader = () => {
9370
+ const { currentPage } = useWidgetPage(exports.WidgetType.FeatureCard);
9371
+ const HeaderTemplate = getFeatureCardHeader(currentPage?.header?.templateName);
9372
+ return jsxRuntime.jsx(HeaderTemplate, {});
9373
+ };
9374
+
9375
+ const FeatureCardTitle = ({ title, description }) => {
9376
+ const { t } = useGlobalContext();
9377
+ const { attributes, layerInfo, feature } = useWidgetContext(exports.WidgetType.FeatureCard);
9378
+ const zoomToFeatures = useZoomToFeatures();
9379
+ const { alias, name, layerDefinition, titleAttribute } = layerInfo || {};
9380
+ const resultDescription = description || alias || name || "";
9381
+ const resultTitle = React.useMemo(() => {
9382
+ const layerDefinitionAttribute = layerDefinition && attributes?.length
9383
+ ? attributes.find(item => item.name === layerDefinition.titleAttribute)
9384
+ : null;
9385
+ const layerDefinitionTitle = layerDefinition?.titleAttribute && layerDefinitionAttribute?.value
9386
+ ? layerDefinitionAttribute.value
9387
+ : null;
9388
+ const sourceTitleAttribute = titleAttribute && attributes?.length ? attributes.find(item => item.name === titleAttribute) : null;
9389
+ const sourceTitle = titleAttribute && sourceTitleAttribute?.value ? sourceTitleAttribute.value : null;
9390
+ return (title ||
9391
+ (!!sourceTitle &&
9392
+ formatAttributeValue({ t, type: sourceTitleAttribute.type, value: sourceTitle, stringFormat: sourceTitleAttribute.stringFormat })) ||
9393
+ (!!layerDefinitionTitle &&
9394
+ formatAttributeValue({
9395
+ t,
9396
+ type: layerDefinitionAttribute.type,
9397
+ value: layerDefinitionTitle,
9398
+ stringFormat: layerDefinitionAttribute.stringFormat,
9399
+ })) ||
9400
+ feature?.id);
9401
+ }, [attributes, feature?.id, layerDefinition, title, titleAttribute]);
9402
+ return (jsxRuntime.jsxs(HeaderTitleContainer, { children: [jsxRuntime.jsx(FeatureTitleContainer, { clickable: true, children: jsxRuntime.jsx(uilibGl.Tooltip, { arrow: true, placement: "top", content: t("zoomToFeature", { ns: "dashboard" }), delay: [600, 0], children: ref => (jsxRuntime.jsx(uilibGl.FlexSpan, { ref: ref, onClick: () => zoomToFeatures([feature]), children: resultTitle })) }) }), jsxRuntime.jsx(LayerDescription, { title: resultDescription, children: resultDescription })] }));
9403
+ };
9404
+
8654
9405
  const HiddenFilters = styled(uilibGl.Flex) `
8655
9406
  flex-wrap: wrap;
8656
9407
  margin-top: -1.25rem;
@@ -8673,15 +9424,15 @@ function isIsoDate(value) {
8673
9424
  const d = new Date(value);
8674
9425
  return !isNaN(d.getTime());
8675
9426
  }
8676
- const HiddenTitleItems = react.memo(({ elementConfig, config, type, filter }) => {
9427
+ const HiddenTitleItems = React.memo(({ elementConfig, config, type, filter }) => {
8677
9428
  const { filters, dataSources, changeFilters, expandedContainers } = useWidgetContext(type);
8678
9429
  const { filters: configFilters } = config || {};
8679
9430
  const { id, options, children } = elementConfig || {};
8680
9431
  const { fontColor, expandable, expanded } = options || {};
8681
9432
  const isVisible = isVisibleContainer(id, expandable, expanded, expandedContainers);
8682
- const filterItems = react.useMemo(() => children?.filter(child => child.options?.filterName), [children]);
8683
- const configFilter = react.useMemo(() => getConfigFilter(filter, configFilters), [configFilters, filter]);
8684
- const clearFilter = react.useCallback((name, value) => {
9433
+ const filterItems = React.useMemo(() => children?.filter(child => child.options?.filterName), [children]);
9434
+ const configFilter = React.useMemo(() => getConfigFilter(filter, configFilters), [configFilters, filter]);
9435
+ const clearFilter = React.useCallback((name, value) => {
8685
9436
  changeFilters({
8686
9437
  [name]: {
8687
9438
  value: value &&
@@ -8693,26 +9444,26 @@ const HiddenTitleItems = react.memo(({ elementConfig, config, type, filter }) =>
8693
9444
  },
8694
9445
  }, configFilter?.resetFilters);
8695
9446
  }, [changeFilters, configFilter?.resetFilters, configFilter?.valueType, filters]);
8696
- const selectedItems = react.useMemo(() => getFilterSelectedItems(filterItems, filters, configFilters), [configFilters, filters, filterItems]);
8697
- const isLoading = react.useMemo(() => elementConfig?.children?.some(child => {
9447
+ const selectedItems = React.useMemo(() => getFilterSelectedItems(filterItems, filters, configFilters), [configFilters, filters, filterItems]);
9448
+ const isLoading = React.useMemo(() => elementConfig?.children?.some(child => {
8698
9449
  const currentFilter = configFilters?.find(({ name }) => name === child.options?.filterName);
8699
9450
  return (!!currentFilter &&
8700
9451
  !!child.options?.filterName &&
8701
9452
  !dataSources?.some(({ name }) => name === currentFilter.relatedDataSource));
8702
9453
  }), [configFilters, dataSources, elementConfig?.children]);
8703
- const chipProps = react.useMemo(() => ({
9454
+ const chipProps = React.useMemo(() => ({
8704
9455
  $fontSize: "0.75rem",
8705
9456
  $fontColor: fontColor,
8706
9457
  $bgColor: fontColor ? transparentizeColor(fontColor, 6) : null,
8707
9458
  $radius: "0.25rem",
8708
9459
  }), [fontColor]);
8709
- const formatValue = react.useCallback((value) => {
9460
+ const formatValue = React.useCallback((value) => {
8710
9461
  return typeof value === "object" ? uilibGl.dateFormat(value) : isIsoDate(value) ? uilibGl.dateFormat(new Date(value)) : value;
8711
9462
  }, []);
8712
- const formatJoin = react.useCallback((valueType) => {
9463
+ const formatJoin = React.useCallback((valueType) => {
8713
9464
  return valueType === "range" ? " - " : ", ";
8714
9465
  }, []);
8715
- const renderItem = react.useCallback((filterName, value, index) => {
9466
+ const renderItem = React.useCallback((filterName, value, index) => {
8716
9467
  return (jsxRuntime.jsxs(DashboardChip$1, { ...chipProps, children: [value, " ", jsxRuntime.jsx(uilibGl.IconButton, { kind: "close", onClick: () => clearFilter(filterName, value) })] }, index));
8717
9468
  }, [chipProps, clearFilter]);
8718
9469
  if (isLoading || isVisible || !selectedItems.length)
@@ -8749,7 +9500,7 @@ const PageNavigator = styled(uilibGl.Flex) `
8749
9500
  }
8750
9501
  `;
8751
9502
 
8752
- const Pagination = react.memo(({ type }) => {
9503
+ const Pagination = React.memo(({ type = exports.WidgetType.Dashboard }) => {
8753
9504
  const { nextPage, prevPage } = useWidgetContext(type);
8754
9505
  const { pages } = useWidgetConfig(type);
8755
9506
  return (jsxRuntime.jsxs(PageNavigator, { children: [jsxRuntime.jsx(uilibGl.IconButton, { kind: "prev", onClick: () => prevPage(pages.length) }), jsxRuntime.jsx(uilibGl.IconButton, { kind: "next", onClick: () => nextPage(pages.length) })] }));
@@ -8784,10 +9535,10 @@ const StyledSvg = styled.div `
8784
9535
  ${({ $fontColor }) => !!$fontColor && StyledSvgColorMixin};
8785
9536
  `;
8786
9537
 
8787
- const SvgImage = react.memo(({ url, width, height, fontColor }) => {
8788
- const [svg, setSvg] = react.useState(null);
8789
- const [loading, setLoading] = react.useState(false);
8790
- react.useEffect(() => {
9538
+ const SvgImage = React.memo(({ url, width, height, fontColor }) => {
9539
+ const [svg, setSvg] = React.useState(null);
9540
+ const [loading, setLoading] = React.useState(false);
9541
+ React.useEffect(() => {
8791
9542
  if (!url || loading) {
8792
9543
  return;
8793
9544
  }
@@ -8799,7 +9550,7 @@ const SvgImage = react.memo(({ url, width, height, fontColor }) => {
8799
9550
  setLoading(false);
8800
9551
  });
8801
9552
  }, [url]);
8802
- react.useEffect(() => () => {
9553
+ React.useEffect(() => () => {
8803
9554
  setSvg(null);
8804
9555
  }, []);
8805
9556
  if (!svg?.endsWith("</svg>"))
@@ -8807,11 +9558,11 @@ const SvgImage = react.memo(({ url, width, height, fontColor }) => {
8807
9558
  return (jsxRuntime.jsx(StyledSvg, { "$width": width, "$height": height, "$fontColor": fontColor, dangerouslySetInnerHTML: { __html: svg } }));
8808
9559
  });
8809
9560
 
8810
- const TextTrim = react.memo(({ maxLength, expandable, lineBreak, children }) => {
9561
+ const TextTrim = React.memo(({ maxLength, expandable, lineBreak, children }) => {
8811
9562
  const { t } = useGlobalContext();
8812
9563
  const [expanded, toggleExpanded] = useToggle();
8813
9564
  const text = children?.toString();
8814
- const formatValue = react.useCallback((value) => {
9565
+ const formatValue = React.useCallback((value) => {
8815
9566
  if (!lineBreak)
8816
9567
  return value;
8817
9568
  return jsxRuntime.jsx("div", { dangerouslySetInnerHTML: { __html: lodash.unescape(value).split(lineBreak).join("<br />") } });
@@ -8884,7 +9635,7 @@ const getClientStyleItemPrefixSuffix = (geometryType, type) => {
8884
9635
  const VectorLayer = ({ layer, tileUrl, visible, beforeId, getLayerTempStyle, }) => {
8885
9636
  const clientStyle = layer?.configuration?.clientStyle;
8886
9637
  const { idAttribute, geometryType } = layer.layerDefinition || {};
8887
- const renderLayerByGeometryType = react.useCallback(() => {
9638
+ const renderLayerByGeometryType = React.useCallback(() => {
8888
9639
  const visibility = visible ? "visible" : "none";
8889
9640
  switch (geometryType) {
8890
9641
  case api.GeometryType.Point:
@@ -8951,7 +9702,7 @@ const VectorLayer = ({ layer, tileUrl, visible, beforeId, getLayerTempStyle, })
8951
9702
  visible,
8952
9703
  getLayerTempStyle,
8953
9704
  ]);
8954
- const renderClientStyle = react.useCallback(() => {
9705
+ const renderClientStyle = React.useCallback(() => {
8955
9706
  return clientStyle.items.map(mockItem => {
8956
9707
  const prefixSuffix = getClientStyleItemPrefixSuffix(geometryType, mockItem.type);
8957
9708
  const isExtrusionItem = mockItem.type === "fill-extrusion";
@@ -8999,7 +9750,7 @@ const VectorLayer = ({ layer, tileUrl, visible, beforeId, getLayerTempStyle, })
8999
9750
  };
9000
9751
 
9001
9752
  const Layer = ({ layer, layerType, visible, beforeId, tileUrl, getLayerTempStyle, onMount = () => null, }) => {
9002
- react.useEffect(onMount, []); // eslint-disable-line
9753
+ React.useEffect(onMount, []); // eslint-disable-line
9003
9754
  if (!layer) {
9004
9755
  return null;
9005
9756
  }
@@ -9042,23 +9793,6 @@ const Map$1 = ({ zIndex, lowerSiblings, upperSiblings, onError, children, ...res
9042
9793
  }, children: children }), upperSiblings] }));
9043
9794
  };
9044
9795
 
9045
- exports.ThemeName = void 0;
9046
- (function (ThemeName) {
9047
- ThemeName["Light"] = "light";
9048
- ThemeName["Dark"] = "dark";
9049
- })(exports.ThemeName || (exports.ThemeName = {}));
9050
-
9051
- exports.TmsType = void 0;
9052
- (function (TmsType) {
9053
- TmsType["WMS"] = "WMS";
9054
- TmsType["TMS"] = "TMS";
9055
- TmsType["ArcGIS"] = "ArcGIS";
9056
- })(exports.TmsType || (exports.TmsType = {}));
9057
- exports.EditGeometryType = void 0;
9058
- (function (EditGeometryType) {
9059
- EditGeometryType["Raster"] = "raster";
9060
- })(exports.EditGeometryType || (exports.EditGeometryType = {}));
9061
-
9062
9796
  exports.AddFeatureButton = AddFeatureButton;
9063
9797
  exports.AddFeatureContainer = AddFeatureContainer;
9064
9798
  exports.AttributeGalleryContainer = AttributeGalleryContainer;
@@ -9099,6 +9833,8 @@ exports.DashboardCheckbox = DashboardCheckbox;
9099
9833
  exports.DashboardChip = DashboardChip;
9100
9834
  exports.DashboardContent = DashboardContent;
9101
9835
  exports.DashboardContext = DashboardContext;
9836
+ exports.DashboardDefaultHeader = DashboardDefaultHeader;
9837
+ exports.DashboardHeader = DashboardHeader;
9102
9838
  exports.DashboardPlaceholder = DashboardPlaceholder;
9103
9839
  exports.DashboardPlaceholderWrap = DashboardPlaceholderWrap;
9104
9840
  exports.DashboardProvider = DashboardProvider;
@@ -9109,6 +9845,7 @@ exports.DataSourceErrorContainer = DataSourceErrorContainer;
9109
9845
  exports.DataSourceInnerContainer = DataSourceInnerContainer;
9110
9846
  exports.DataSourceProgressContainer = DataSourceProgressContainer;
9111
9847
  exports.DefaultAttributesContainer = DefaultAttributesContainer;
9848
+ exports.DefaultHeaderContainer = DefaultHeaderContainer;
9112
9849
  exports.DividerContainer = DividerContainer;
9113
9850
  exports.ElementButton = ElementButton;
9114
9851
  exports.ElementCamera = ElementCamera;
@@ -9127,24 +9864,40 @@ exports.FEATURE_CARD_DEFAULT_COLORS = FEATURE_CARD_DEFAULT_COLORS;
9127
9864
  exports.FEATURE_CARD_OTHER_COLOR = FEATURE_CARD_OTHER_COLOR;
9128
9865
  exports.FILTERED_VALUE_OPACITY = FILTERED_VALUE_OPACITY;
9129
9866
  exports.FILTER_PREFIX = FILTER_PREFIX;
9867
+ exports.FeatureCardButtons = FeatureCardButtons;
9130
9868
  exports.FeatureCardContext = FeatureCardContext;
9869
+ exports.FeatureCardDefaultHeader = FeatureCardDefaultHeader;
9870
+ exports.FeatureCardGradientHeader = FeatureCardGradientHeader;
9871
+ exports.FeatureCardHeader = FeatureCardHeader;
9872
+ exports.FeatureCardIconHeader = FeatureCardIconHeader;
9131
9873
  exports.FeatureCardProvider = FeatureCardProvider;
9874
+ exports.FeatureCardSlideshowHeader = FeatureCardSlideshowHeader;
9875
+ exports.FeatureCardTitle = FeatureCardTitle;
9876
+ exports.FeatureControls = FeatureControls;
9877
+ exports.FeatureTitleContainer = FeatureTitleContainer;
9132
9878
  exports.FiltersContainer = FiltersContainer;
9133
9879
  exports.GEOMETRY_ATTRIBUTE = GEOMETRY_ATTRIBUTE;
9134
9880
  exports.GlobalContext = GlobalContext;
9135
9881
  exports.GlobalProvider = GlobalProvider;
9882
+ exports.Header = Header;
9883
+ exports.HeaderContainer = HeaderContainer;
9884
+ exports.HeaderFrontView = HeaderFrontView;
9885
+ exports.HeaderTitleContainer = HeaderTitleContainer;
9136
9886
  exports.HiddenTitleItems = HiddenTitleItems;
9137
9887
  exports.IconContainer = IconContainer;
9138
9888
  exports.ImageContainer = ImageContainer;
9139
9889
  exports.LEFT_PANEL_HEADER_HEIGHT = LEFT_PANEL_HEADER_HEIGHT;
9140
9890
  exports.Layer = Layer;
9891
+ exports.LayerDescription = LayerDescription;
9141
9892
  exports.LayerGroup = LayerGroup;
9142
9893
  exports.LayerGroupList = LayerGroupList;
9894
+ exports.LayerIcon = LayerIcon;
9143
9895
  exports.LayerListContainer = LayerListContainer;
9144
9896
  exports.LayerTree = LayerTree;
9145
9897
  exports.LayersContainer = LayersContainer;
9146
9898
  exports.LayersListWrapper = LayersListWrapper;
9147
9899
  exports.LinearProgressContainer = LinearProgressContainer;
9900
+ exports.LogoContainer = LogoContainer;
9148
9901
  exports.MAX_CHART_WIDTH = MAX_CHART_WIDTH;
9149
9902
  exports.Map = Map$1;
9150
9903
  exports.MapContext = MapContext;
@@ -9154,6 +9907,7 @@ exports.NUMERIC_ATTRIBUTE_TYPES = NUMERIC_ATTRIBUTE_TYPES;
9154
9907
  exports.NoLiveSnapshotContainer = NoLiveSnapshotContainer;
9155
9908
  exports.OneColumnContainer = OneColumnContainer;
9156
9909
  exports.PageNavigator = PageNavigator;
9910
+ exports.PageTitle = PageTitle;
9157
9911
  exports.PagesContainer = PagesContainer;
9158
9912
  exports.Pagination = Pagination;
9159
9913
  exports.PresentationHeader = PresentationHeader;
@@ -9167,7 +9921,7 @@ exports.RoundedBackgroundContainer = RoundedBackgroundContainer;
9167
9921
  exports.ServerNotificationsContext = ServerNotificationsContext;
9168
9922
  exports.ServerNotificationsProvider = ServerNotificationsProvider;
9169
9923
  exports.SlideshowContainer = SlideshowContainer;
9170
- exports.SmallPreviewContainer = SmallPreviewContainer;
9924
+ exports.SmallPreviewContainer = SmallPreviewContainer$1;
9171
9925
  exports.SmallPreviewControl = SmallPreviewControl;
9172
9926
  exports.SmallPreviewCounter = SmallPreviewCounter;
9173
9927
  exports.SmallPreviewImages = SmallPreviewImages;
@@ -9179,6 +9933,8 @@ exports.TIME_ZONE_FORMAT = TIME_ZONE_FORMAT;
9179
9933
  exports.TabsContainer = TabsContainer;
9180
9934
  exports.TextTrim = TextTrim;
9181
9935
  exports.TitleContainer = TitleContainer;
9936
+ exports.TopContainer = TopContainer;
9937
+ exports.TopContainerButtons = TopContainerButtons;
9182
9938
  exports.TwoColumnContainer = TwoColumnContainer;
9183
9939
  exports.TwoColumnsInnerContainer = TwoColumnsInnerContainer;
9184
9940
  exports.addDataSource = addDataSource;
@@ -9216,6 +9972,7 @@ exports.getChartFilterName = getChartFilterName;
9216
9972
  exports.getChartMarkers = getChartMarkers;
9217
9973
  exports.getConfigFilter = getConfigFilter;
9218
9974
  exports.getContainerComponent = getContainerComponent;
9975
+ exports.getDashboardHeader = getDashboardHeader;
9219
9976
  exports.getDataFromAttributes = getDataFromAttributes;
9220
9977
  exports.getDataFromRelatedFeatures = getDataFromRelatedFeatures;
9221
9978
  exports.getDataSource = getDataSource;
@@ -9224,6 +9981,7 @@ exports.getDate = getDate;
9224
9981
  exports.getDefaultConfig = getDefaultConfig;
9225
9982
  exports.getElementValue = getElementValue;
9226
9983
  exports.getFeatureAttributes = getFeatureAttributes;
9984
+ exports.getFeatureCardHeader = getFeatureCardHeader;
9227
9985
  exports.getFilterComponent = getFilterComponent;
9228
9986
  exports.getFilterSelectedItems = getFilterSelectedItems;
9229
9987
  exports.getFilterValue = getFilterValue;
@@ -9275,6 +10033,7 @@ exports.useExpandableContainers = useExpandableContainers;
9275
10033
  exports.useFeatureFilters = useFeatureFilters;
9276
10034
  exports.useGetConfigLayer = useGetConfigLayer;
9277
10035
  exports.useGlobalContext = useGlobalContext;
10036
+ exports.useHeaderRender = useHeaderRender;
9278
10037
  exports.useLayerParams = useLayerParams;
9279
10038
  exports.useMapContext = useMapContext;
9280
10039
  exports.useMapDraw = useMapDraw;