@c8y/ngx-components 1021.51.1 → 1021.54.2

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 (252) hide show
  1. package/alarm-event-selector/alarm-event-attributes-form/alarm-event-attributes-form.component.d.ts +10 -2
  2. package/alarm-event-selector/alarm-event-attributes-form/alarm-event-attributes-form.component.d.ts.map +1 -1
  3. package/alarm-event-selector/alarm-event-attributes-form/alarm-event-attributes-form.model.d.ts +11 -0
  4. package/alarm-event-selector/alarm-event-attributes-form/alarm-event-attributes-form.model.d.ts.map +1 -0
  5. package/alarm-event-selector/alarm-event-selection-list/alarm-event-selection-list.component.d.ts +10 -4
  6. package/alarm-event-selector/alarm-event-selection-list/alarm-event-selection-list.component.d.ts.map +1 -1
  7. package/alarm-event-selector/alarm-event-selector-list-item/alarm-event-selector-list-item.component.d.ts +3 -1
  8. package/alarm-event-selector/alarm-event-selector-list-item/alarm-event-selector-list-item.component.d.ts.map +1 -1
  9. package/alarm-event-selector/alarm-event-selector.model.d.ts +7 -0
  10. package/alarm-event-selector/alarm-event-selector.model.d.ts.map +1 -1
  11. package/core/dashboard/dashboard-child-action.component.d.ts +1 -1
  12. package/core/dashboard/dashboard-child-action.component.d.ts.map +1 -1
  13. package/core/dashboard/dashboard.module.d.ts +29 -29
  14. package/core/dashboard/index.d.ts +3 -0
  15. package/core/dashboard/index.d.ts.map +1 -1
  16. package/core/dashboard/widgets-dashboard.component.d.ts +1 -1
  17. package/core/dashboard/widgets-dashboard.component.d.ts.map +1 -1
  18. package/core/dashboard/wiget-time-context/aggregation-picker/aggregation-picker.component.d.ts +1 -1
  19. package/core/dashboard/wiget-time-context/aggregation-picker/aggregation-picker.component.d.ts.map +1 -1
  20. package/core/dashboard/wiget-time-context/realtime-control/realtime-control.component.d.ts +1 -1
  21. package/core/dashboard/wiget-time-context/realtime-control/realtime-control.component.d.ts.map +1 -1
  22. package/core/dashboard/wiget-time-context/widget-time-context-date-range.service.d.ts +10 -0
  23. package/core/dashboard/wiget-time-context/widget-time-context-date-range.service.d.ts.map +1 -0
  24. package/core/dashboard/wiget-time-context/widget-time-context.component.d.ts +7 -4
  25. package/core/dashboard/wiget-time-context/widget-time-context.component.d.ts.map +1 -1
  26. package/core/dashboard/wiget-time-context/widget-time-context.model.d.ts +0 -1
  27. package/core/dashboard/wiget-time-context/widget-time-context.model.d.ts.map +1 -1
  28. package/core/date-picker/date-picker.component.d.ts +1 -1
  29. package/core/date-picker/date-picker.component.d.ts.map +1 -1
  30. package/core/date-picker/date-picker.module.d.ts +9 -9
  31. package/core/date-time-picker/close-date-picker.directive.d.ts +1 -1
  32. package/core/date-time-picker/close-date-picker.directive.d.ts.map +1 -1
  33. package/core/date-time-picker/date-time-picker.component.d.ts +6 -3
  34. package/core/date-time-picker/date-time-picker.component.d.ts.map +1 -1
  35. package/core/date-time-picker/date-time-picker.module.d.ts +11 -11
  36. package/core/date-time-picker/date-time-picker.module.d.ts.map +1 -1
  37. package/core/forms/forms.module.d.ts +18 -18
  38. package/core/forms/required-input-placeholder.directive.d.ts +1 -1
  39. package/core/forms/required-input-placeholder.directive.d.ts.map +1 -1
  40. package/core/router/router.service.d.ts.map +1 -1
  41. package/core/user/user-menu.service.d.ts +36 -11
  42. package/core/user/user-menu.service.d.ts.map +1 -1
  43. package/datapoint-explorer/c8y-ngx-components-datapoint-explorer.d.ts.map +1 -0
  44. package/datapoint-explorer/datapoint-explorer.module.d.ts +7 -0
  45. package/datapoint-explorer/datapoint-explorer.module.d.ts.map +1 -0
  46. package/datapoint-explorer/index.d.ts +2 -0
  47. package/datapoint-explorer/index.d.ts.map +1 -0
  48. package/datapoint-explorer/view/c8y-ngx-components-datapoint-explorer-view.d.ts.map +1 -0
  49. package/datapoint-explorer/view/configuration/naming-dictionary.d.ts +3 -0
  50. package/datapoint-explorer/view/configuration/naming-dictionary.d.ts.map +1 -0
  51. package/datapoint-explorer/view/configuration/workspace-configuration.component.d.ts +38 -0
  52. package/datapoint-explorer/view/configuration/workspace-configuration.component.d.ts.map +1 -0
  53. package/datapoint-explorer/view/configuration/workspace-configuration.model.d.ts +7 -0
  54. package/datapoint-explorer/view/configuration/workspace-configuration.model.d.ts.map +1 -0
  55. package/datapoint-explorer/view/configuration/workspace-configuration.service.d.ts +15 -0
  56. package/datapoint-explorer/view/configuration/workspace-configuration.service.d.ts.map +1 -0
  57. package/datapoint-explorer/view/create-new-report-modal/create-new-report-modal.component.d.ts +25 -0
  58. package/datapoint-explorer/view/create-new-report-modal/create-new-report-modal.component.d.ts.map +1 -0
  59. package/datapoint-explorer/view/datapoint-explorer.component.d.ts +67 -0
  60. package/datapoint-explorer/view/datapoint-explorer.component.d.ts.map +1 -0
  61. package/datapoint-explorer/view/datapoint-explorer.service.d.ts +9 -0
  62. package/datapoint-explorer/view/datapoint-explorer.service.d.ts.map +1 -0
  63. package/datapoint-explorer/view/index.d.ts +2 -0
  64. package/datapoint-explorer/view/index.d.ts.map +1 -0
  65. package/datapoint-explorer/view/send-as-widget-to-report-modal/send-as-widget-to-report-modal.component.d.ts +23 -0
  66. package/datapoint-explorer/view/send-as-widget-to-report-modal/send-as-widget-to-report-modal.component.d.ts.map +1 -0
  67. package/echart/c8y-ngx-components-echart.d.ts.map +1 -0
  68. package/echart/chart-alerts/chart-alerts.component.d.ts +8 -0
  69. package/echart/chart-alerts/chart-alerts.component.d.ts.map +1 -0
  70. package/echart/charts.component.d.ts +63 -0
  71. package/echart/charts.component.d.ts.map +1 -0
  72. package/echart/index.d.ts +5 -0
  73. package/echart/index.d.ts.map +1 -0
  74. package/echart/models/c8y-ngx-components-echart-models.d.ts.map +1 -0
  75. package/echart/models/chart.model.d.ts +37 -0
  76. package/echart/models/chart.model.d.ts.map +1 -0
  77. package/echart/models/datapoints-graph-widget.model.d.ts +126 -0
  78. package/echart/models/datapoints-graph-widget.model.d.ts.map +1 -0
  79. package/echart/models/index.d.ts +4 -0
  80. package/echart/models/index.d.ts.map +1 -0
  81. package/echart/models/svg-icons.model.d.ts +22 -0
  82. package/echart/models/svg-icons.model.d.ts.map +1 -0
  83. package/echart/services/chart-alarms.service.d.ts +17 -0
  84. package/echart/services/chart-alarms.service.d.ts.map +1 -0
  85. package/echart/services/chart-events.service.d.ts +17 -0
  86. package/echart/services/chart-events.service.d.ts.map +1 -0
  87. package/echart/services/chart-realtime.service.d.ts +35 -0
  88. package/echart/services/chart-realtime.service.d.ts.map +1 -0
  89. package/echart/services/chart-types.service.d.ts +15 -0
  90. package/echart/services/chart-types.service.d.ts.map +1 -0
  91. package/echart/services/custom-measurements.service.d.ts +12 -0
  92. package/echart/services/custom-measurements.service.d.ts.map +1 -0
  93. package/echart/services/echarts-options.service.d.ts +143 -0
  94. package/echart/services/echarts-options.service.d.ts.map +1 -0
  95. package/echart/services/y-axis.service.d.ts +17 -0
  96. package/echart/services/y-axis.service.d.ts.map +1 -0
  97. package/esm2022/alarm-event-selector/alarm-event-attributes-form/alarm-event-attributes-form.component.mjs +37 -12
  98. package/esm2022/alarm-event-selector/alarm-event-attributes-form/alarm-event-attributes-form.model.mjs +2 -0
  99. package/esm2022/alarm-event-selector/alarm-event-selection-list/alarm-event-selection-list.component.mjs +39 -12
  100. package/esm2022/alarm-event-selector/alarm-event-selector-list-item/alarm-event-selector-list-item.component.mjs +6 -4
  101. package/esm2022/alarm-event-selector/alarm-event-selector.component.mjs +1 -1
  102. package/esm2022/alarm-event-selector/alarm-event-selector.model.mjs +1 -1
  103. package/esm2022/alarm-event-selector/custom-alarm-event-form/custom-alarm-event-form.component.mjs +1 -1
  104. package/esm2022/alarms/alarms-filter.component.mjs +3 -3
  105. package/esm2022/core/dashboard/dashboard-child-action.component.mjs +3 -3
  106. package/esm2022/core/dashboard/dashboard.module.mjs +17 -14
  107. package/esm2022/core/dashboard/index.mjs +4 -1
  108. package/esm2022/core/dashboard/widgets-dashboard.component.mjs +9 -9
  109. package/esm2022/core/dashboard/wiget-time-context/aggregation-picker/aggregation-picker.component.mjs +11 -9
  110. package/esm2022/core/dashboard/wiget-time-context/realtime-control/realtime-control.component.mjs +8 -7
  111. package/esm2022/core/dashboard/wiget-time-context/widget-time-context-date-range.service.mjs +23 -0
  112. package/esm2022/core/dashboard/wiget-time-context/widget-time-context.component.mjs +70 -26
  113. package/esm2022/core/dashboard/wiget-time-context/widget-time-context.model.mjs +1 -1
  114. package/esm2022/core/date-picker/date-picker.component.mjs +29 -14
  115. package/esm2022/core/date-picker/date-picker.module.mjs +9 -7
  116. package/esm2022/core/date-time-picker/close-date-picker.directive.mjs +4 -3
  117. package/esm2022/core/date-time-picker/date-time-picker.component.mjs +57 -29
  118. package/esm2022/core/date-time-picker/date-time-picker.module.mjs +11 -7
  119. package/esm2022/core/forms/forms.module.mjs +3 -3
  120. package/esm2022/core/forms/ip-range-input-list.component.mjs +5 -5
  121. package/esm2022/core/forms/required-input-placeholder.directive.mjs +4 -3
  122. package/esm2022/core/router/router.service.mjs +1 -1
  123. package/esm2022/core/user/user-menu.service.mjs +57 -17
  124. package/esm2022/datapoint-explorer/c8y-ngx-components-datapoint-explorer.mjs +5 -0
  125. package/esm2022/datapoint-explorer/datapoint-explorer.module.mjs +65 -0
  126. package/esm2022/datapoint-explorer/index.mjs +2 -0
  127. package/esm2022/datapoint-explorer/view/c8y-ngx-components-datapoint-explorer-view.mjs +5 -0
  128. package/esm2022/datapoint-explorer/view/configuration/naming-dictionary.mjs +65 -0
  129. package/esm2022/datapoint-explorer/view/configuration/workspace-configuration.component.mjs +169 -0
  130. package/esm2022/datapoint-explorer/view/configuration/workspace-configuration.model.mjs +2 -0
  131. package/esm2022/datapoint-explorer/view/configuration/workspace-configuration.service.mjs +40 -0
  132. package/esm2022/datapoint-explorer/view/create-new-report-modal/create-new-report-modal.component.mjs +64 -0
  133. package/esm2022/datapoint-explorer/view/datapoint-explorer.component.mjs +279 -0
  134. package/esm2022/datapoint-explorer/view/datapoint-explorer.service.mjs +23 -0
  135. package/esm2022/datapoint-explorer/view/index.mjs +2 -0
  136. package/esm2022/datapoint-explorer/view/send-as-widget-to-report-modal/send-as-widget-to-report-modal.component.mjs +48 -0
  137. package/esm2022/datapoint-selector/datapoint-attributes-form/datapoint-attributes-form.component.mjs +3 -3
  138. package/esm2022/datapoint-selector/datapoint-selector-list-item/datapoint-selector-list-item.component.mjs +3 -3
  139. package/esm2022/datapoints-export-selector/datapoints-export-selector-modal/datapoints-export-selector-file-exporter/datapoints-exports-selector-file-types/datapoints-exports-selector-file-types.component.mjs +3 -3
  140. package/esm2022/datapoints-export-selector/datapoints-export-selector.component.mjs +3 -3
  141. package/esm2022/device-list/add-smart-group.component.mjs +3 -3
  142. package/esm2022/echart/c8y-ngx-components-echart.mjs +5 -0
  143. package/esm2022/echart/chart-alerts/chart-alerts.component.mjs +16 -0
  144. package/esm2022/echart/charts.component.mjs +560 -0
  145. package/esm2022/echart/index.mjs +5 -0
  146. package/esm2022/echart/models/c8y-ngx-components-echart-models.mjs +5 -0
  147. package/esm2022/echart/models/chart.model.mjs +2 -0
  148. package/esm2022/echart/models/datapoints-graph-widget.model.mjs +17 -0
  149. package/esm2022/echart/models/index.mjs +4 -0
  150. package/esm2022/echart/models/svg-icons.model.mjs +22 -0
  151. package/esm2022/echart/services/chart-alarms.service.mjs +58 -0
  152. package/esm2022/echart/services/chart-events.service.mjs +44 -0
  153. package/esm2022/echart/services/chart-realtime.service.mjs +193 -0
  154. package/esm2022/echart/services/chart-types.service.mjs +102 -0
  155. package/esm2022/echart/services/custom-measurements.service.mjs +52 -0
  156. package/esm2022/echart/services/echarts-options.service.mjs +857 -0
  157. package/esm2022/echart/services/y-axis.service.mjs +150 -0
  158. package/esm2022/icon-selector/icons/platform/index.mjs +10 -8
  159. package/esm2022/interval-picker/interval-picker.component.mjs +3 -3
  160. package/esm2022/interval-picker/interval-picker.model.mjs +1 -1
  161. package/esm2022/report-dashboard/index.mjs +2 -1
  162. package/esm2022/time-context/c8y-ngx-components-time-context.mjs +5 -0
  163. package/esm2022/time-context/index.mjs +2 -0
  164. package/esm2022/time-context/time-context.component.mjs +217 -0
  165. package/esm2022/time-context/time-context.service.mjs +83 -0
  166. package/esm2022/translation-editor/lazy/translation-editor/translation-editor.component.mjs +6 -5
  167. package/esm2022/widgets/cockpit-exports/index.mjs +8 -1
  168. package/esm2022/widgets/definitions/datapoints-graph/c8y-ngx-components-widgets-definitions-datapoints-graph.mjs +5 -0
  169. package/esm2022/widgets/definitions/datapoints-graph/index.mjs +31 -0
  170. package/esm2022/widgets/definitions/index.mjs +3 -2
  171. package/esm2022/widgets/implementations/datapoints-graph/c8y-ngx-components-widgets-implementations-datapoints-graph.mjs +5 -0
  172. package/esm2022/widgets/implementations/datapoints-graph/datapoints-graph-config/datapoints-graph-widget-config.component.mjs +239 -0
  173. package/esm2022/widgets/implementations/datapoints-graph/datapoints-graph-view/datapoints-graph-widget-view.component.mjs +241 -0
  174. package/esm2022/widgets/implementations/datapoints-graph/index.mjs +3 -0
  175. package/fesm2022/c8y-ngx-components-alarm-event-selector.mjs +80 -28
  176. package/fesm2022/c8y-ngx-components-alarm-event-selector.mjs.map +1 -1
  177. package/fesm2022/c8y-ngx-components-alarms.mjs +2 -2
  178. package/fesm2022/c8y-ngx-components-alarms.mjs.map +1 -1
  179. package/fesm2022/c8y-ngx-components-datapoint-explorer-view.mjs +658 -0
  180. package/fesm2022/c8y-ngx-components-datapoint-explorer-view.mjs.map +1 -0
  181. package/fesm2022/c8y-ngx-components-datapoint-explorer.mjs +72 -0
  182. package/fesm2022/c8y-ngx-components-datapoint-explorer.mjs.map +1 -0
  183. package/fesm2022/c8y-ngx-components-datapoint-selector.mjs +4 -4
  184. package/fesm2022/c8y-ngx-components-datapoint-selector.mjs.map +1 -1
  185. package/fesm2022/c8y-ngx-components-datapoints-export-selector.mjs +4 -4
  186. package/fesm2022/c8y-ngx-components-datapoints-export-selector.mjs.map +1 -1
  187. package/fesm2022/c8y-ngx-components-device-list.mjs +2 -2
  188. package/fesm2022/c8y-ngx-components-device-list.mjs.map +1 -1
  189. package/fesm2022/c8y-ngx-components-echart-models.mjs +46 -0
  190. package/fesm2022/c8y-ngx-components-echart-models.mjs.map +1 -0
  191. package/fesm2022/c8y-ngx-components-echart.mjs +2023 -0
  192. package/fesm2022/c8y-ngx-components-echart.mjs.map +1 -0
  193. package/fesm2022/c8y-ngx-components-icon-selector-icons-platform.mjs +9 -7
  194. package/fesm2022/c8y-ngx-components-icon-selector-icons-platform.mjs.map +1 -1
  195. package/fesm2022/c8y-ngx-components-interval-picker.mjs +2 -2
  196. package/fesm2022/c8y-ngx-components-interval-picker.mjs.map +1 -1
  197. package/fesm2022/c8y-ngx-components-report-dashboard.mjs +1 -1
  198. package/fesm2022/c8y-ngx-components-time-context.mjs +300 -0
  199. package/fesm2022/c8y-ngx-components-time-context.mjs.map +1 -0
  200. package/fesm2022/c8y-ngx-components-translation-editor-lazy.mjs +5 -4
  201. package/fesm2022/c8y-ngx-components-translation-editor-lazy.mjs.map +1 -1
  202. package/fesm2022/c8y-ngx-components-widgets-cockpit-exports.mjs +7 -0
  203. package/fesm2022/c8y-ngx-components-widgets-cockpit-exports.mjs.map +1 -1
  204. package/fesm2022/c8y-ngx-components-widgets-definitions-datapoints-graph.mjs +38 -0
  205. package/fesm2022/c8y-ngx-components-widgets-definitions-datapoints-graph.mjs.map +1 -0
  206. package/fesm2022/c8y-ngx-components-widgets-definitions.mjs +2 -1
  207. package/fesm2022/c8y-ngx-components-widgets-definitions.mjs.map +1 -1
  208. package/fesm2022/c8y-ngx-components-widgets-implementations-datapoints-graph.mjs +467 -0
  209. package/fesm2022/c8y-ngx-components-widgets-implementations-datapoints-graph.mjs.map +1 -0
  210. package/fesm2022/c8y-ngx-components.mjs +429 -277
  211. package/fesm2022/c8y-ngx-components.mjs.map +1 -1
  212. package/icon-selector/icons/index.d.ts +1 -1
  213. package/icon-selector/icons/platform/index.d.ts +2 -2
  214. package/icon-selector/icons/platform/index.d.ts.map +1 -1
  215. package/interval-picker/interval-picker.component.d.ts.map +1 -1
  216. package/interval-picker/interval-picker.model.d.ts.map +1 -1
  217. package/locales/de.po +206 -279
  218. package/locales/es.po +206 -279
  219. package/locales/fr.po +206 -279
  220. package/locales/ja_JP.po +196 -279
  221. package/locales/ko.po +455 -444
  222. package/locales/locales.pot +190 -272
  223. package/locales/nl.po +206 -279
  224. package/locales/pl.po +206 -279
  225. package/locales/pt_BR.po +206 -279
  226. package/locales/zh_CN.po +450 -443
  227. package/locales/zh_TW.po +452 -445
  228. package/package.json +1 -1
  229. package/report-dashboard/index.d.ts +1 -0
  230. package/report-dashboard/index.d.ts.map +1 -1
  231. package/time-context/c8y-ngx-components-time-context.d.ts.map +1 -0
  232. package/time-context/index.d.ts +2 -0
  233. package/time-context/index.d.ts.map +1 -0
  234. package/time-context/time-context.component.d.ts +61 -0
  235. package/time-context/time-context.component.d.ts.map +1 -0
  236. package/time-context/time-context.service.d.ts +25 -0
  237. package/time-context/time-context.service.d.ts.map +1 -0
  238. package/translation-editor/lazy/translation-editor/translation-editor.component.d.ts.map +1 -1
  239. package/widgets/cockpit-exports/index.d.ts +6 -0
  240. package/widgets/cockpit-exports/index.d.ts.map +1 -1
  241. package/widgets/definitions/datapoints-graph/c8y-ngx-components-widgets-definitions-datapoints-graph.d.ts.map +1 -0
  242. package/widgets/definitions/datapoints-graph/index.d.ts +16 -0
  243. package/widgets/definitions/datapoints-graph/index.d.ts.map +1 -0
  244. package/widgets/definitions/index.d.ts +1 -0
  245. package/widgets/definitions/index.d.ts.map +1 -1
  246. package/widgets/implementations/datapoints-graph/c8y-ngx-components-widgets-implementations-datapoints-graph.d.ts.map +1 -0
  247. package/widgets/implementations/datapoints-graph/datapoints-graph-config/datapoints-graph-widget-config.component.d.ts +43 -0
  248. package/widgets/implementations/datapoints-graph/datapoints-graph-config/datapoints-graph-widget-config.component.d.ts.map +1 -0
  249. package/widgets/implementations/datapoints-graph/datapoints-graph-view/datapoints-graph-widget-view.component.d.ts +54 -0
  250. package/widgets/implementations/datapoints-graph/datapoints-graph-view/datapoints-graph-widget-view.component.d.ts.map +1 -0
  251. package/widgets/implementations/datapoints-graph/index.d.ts +3 -0
  252. package/widgets/implementations/datapoints-graph/index.d.ts.map +1 -0
@@ -0,0 +1,857 @@
1
+ /* eslint-disable @typescript-eslint/no-explicit-any */
2
+ import { Injectable } from '@angular/core';
3
+ import { DatePipe } from '@c8y/ngx-components';
4
+ import { YAxisService } from './y-axis.service';
5
+ import { ChartTypesService } from './chart-types.service';
6
+ import { AlarmStatus } from '@c8y/client';
7
+ import { ICONS_MAP } from '../models/svg-icons.model';
8
+ import { Router } from '@angular/router';
9
+ import { AlarmSeverityToIconPipe, AlarmSeverityToLabelPipe } from '@c8y/ngx-components/alarms';
10
+ import { INTERVALS } from '@c8y/ngx-components/interval-picker';
11
+ import * as i0 from "@angular/core";
12
+ import * as i1 from "@c8y/ngx-components";
13
+ import * as i2 from "./y-axis.service";
14
+ import * as i3 from "./chart-types.service";
15
+ import * as i4 from "@c8y/ngx-components/alarms";
16
+ import * as i5 from "@angular/router";
17
+ const INDEX_HTML = '/index.html';
18
+ export class EchartsOptionsService {
19
+ constructor(datePipe, yAxisService, chartTypesService, severityIconPipe, severityLabelPipe, router) {
20
+ this.datePipe = datePipe;
21
+ this.yAxisService = yAxisService;
22
+ this.chartTypesService = chartTypesService;
23
+ this.severityIconPipe = severityIconPipe;
24
+ this.severityLabelPipe = severityLabelPipe;
25
+ this.router = router;
26
+ this.TOOLTIP_WIDTH = 300;
27
+ }
28
+ getChartOptions(datapointsWithValues, timeRange, showSplitLines, events, alarms, displayOptions, selectedTimeRange, aggregatedDatapoints, sliderZoomUsed = false) {
29
+ const yAxis = this.yAxisService.getYAxis(datapointsWithValues, {
30
+ showSplitLines: showSplitLines.YAxis,
31
+ mergeMatchingDatapoints: displayOptions.mergeMatchingDatapoints,
32
+ showLabelAndUnit: displayOptions.showLabelAndUnit
33
+ });
34
+ const leftAxis = yAxis.filter(yx => yx.position === 'left');
35
+ const gridLeft = leftAxis.length ? leftAxis.length * this.yAxisService.Y_AXIS_OFFSET : 16;
36
+ const rightAxis = yAxis.filter(yx => yx.position === 'right');
37
+ const gridRight = rightAxis.length ? rightAxis.length * this.yAxisService.Y_AXIS_OFFSET : 16;
38
+ let intervalInMs = this.calculateExtendedIntervalInMs(selectedTimeRange?.interval || timeRange.interval || 'hours', selectedTimeRange || timeRange);
39
+ if (sliderZoomUsed) {
40
+ intervalInMs = this.calculateExtendedIntervalInMs(timeRange.interval || 'hours', timeRange);
41
+ }
42
+ return {
43
+ grid: {
44
+ containLabel: false, // axis labels are not taken into account to calculate graph grid
45
+ left: gridLeft,
46
+ top: 16,
47
+ right: gridRight,
48
+ bottom: 68
49
+ },
50
+ dataZoom: [
51
+ {
52
+ type: 'inside',
53
+ // TODO: use 'none' only when this bug is fixed https://github.com/apache/echarts/issues/17858
54
+ filterMode: datapointsWithValues.some(dp => dp.lineType === 'bars') ? 'filter' : 'none',
55
+ zoomOnMouseWheel: true,
56
+ startValue: selectedTimeRange
57
+ ? selectedTimeRange.dateFrom.valueOf()
58
+ : timeRange.dateFrom.valueOf(),
59
+ endValue: selectedTimeRange
60
+ ? selectedTimeRange.dateTo.valueOf()
61
+ : timeRange.dateTo.valueOf()
62
+ },
63
+ {
64
+ type: 'slider',
65
+ show: displayOptions.showSlider,
66
+ bottom: 8,
67
+ realtime: false
68
+ }
69
+ ], // on realtime, 'none' will cause extending chart line to left edge of the chart
70
+ animation: false,
71
+ toolbox: {
72
+ show: true,
73
+ itemSize: 0, // toolbox is needed for zooming in action, but we provide our own buttons
74
+ feature: {
75
+ dataZoom: {
76
+ yAxisIndex: 'none'
77
+ }
78
+ }
79
+ },
80
+ tooltip: {
81
+ trigger: 'axis',
82
+ axisPointer: {
83
+ type: 'cross',
84
+ snap: true
85
+ },
86
+ backgroundColor: 'rgba(255, 255, 255, 0.9)',
87
+ formatter: this.getTooltipFormatter(),
88
+ appendToBody: true,
89
+ position: this.tooltipPosition(),
90
+ transitionDuration: 0
91
+ },
92
+ legend: {
93
+ show: false
94
+ },
95
+ xAxis: {
96
+ min: new Date(timeRange.dateFrom).valueOf() - intervalInMs,
97
+ max: timeRange.dateTo,
98
+ type: 'time',
99
+ animation: false,
100
+ axisPointer: {
101
+ label: {
102
+ show: false
103
+ }
104
+ },
105
+ axisLine: {
106
+ // align X axis to 0 of Y axis of datapoint with lineType 'bars'
107
+ onZeroAxisIndex: datapointsWithValues.findIndex(dp => dp.lineType === 'bars')
108
+ },
109
+ axisLabel: {
110
+ hideOverlap: true,
111
+ borderWidth: 2, // as there is no margin for labels spacing, transparent border is a workaround
112
+ borderColor: 'transparent'
113
+ },
114
+ splitLine: {
115
+ show: showSplitLines.XAxis,
116
+ lineStyle: { opacity: 0.8, type: 'dashed', width: 2 }
117
+ }
118
+ },
119
+ yAxis,
120
+ series: [
121
+ ...this.getAggregatedSeries(aggregatedDatapoints || []).filter(series => series !== undefined),
122
+ ...this.getChartSeries(datapointsWithValues, events, alarms, displayOptions)
123
+ ]
124
+ };
125
+ }
126
+ calculateExtendedIntervalInMs(interval, selectedTimeRange) {
127
+ let intervalInMs = INTERVALS.find(i => i.id === interval).timespanInMs;
128
+ switch (interval) {
129
+ case 'minutes':
130
+ intervalInMs = INTERVALS.find(i => i.id === interval).timespanInMs * 60;
131
+ break;
132
+ case 'hours':
133
+ intervalInMs = INTERVALS.find(i => i.id === interval).timespanInMs * 24;
134
+ break;
135
+ case 'days':
136
+ intervalInMs = INTERVALS.find(i => i.id === interval).timespanInMs * 28;
137
+ break;
138
+ case 'weeks':
139
+ intervalInMs = INTERVALS.find(i => i.id === interval).timespanInMs * 12;
140
+ break;
141
+ case 'months':
142
+ intervalInMs = INTERVALS.find(i => i.id === interval).timespanInMs * 12;
143
+ break;
144
+ case 'custom':
145
+ intervalInMs =
146
+ (new Date(selectedTimeRange.dateTo).valueOf() -
147
+ new Date(selectedTimeRange.dateFrom).valueOf()) *
148
+ 12;
149
+ break;
150
+ default:
151
+ intervalInMs = INTERVALS.find(i => i.id === interval).timespanInMs;
152
+ break;
153
+ }
154
+ return intervalInMs;
155
+ }
156
+ getAggregatedSeries(aggregatedDatapoints) {
157
+ const series = [];
158
+ aggregatedDatapoints.forEach((dp, idx) => {
159
+ const renderType = dp.renderType || 'min';
160
+ if (renderType === 'area') {
161
+ series.push(this.getSingleSeries(dp, 'min', idx, true, 'aggr'));
162
+ series.push(this.getSingleSeries(dp, 'max', idx, true, 'aggr'));
163
+ }
164
+ else {
165
+ series.push(this.getSingleSeries(dp, renderType, idx, false, 'aggr'));
166
+ }
167
+ });
168
+ series.forEach((s) => {
169
+ s.datapointId = 'aggregated';
170
+ s.typeOfSeries = 'fake';
171
+ s.itemStyle = {
172
+ ...s.itemStyle,
173
+ opacity: 0
174
+ };
175
+ s.lineStyle = {
176
+ ...s.lineStyle,
177
+ opacity: 0
178
+ };
179
+ });
180
+ return [series[0]];
181
+ }
182
+ /**
183
+ * This method is used to get the series for alarms and events.
184
+ * @param dp - The data point.
185
+ * @param renderType - The render type.
186
+ * @param isMinMaxChart - If the chart is min max chart.
187
+ * @param items - All alarms or events which should be displayed on the chart.
188
+ * @param itemType - The item type.
189
+ * @param id - The id of the device
190
+ */
191
+ getAlarmOrEventSeries(dp, renderType, isMinMaxChart = false, items = [], itemType = 'alarm', displayOptions = { displayMarkedLine: true, displayMarkedPoint: true }, id, idx, realtime) {
192
+ if (!items.length) {
193
+ return [];
194
+ }
195
+ if (!displayOptions.displayMarkedLine && !displayOptions.displayMarkedPoint) {
196
+ return [];
197
+ }
198
+ //filter items that are not __hidden
199
+ const filteredItems = items.filter(item => !item['__hidden']);
200
+ const itemsByType = this.groupByType(filteredItems, 'type');
201
+ const isAlarm = itemType === 'alarm';
202
+ return Object.entries(itemsByType).flatMap(([type, itemsOfType]) => {
203
+ // Main series data
204
+ const mainData = itemsOfType.map(item => [item.creationTime, null, 'markLineFlag']);
205
+ // Is a specific datapoint template selected for this alarm/event type?
206
+ let isDpTemplateSelected = false;
207
+ // MarkPoint data
208
+ const markPointData = itemsOfType.reduce((acc, item) => {
209
+ isDpTemplateSelected =
210
+ item['selectedDatapoint'] &&
211
+ dp['__target'] &&
212
+ item['selectedDatapoint']['fragment'] === dp['fragment'] &&
213
+ item['selectedDatapoint']['series'] === dp['series'] &&
214
+ item['selectedDatapoint']['target'] === dp['__target']['id'];
215
+ if (dp.__target?.id === item.source.id) {
216
+ const isCleared = isAlarm && item.status === AlarmStatus.CLEARED;
217
+ const isEvent = !isAlarm;
218
+ return acc.concat(this.createMarkPoint(item, dp, isCleared, isEvent, realtime));
219
+ }
220
+ else {
221
+ if (!item.creationTime) {
222
+ return [];
223
+ }
224
+ return acc.concat([
225
+ {
226
+ coord: [item.creationTime, null],
227
+ name: item.type,
228
+ itemType: item.type,
229
+ itemStyle: { color: item['color'] }
230
+ }
231
+ ]);
232
+ }
233
+ }, []);
234
+ // Construct series with markPoint
235
+ const seriesWithMarkPoint = {
236
+ id: `${type}/${dp.__target?.id}+${id ? id : ''}-markPoint`,
237
+ name: `${type}-markPoint`,
238
+ typeOfSeries: itemType,
239
+ data: mainData,
240
+ isDpTemplateSelected,
241
+ position: 'bottom',
242
+ silent: true,
243
+ markPoint: {
244
+ showSymbol: true,
245
+ symbolKeepAspect: true,
246
+ data: markPointData
247
+ },
248
+ yAxisIndex: idx,
249
+ ...this.chartTypesService.getSeriesOptions(dp, isMinMaxChart, renderType)
250
+ };
251
+ const markLineData = this.createMarkLine(itemsOfType);
252
+ // Construct series with markLine
253
+ const seriesWithMarkLine = {
254
+ id: `${type}/${dp.__target?.id}+${id ? id : ''}-markLine`,
255
+ name: `${type}-markLine`,
256
+ typeOfSeries: itemType,
257
+ isDpTemplateSelected,
258
+ data: mainData,
259
+ markLine: {
260
+ showSymbol: false,
261
+ symbol: ['none', 'none'], // no symbol at the start/end of the line
262
+ data: markLineData
263
+ },
264
+ ...this.chartTypesService.getSeriesOptions(dp, isMinMaxChart, renderType)
265
+ };
266
+ //depending on the options return only the required series
267
+ if (displayOptions.displayMarkedLine && displayOptions.displayMarkedPoint) {
268
+ return [seriesWithMarkLine, seriesWithMarkPoint];
269
+ }
270
+ else if (displayOptions.displayMarkedLine) {
271
+ return [seriesWithMarkLine];
272
+ }
273
+ else if (displayOptions.displayMarkedPoint) {
274
+ return [seriesWithMarkPoint];
275
+ }
276
+ else {
277
+ return null;
278
+ }
279
+ });
280
+ }
281
+ /**
282
+ * This method is used to get tooltip formatter for alarms and events.
283
+ * @param tooltipParams - The tooltip parameters.
284
+ * @param params - The parameters data.
285
+ * @param allEvents - All events.
286
+ * @param allAlarms - All alarms.
287
+ * @returns The formatted string for the tooltip.
288
+ */
289
+ getTooltipFormatterForAlarmAndEvents(tooltipParams, params, allEvents, allAlarms) {
290
+ if (!Array.isArray(tooltipParams)) {
291
+ return '';
292
+ }
293
+ const XAxisValue = tooltipParams[0].data[0];
294
+ const YAxisReadings = [];
295
+ const allSeries = this.echartsInstance?.getOption()['series'];
296
+ // filter out alarm and event series
297
+ const allDataPointSeries = allSeries.filter(series => series['typeOfSeries'] !== 'alarm' && series['typeOfSeries'] !== 'event');
298
+ this.processSeries(allDataPointSeries, XAxisValue, YAxisReadings);
299
+ // find event and alarm of the same type as the hovered markedLine or markedPoint
300
+ const event = allEvents.find(e => e.type === params.data.itemType);
301
+ const alarm = allAlarms.find(a => a.type === params.data.itemType);
302
+ let value = '';
303
+ if (event) {
304
+ value = this.processEvent(event);
305
+ }
306
+ if (alarm) {
307
+ this.processAlarm(alarm).then(alarmVal => {
308
+ value = alarmVal;
309
+ YAxisReadings.push(value);
310
+ const options = this.echartsInstance?.getOption();
311
+ if (!options.tooltip || !Array.isArray(options.tooltip)) {
312
+ return;
313
+ }
314
+ const updatedOptions = {
315
+ tooltip: options['tooltip'][0]
316
+ };
317
+ if (!updatedOptions.tooltip) {
318
+ return;
319
+ }
320
+ updatedOptions.tooltip.formatter = `<div style="width: ${this.TOOLTIP_WIDTH}px">${YAxisReadings.join('')}</div>`;
321
+ updatedOptions.tooltip.transitionDuration = 0;
322
+ updatedOptions.tooltip.position = this.tooltipPosition();
323
+ this.echartsInstance?.setOption(updatedOptions);
324
+ return;
325
+ });
326
+ }
327
+ YAxisReadings.push(value);
328
+ return `<div style="width: 300px">${YAxisReadings.join('')}</div>`;
329
+ }
330
+ tooltipPosition() {
331
+ let lastPositionOfTooltip = {};
332
+ if (this.tooltipPositionCallback) {
333
+ return this.tooltipPositionCallback;
334
+ }
335
+ this.tooltipPositionCallback = (point, // position of mouse in chart [X, Y]; 0,0 is top left corner
336
+ _, // tooltip data
337
+ dom, // tooltip element
338
+ __, size // size of chart
339
+ ) => {
340
+ const offset = 10;
341
+ const [mouseX, mouseY] = point;
342
+ const chartWidth = size?.viewSize[0] || 0;
343
+ const chartHeight = size?.viewSize[1] || 0;
344
+ const tooltipWidth = size?.contentSize[0] || 0;
345
+ const tooltipHeight = size?.contentSize[1] || 0;
346
+ const tooltipRect = dom?.getBoundingClientRect();
347
+ const tooltipOverflowsBottomEdge = tooltipRect.bottom > window.innerHeight;
348
+ const tooltipOverflowsRightEdge = tooltipRect.right > window.innerWidth;
349
+ const tooltipWouldOverflowBottomEdgeOnPositionChange = !lastPositionOfTooltip.top &&
350
+ tooltipRect.bottom + 2 * offset + tooltipHeight > window.innerHeight;
351
+ const tooltipWouldOverflowRightEdgeOnPositionChange = !lastPositionOfTooltip.left &&
352
+ tooltipRect.right + 2 * offset + tooltipWidth > window.innerWidth;
353
+ let verticalPosition = {
354
+ top: mouseY + offset
355
+ };
356
+ let horizontalPosition = {
357
+ left: mouseX + offset
358
+ };
359
+ if (tooltipOverflowsBottomEdge || tooltipWouldOverflowBottomEdgeOnPositionChange) {
360
+ verticalPosition = {
361
+ bottom: chartHeight - mouseY + offset
362
+ };
363
+ }
364
+ if (tooltipOverflowsRightEdge || tooltipWouldOverflowRightEdgeOnPositionChange) {
365
+ horizontalPosition = {
366
+ right: chartWidth - mouseX + offset
367
+ };
368
+ }
369
+ lastPositionOfTooltip = {
370
+ ...verticalPosition,
371
+ ...horizontalPosition
372
+ };
373
+ return lastPositionOfTooltip;
374
+ };
375
+ return this.tooltipPositionCallback;
376
+ }
377
+ /**
378
+ * This method is used to add the data point info to the tooltip.
379
+ * @param allDataPointSeries - All the data point series.
380
+ * @param XAxisValue - The X Axis value.
381
+ * @param YAxisReadings - The Y Axis readings.
382
+ */
383
+ processSeries(allDataPointSeries, XAxisValue, YAxisReadings) {
384
+ allDataPointSeries.forEach((series) => {
385
+ let value = '';
386
+ if (series.id.endsWith('/min')) {
387
+ value = this.processMinSeries(series, allDataPointSeries, XAxisValue);
388
+ }
389
+ else if (!series.id.endsWith('/max')) {
390
+ value = this.processRegularSeries(series, XAxisValue);
391
+ }
392
+ if (value) {
393
+ YAxisReadings.push(`<div class="d-flex a-i-center p-b-8"><span class='dlt-c8y-icon-circle m-r-4' style='color: ${series.itemStyle.color};'></span>` + // color circle
394
+ `<strong>${series.datapointLabel}</strong></div>` + // name
395
+ `${value}` // single value or min-max range
396
+ );
397
+ }
398
+ });
399
+ }
400
+ /**
401
+ * This method is used to process the min series.
402
+ * @param series - The series.
403
+ * @param allDataPointSeries - All the data point series.
404
+ * @param XAxisValue - The X Axis value.
405
+ * @returns The processed value.
406
+ */
407
+ processMinSeries(series, allDataPointSeries, XAxisValue) {
408
+ const minValue = this.findValueForExactOrEarlierTimestamp(series.data, XAxisValue);
409
+ if (!minValue) {
410
+ return '';
411
+ }
412
+ const maxSeries = allDataPointSeries.find(s => s['id'] === series.id.replace('/min', '/max'));
413
+ const maxValue = this.findValueForExactOrEarlierTimestamp(maxSeries?.['data'], XAxisValue);
414
+ return (`<div class="d-flex a-i-center separator-top p-t-8 p-b-8"><label class="text-12 m-r-8 m-b-0">${this.datePipe.transform(minValue[0])}</label>` +
415
+ `<span class="m-l-auto text-12">${minValue[1]} — ${maxValue?.[1]}` +
416
+ (series.datapointUnit ? ` ${series.datapointUnit}` : '') +
417
+ `</span></div>`);
418
+ }
419
+ /**
420
+ * This method is used to process the regular series.
421
+ * @param series - The series.
422
+ * @param XAxisValue - The X Axis value.
423
+ * @returns The processed value.
424
+ */
425
+ processRegularSeries(series, XAxisValue) {
426
+ const seriesValue = this.findValueForExactOrEarlierTimestamp(series.data, XAxisValue);
427
+ if (!seriesValue) {
428
+ return '';
429
+ }
430
+ return (`<div class="d-flex a-i-center p-t-8 p-b-8 separator-top">` +
431
+ `<label class="m-b-0 m-r-8 text-12">${this.datePipe.transform(seriesValue[0])}</label><span class="m-l-auto text-12">` +
432
+ seriesValue[1]?.toString() +
433
+ (series.datapointUnit ? ` ${series.datapointUnit}` : '') +
434
+ `</span></div>`);
435
+ }
436
+ /**
437
+ * This method is used to process the event tooltip.
438
+ * @param event - The event object.
439
+ * @returns The processed value.
440
+ */
441
+ processEvent(event) {
442
+ let value = `<ul class="list-unstyled small separator-top">`;
443
+ value += `<li class="p-t-4 p-b-4 d-flex separator-bottom text-no-wrap"><label class="small m-b-0 m-r-8">Event type</label><code class="m-l-auto">${event.type}</code></li>`;
444
+ value += `<li class="p-t-4 p-b-4 d-flex separator-bottom text-no-wrap"><label class="small m-b-0 m-r-8">Event text</label><span class="m-l-auto">${event.text}<span></li>`;
445
+ value += `<li class="p-t-4 p-b-4 d-flex separator-bottom text-no-wrap"><label class="small m-b-0 m-r-8">Last update</label><span class="m-l-auto">${this.datePipe.transform(event['lastUpdated'])}<span></li>`;
446
+ value += `</ul>`;
447
+ return value;
448
+ }
449
+ /**
450
+ * This method is used to process the alarm tooltip.
451
+ * @param alarm - The alarm object.
452
+ * @returns The processed value.
453
+ */
454
+ async processAlarm(alarm) {
455
+ let value = `<ul class="list-unstyled small separator-top m-0">`;
456
+ value += `<li class="p-t-4 p-b-4 d-flex a-i-center separator-bottom text-no-wrap"><label class="text-label-small m-b-0 m-r-8">Alarm Severity</label>`;
457
+ value += `<span class="small d-inline-flex a-i-center gap-4 m-l-auto"><i class="stroked-icon icon-14 status dlt-c8y-icon-${this.severityIconPipe.transform(alarm.severity)} ${alarm.severity.toLowerCase()}" > </i> ${this.severityLabelPipe.transform(alarm.severity)} </span></li>`;
458
+ value += `<li class="p-t-4 p-b-4 d-flex separator-bottom text-no-wrap"><label class="text-label-small m-b-0 m-r-8">Alarm Type</label><span class="small m-l-auto"><code>${alarm.type}</code></span></li>`;
459
+ value += `<li class="p-t-4 p-b-4 d-flex separator-bottom text-no-wrap"><label class="text-label-small m-b-0 m-r-8">Message</label><span class="small m-l-auto" style="overflow: hidden; text-overflow: ellipsis;" title="${alarm.text}">${alarm.text}</span></li>`;
460
+ value += `<li class="p-t-4 p-b-4 d-flex separator-bottom text-no-wrap"><label class="text-label-small m-b-0 m-r-8">Last Updated</label><span class="small m-l-auto">${this.datePipe.transform(alarm['lastUpdated'])}</span></li>`;
461
+ const exists = await this.alarmRouteExists();
462
+ if (exists) {
463
+ const currentUrl = window.location.href;
464
+ const baseUrlIndex = currentUrl.indexOf(INDEX_HTML);
465
+ const baseUrl = currentUrl.substring(0, baseUrlIndex + INDEX_HTML.length);
466
+ value += `<li class="p-t-4 p-b-4 d-flex separator-bottom text-no-wrap"><label class="text-label-small m-b-0 m-r-8">Link</label><span class="small m-l-auto"><a href="${baseUrl}#/alarms/${alarm.id}">Alarm Details</a></span></li>`;
467
+ }
468
+ value += `<li class="p-t-4 p-b-4 d-flex text-no-wrap"><label class="text-label-small m-b-0 m-r-8">Alarm count</label><span class="small m-l-auto"><span class="badge badge-info">${alarm.count}</span></span></li>`;
469
+ value += `</ul>`;
470
+ return value;
471
+ }
472
+ async alarmRouteExists() {
473
+ const exists = this.router.config.some(route => {
474
+ return `${route.path}` === 'alarms';
475
+ });
476
+ return exists;
477
+ }
478
+ getChartSeries(datapointsWithValues, events, alarms, displayOptions) {
479
+ const series = [];
480
+ let eventSeries = [];
481
+ let alarmSeries = [];
482
+ datapointsWithValues.forEach((dp, idx) => {
483
+ const renderType = dp.renderType || 'min';
484
+ if (renderType === 'area') {
485
+ series.push(this.getSingleSeries(dp, 'min', idx, true));
486
+ series.push(this.getSingleSeries(dp, 'max', idx, true));
487
+ }
488
+ else {
489
+ series.push(this.getSingleSeries(dp, renderType, idx, false));
490
+ }
491
+ const newEventSeries = this.getAlarmOrEventSeries(dp, renderType, false, events, 'event', displayOptions, null, idx);
492
+ const newAlarmSeries = this.getAlarmOrEventSeries(dp, renderType, false, alarms, 'alarm', displayOptions, null, idx);
493
+ eventSeries = [...eventSeries, ...newEventSeries];
494
+ alarmSeries = [...alarmSeries, ...newAlarmSeries];
495
+ });
496
+ const deduplicateFilterCallback = (obj1, i, arr) => {
497
+ const duplicates = arr.filter(obj2 => obj1['id'] === obj2['id'] && i !== arr.indexOf(obj2));
498
+ if (duplicates.length > 0) {
499
+ return obj1['isDpTemplateSelected'];
500
+ }
501
+ return true;
502
+ };
503
+ const deduplicateFilterCallbackFallback = (obj1, i, arr) => arr.findIndex(obj2 => obj2['id'] === obj1['id']) === i;
504
+ let deduplicatedEvents = eventSeries.filter(deduplicateFilterCallback);
505
+ let deduplicatedAlarms = alarmSeries.filter(deduplicateFilterCallback);
506
+ if (deduplicatedAlarms.length === 0) {
507
+ deduplicatedAlarms = alarmSeries.filter(deduplicateFilterCallbackFallback);
508
+ }
509
+ if (deduplicatedEvents.length === 0) {
510
+ deduplicatedEvents = eventSeries.filter(deduplicateFilterCallbackFallback);
511
+ }
512
+ return [...series, ...deduplicatedEvents, ...deduplicatedAlarms];
513
+ }
514
+ groupByType(items, typeField) {
515
+ return items.reduce((grouped, item) => {
516
+ (grouped[item[typeField]] = grouped[item[typeField]] || []).push(item);
517
+ return grouped;
518
+ }, {});
519
+ }
520
+ /**
521
+ * This method interpolates between two data points. The goal is to place the markPoint on the chart in the right place.
522
+ * @param dpValuesArray array of data points
523
+ * @param targetTime time of the alarm or event
524
+ * @returns interpolated data point
525
+ */
526
+ interpolateBetweenTwoDps(dpValuesArray, targetTime) {
527
+ let maxValue;
528
+ let minValue;
529
+ return dpValuesArray.reduce((acc, curr, idx, arr) => {
530
+ if (new Date(curr.time).getTime() <= targetTime) {
531
+ if (idx === arr.length - 1) {
532
+ return {
533
+ time: targetTime,
534
+ values: [{ min: minValue, max: maxValue }]
535
+ };
536
+ }
537
+ const nextDp = arr[idx + 1];
538
+ if (new Date(nextDp.time).getTime() >= targetTime) {
539
+ const timeDiff = new Date(nextDp.time).getTime() - new Date(curr.time).getTime();
540
+ const targetTimeDiff = targetTime - new Date(curr.time).getTime();
541
+ const minValueDiff = nextDp.values[0]?.min - curr.values[0]?.min;
542
+ const maxValueDiff = nextDp.values[0]?.max - curr.values[0]?.max;
543
+ minValue = curr.values[0]?.min + (minValueDiff * targetTimeDiff) / timeDiff;
544
+ maxValue = curr.values[0]?.max + (maxValueDiff * targetTimeDiff) / timeDiff;
545
+ return {
546
+ time: targetTime,
547
+ values: [{ min: minValue, max: maxValue }]
548
+ };
549
+ }
550
+ }
551
+ return acc;
552
+ });
553
+ }
554
+ getClosestDpValueToTargetTime(dpValuesArray, targetTime) {
555
+ return dpValuesArray.reduce((prev, curr) =>
556
+ //should take the value closest to the target time, for realtime the current time would always change
557
+ Math.abs(new Date(curr.time).getTime() - targetTime) <
558
+ Math.abs(new Date(prev.time).getTime() - targetTime)
559
+ ? curr
560
+ : prev);
561
+ }
562
+ /**
563
+ * This method creates a markPoint on the chart which represents the icon of the alarm or event.
564
+ * @param item Single alarm or event
565
+ * @param dp Data point
566
+ * @param isCleared If the alarm is cleared in case of alarm
567
+ * @param isEvent If the item is an event
568
+ * @param realtime If the chart is in realtime mode
569
+ * @returns MarkPointDataItemOption[]
570
+ */
571
+ createMarkPoint(item, dp, isCleared, isEvent, realtime) {
572
+ // check if dp.values object is empty
573
+ if (!item.creationTime || Object.keys(dp.values).length === 0) {
574
+ return [];
575
+ }
576
+ const dpValuesArray = Object.entries(dp.values).map(([time, values]) => ({
577
+ time: new Date(time).getTime(),
578
+ values
579
+ }));
580
+ const creationTime = new Date(item.creationTime).getTime();
581
+ const lastUpdatedTime = new Date(item['lastUpdated']).getTime();
582
+ let coordCreationTime;
583
+ let coordLastUpdatedTime;
584
+ if (realtime) {
585
+ const lastValue = dpValuesArray[dpValuesArray.length - 1];
586
+ coordCreationTime = [
587
+ item.creationTime,
588
+ lastValue?.values[0]?.min ?? lastValue?.values[1] ?? null
589
+ ];
590
+ coordLastUpdatedTime = [
591
+ item['lastUpdated'],
592
+ lastValue?.values[0]?.min ?? lastValue?.values[1] ?? null
593
+ ];
594
+ }
595
+ else {
596
+ const closestDpValue = this.interpolateBetweenTwoDps(dpValuesArray, creationTime);
597
+ const dpValuesForNewAlarms = this.getClosestDpValueToTargetTime(dpValuesArray, creationTime);
598
+ const closestDpValueLastUpdated = this.interpolateBetweenTwoDps(dpValuesArray, lastUpdatedTime);
599
+ const dpValuesForNewAlarmsLastUpdated = this.getClosestDpValueToTargetTime(dpValuesArray, lastUpdatedTime);
600
+ coordCreationTime = [
601
+ item.creationTime,
602
+ closestDpValue?.values[0]?.min ??
603
+ closestDpValue?.values[1] ??
604
+ dpValuesForNewAlarms?.values[0]?.min ??
605
+ dpValuesForNewAlarms?.values[1] ??
606
+ null
607
+ ];
608
+ coordLastUpdatedTime = [
609
+ item['lastUpdated'],
610
+ closestDpValueLastUpdated?.values[0]?.min ??
611
+ closestDpValueLastUpdated?.values[1] ??
612
+ dpValuesForNewAlarmsLastUpdated?.values[0]?.min ??
613
+ dpValuesForNewAlarmsLastUpdated?.values[1] ??
614
+ null
615
+ ];
616
+ }
617
+ if (isEvent) {
618
+ return [
619
+ {
620
+ coord: coordCreationTime,
621
+ name: item.type,
622
+ itemType: item.type,
623
+ itemStyle: {
624
+ color: item['color']
625
+ },
626
+ symbol: 'circle',
627
+ symbolSize: 24
628
+ },
629
+ {
630
+ coord: coordCreationTime,
631
+ name: item.type,
632
+ itemType: item.type,
633
+ itemStyle: { color: 'white' },
634
+ symbol: ICONS_MAP.EVENT,
635
+ symbolSize: 16
636
+ }
637
+ ];
638
+ }
639
+ return isCleared
640
+ ? [
641
+ {
642
+ coord: coordCreationTime,
643
+ name: item.type,
644
+ itemType: item.type,
645
+ itemStyle: {
646
+ color: item['color']
647
+ },
648
+ symbol: 'circle',
649
+ symbolSize: 24
650
+ },
651
+ {
652
+ coord: coordCreationTime,
653
+ name: item.type,
654
+ itemType: item.type,
655
+ itemStyle: { color: 'white' },
656
+ symbol: ICONS_MAP[item.severity],
657
+ symbolSize: 16
658
+ },
659
+ {
660
+ coord: coordLastUpdatedTime,
661
+ name: item.type,
662
+ itemType: item.type,
663
+ itemStyle: {
664
+ color: item['color']
665
+ },
666
+ symbol: 'circle',
667
+ symbolSize: 24
668
+ },
669
+ {
670
+ coord: coordLastUpdatedTime,
671
+ name: item.type,
672
+ itemType: item.type,
673
+ itemStyle: { color: 'white' },
674
+ symbol: ICONS_MAP.CLEARED,
675
+ symbolSize: 16
676
+ }
677
+ ]
678
+ : [
679
+ {
680
+ coord: coordCreationTime,
681
+ name: item.type,
682
+ itemType: item.type,
683
+ itemStyle: {
684
+ color: item['color']
685
+ },
686
+ symbol: 'circle',
687
+ symbolSize: 24
688
+ },
689
+ {
690
+ coord: coordCreationTime,
691
+ name: item.type,
692
+ itemType: item.type,
693
+ itemStyle: { color: 'white' },
694
+ symbol: ICONS_MAP[item.severity],
695
+ symbolSize: 16
696
+ },
697
+ {
698
+ coord: coordLastUpdatedTime,
699
+ name: item.type,
700
+ itemType: item.type,
701
+ itemStyle: {
702
+ color: item['color']
703
+ },
704
+ symbol: 'circle',
705
+ symbolSize: 24
706
+ },
707
+ {
708
+ coord: coordLastUpdatedTime,
709
+ name: item.type,
710
+ itemType: item.type,
711
+ itemStyle: { color: 'white' },
712
+ symbol: ICONS_MAP[item.severity],
713
+ symbolSize: 16
714
+ }
715
+ ];
716
+ }
717
+ /**
718
+ * This method creates a markLine on the chart which represents the line between every alarm or event on the chart.
719
+ * @param items Array of alarms or events
720
+ * @returns MarkLineDataItemOptionBase[]
721
+ */
722
+ createMarkLine(items) {
723
+ return items.reduce((acc, item) => {
724
+ if (!item.creationTime) {
725
+ return acc;
726
+ }
727
+ if (item.creationTime === item['lastUpdated']) {
728
+ return acc.concat([
729
+ {
730
+ xAxis: item.creationTime,
731
+ itemType: item.type,
732
+ label: { show: false, formatter: () => item.type },
733
+ itemStyle: { color: item['color'] }
734
+ }
735
+ ]);
736
+ }
737
+ else {
738
+ return acc.concat([
739
+ {
740
+ xAxis: item.creationTime,
741
+ itemType: item.type,
742
+ label: { show: false, formatter: () => item.type },
743
+ itemStyle: { color: item['color'] }
744
+ },
745
+ {
746
+ xAxis: item['lastUpdated'],
747
+ itemType: item.type,
748
+ label: { show: false, formatter: () => item.type },
749
+ itemStyle: { color: item['color'] }
750
+ }
751
+ ]);
752
+ }
753
+ }, []);
754
+ }
755
+ getSingleSeries(dp, renderType, idx, isMinMaxChart = false, seriesType = '') {
756
+ const datapointId = dp.__target?.id + dp.fragment + dp.series;
757
+ return {
758
+ datapointId,
759
+ datapointUnit: dp.unit || '',
760
+ // 'id' property is needed as 'seriesId' in tooltip formatter
761
+ id: isMinMaxChart
762
+ ? `${datapointId}/${renderType}${seriesType}`
763
+ : `${datapointId}${seriesType}`,
764
+ name: `${dp.label} (${dp.__target?.['name']})`,
765
+ // datapointLabel used to proper display of tooltip
766
+ datapointLabel: dp.label || '',
767
+ data: Object.entries(dp.values).map(([dateString, values]) => {
768
+ return [dateString, values[0][renderType]];
769
+ }),
770
+ yAxisIndex: idx,
771
+ ...this.chartTypesService.getSeriesOptions(dp, isMinMaxChart, renderType)
772
+ };
773
+ }
774
+ /**
775
+ * This method creates a general tooltip formatter for the chart.
776
+ * @returns TooltipFormatterCallback<TopLevelFormatterParams>
777
+ */
778
+ getTooltipFormatter() {
779
+ return params => {
780
+ if (!Array.isArray(params) || !params[0]?.data) {
781
+ return '';
782
+ }
783
+ const data = params[0].data;
784
+ const XAxisValue = data[0];
785
+ const YAxisReadings = [];
786
+ const allSeries = this.echartsInstance?.getOption()['series'];
787
+ const allDataPointSeries = allSeries.filter(series => series['typeOfSeries'] !== 'alarm' &&
788
+ series['typeOfSeries'] !== 'event' &&
789
+ series['typeOfSeries'] !== 'fake');
790
+ allDataPointSeries.forEach((series) => {
791
+ let value;
792
+ const id = series['id'];
793
+ if (id.endsWith('/min')) {
794
+ const minValue = this.findValueForExactOrEarlierTimestamp(series['data'], XAxisValue);
795
+ if (!minValue) {
796
+ return;
797
+ }
798
+ const maxSeries = allDataPointSeries.find(s => s['id'] === id.replace('/min', '/max'));
799
+ if (!maxSeries) {
800
+ return;
801
+ }
802
+ const maxValue = this.findValueForExactOrEarlierTimestamp(maxSeries['data'], XAxisValue);
803
+ if (maxValue === null) {
804
+ return;
805
+ }
806
+ value =
807
+ `<div class="d-flex a-i-center separator-top p-t-8 p-b-8">` +
808
+ `<label class="text-12 m-r-8 m-b-0">${this.datePipe.transform(minValue[0])}</label>` +
809
+ `<div class="m-l-auto text-12" >${minValue[1]} — ${maxValue[1]}` +
810
+ (series['datapointUnit'] ? ` ${series['datapointUnit']}` : '') +
811
+ `</div></div>`;
812
+ }
813
+ else if (id.endsWith('/max')) {
814
+ // do nothing, value is handled in 'min' case
815
+ return;
816
+ }
817
+ else {
818
+ const seriesValue = this.findValueForExactOrEarlierTimestamp(series['data'], XAxisValue);
819
+ if (!seriesValue) {
820
+ return;
821
+ }
822
+ value =
823
+ `<div class="d-flex a-i-center separator-top p-t-8 p-b-8">` +
824
+ `<label class="text-12 m-r-8 m-b-0">${this.datePipe.transform(seriesValue[0])}</label>` +
825
+ `<div class="m-l-auto text-12" >${seriesValue[1]?.toString()}` +
826
+ (series['datapointUnit'] ? ` ${series['datapointUnit']}` : '') +
827
+ `</div></div>`;
828
+ }
829
+ const itemStyle = series['itemStyle'];
830
+ YAxisReadings.push(`<div class="d-flex a-i-center p-b-8"><span class='dlt-c8y-icon-circle m-r-4' style='color: ${itemStyle.color}'></span>` + // color circle
831
+ `<strong>${series['datapointLabel']} </strong></div>` + // name
832
+ `${value}` // single value or min-max range
833
+ );
834
+ });
835
+ return `<div style="width: ${this.TOOLTIP_WIDTH}px">${YAxisReadings.join('')}</div>`;
836
+ };
837
+ }
838
+ findValueForExactOrEarlierTimestamp(values, timestampString) {
839
+ const timestamp = new Date(timestampString).valueOf();
840
+ return values.reduce((acc, curr) => {
841
+ if (new Date(curr[0]).valueOf() <= timestamp) {
842
+ if (acc === null ||
843
+ Math.abs(new Date(curr[0]).valueOf() - timestamp) <
844
+ Math.abs(new Date(acc[0]).valueOf() - timestamp)) {
845
+ return curr;
846
+ }
847
+ }
848
+ return acc;
849
+ }, null);
850
+ }
851
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: EchartsOptionsService, deps: [{ token: i1.DatePipe }, { token: i2.YAxisService }, { token: i3.ChartTypesService }, { token: i4.AlarmSeverityToIconPipe }, { token: i4.AlarmSeverityToLabelPipe }, { token: i5.Router }], target: i0.ɵɵFactoryTarget.Injectable }); }
852
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: EchartsOptionsService }); }
853
+ }
854
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: EchartsOptionsService, decorators: [{
855
+ type: Injectable
856
+ }], ctorParameters: () => [{ type: i1.DatePipe }, { type: i2.YAxisService }, { type: i3.ChartTypesService }, { type: i4.AlarmSeverityToIconPipe }, { type: i4.AlarmSeverityToLabelPipe }, { type: i5.Router }] });
857
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"echarts-options.service.js","sourceRoot":"","sources":["../../../../echart/services/echarts-options.service.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable @typescript-eslint/no-explicit-any */\nimport { Injectable } from '@angular/core';\nimport { DatePipe } from '@c8y/ngx-components';\nimport type { EChartsOption, SeriesOption } from 'echarts';\nimport { ECharts } from 'echarts';\nimport {\n  DatapointChartRenderType,\n  DpWithValues,\n  DateString,\n  DpValuesItem,\n  MarkLineData,\n  MarkPointData,\n  SeriesDatapointInfo,\n  SeriesValue\n} from '../models';\nimport { YAxisService } from './y-axis.service';\nimport { ChartTypesService } from './chart-types.service';\nimport type { CallbackDataParams, TooltipFormatterCallback } from 'echarts/types/src/util/types';\nimport type { TopLevelFormatterParams } from 'echarts/types/src/component/tooltip/TooltipModel';\nimport { AlarmStatus, IAlarm, IEvent, SeverityType } from '@c8y/client';\nimport { ICONS_MAP } from '../models/svg-icons.model';\nimport { CustomSeriesOptions } from '../models/chart.model';\nimport { Router } from '@angular/router';\nimport { AlarmSeverityToIconPipe, AlarmSeverityToLabelPipe } from '@c8y/ngx-components/alarms';\nimport { Interval, INTERVALS } from '@c8y/ngx-components/interval-picker';\n\ntype TooltipPositionCallback = (\n  point: [number, number], // position of mouse in chart [X, Y]; 0,0 is top left corner\n  _: any, // tooltip data\n  dom: HTMLElement | unknown, // tooltip element\n  __: any, // valid only when mouse is on graphic elements; not relevant in our case\n  size: {\n    contentSize: [number, number]; // size of tooltip\n    viewSize: [number, number]; // size of the chart\n  } | null // size of chart\n) => Partial<Record<'top' | 'bottom' | 'left' | 'right', number>>;\n\nconst INDEX_HTML = '/index.html';\n\n@Injectable()\nexport class EchartsOptionsService {\n  echartsInstance: ECharts | undefined;\n  private TOOLTIP_WIDTH = 300;\n  private tooltipPositionCallback: TooltipPositionCallback | undefined;\n\n  constructor(\n    private datePipe: DatePipe,\n    private yAxisService: YAxisService,\n    private chartTypesService: ChartTypesService,\n    private severityIconPipe: AlarmSeverityToIconPipe,\n    private severityLabelPipe: AlarmSeverityToLabelPipe,\n    private router: Router\n  ) {}\n\n  getChartOptions(\n    datapointsWithValues: DpWithValues[],\n    timeRange: { dateFrom: string; dateTo: string; interval: Interval['id'] },\n    showSplitLines: { YAxis: boolean; XAxis: boolean },\n    events: IEvent[],\n    alarms: IAlarm[],\n    displayOptions: {\n      displayMarkedLine: boolean;\n      displayMarkedPoint: boolean;\n      mergeMatchingDatapoints: boolean;\n      showLabelAndUnit: boolean;\n      showSlider: boolean;\n    },\n    selectedTimeRange?: { dateFrom: string; dateTo: string; interval: Interval['id'] } | null,\n    aggregatedDatapoints?: DpWithValues[],\n    sliderZoomUsed = false\n  ): EChartsOption {\n    const yAxis = this.yAxisService.getYAxis(datapointsWithValues, {\n      showSplitLines: showSplitLines.YAxis,\n      mergeMatchingDatapoints: displayOptions.mergeMatchingDatapoints,\n      showLabelAndUnit: displayOptions.showLabelAndUnit\n    });\n    const leftAxis = yAxis.filter(yx => yx.position === 'left');\n    const gridLeft = leftAxis.length ? leftAxis.length * this.yAxisService.Y_AXIS_OFFSET : 16;\n    const rightAxis = yAxis.filter(yx => yx.position === 'right');\n    const gridRight = rightAxis.length ? rightAxis.length * this.yAxisService.Y_AXIS_OFFSET : 16;\n    let intervalInMs = this.calculateExtendedIntervalInMs(\n      selectedTimeRange?.interval || timeRange.interval || 'hours',\n      selectedTimeRange || timeRange\n    );\n    if (sliderZoomUsed) {\n      intervalInMs = this.calculateExtendedIntervalInMs(timeRange.interval || 'hours', timeRange);\n    }\n    return {\n      grid: {\n        containLabel: false, // axis labels are not taken into account to calculate graph grid\n        left: gridLeft,\n        top: 16,\n        right: gridRight,\n        bottom: 68\n      },\n      dataZoom: [\n        {\n          type: 'inside',\n          // TODO: use 'none' only when this bug is fixed https://github.com/apache/echarts/issues/17858\n          filterMode: datapointsWithValues.some(dp => dp.lineType === 'bars') ? 'filter' : 'none',\n          zoomOnMouseWheel: true,\n          startValue: selectedTimeRange\n            ? selectedTimeRange.dateFrom.valueOf()\n            : timeRange.dateFrom.valueOf(),\n          endValue: selectedTimeRange\n            ? selectedTimeRange.dateTo.valueOf()\n            : timeRange.dateTo.valueOf()\n        },\n        {\n          type: 'slider',\n          show: displayOptions.showSlider,\n          bottom: 8,\n          realtime: false\n        }\n      ], // on realtime, 'none' will cause extending chart line to left edge of the chart\n      animation: false,\n      toolbox: {\n        show: true,\n        itemSize: 0, // toolbox is needed for zooming in action, but we provide our own buttons\n        feature: {\n          dataZoom: {\n            yAxisIndex: 'none'\n          }\n        }\n      },\n      tooltip: {\n        trigger: 'axis',\n        axisPointer: {\n          type: 'cross',\n          snap: true\n        },\n        backgroundColor: 'rgba(255, 255, 255, 0.9)',\n        formatter: this.getTooltipFormatter(),\n        appendToBody: true,\n        position: this.tooltipPosition(),\n        transitionDuration: 0\n      },\n      legend: {\n        show: false\n      },\n      xAxis: {\n        min: new Date(timeRange.dateFrom).valueOf() - intervalInMs,\n        max: timeRange.dateTo,\n        type: 'time',\n        animation: false,\n        axisPointer: {\n          label: {\n            show: false\n          }\n        },\n        axisLine: {\n          // align X axis to 0 of Y axis of datapoint with lineType 'bars'\n          onZeroAxisIndex: datapointsWithValues.findIndex(dp => dp.lineType === 'bars')\n        },\n        axisLabel: {\n          hideOverlap: true,\n          borderWidth: 2, // as there is no margin for labels spacing, transparent border is a workaround\n          borderColor: 'transparent'\n        },\n        splitLine: {\n          show: showSplitLines.XAxis,\n          lineStyle: { opacity: 0.8, type: 'dashed', width: 2 }\n        }\n      },\n      yAxis,\n      series: [\n        ...this.getAggregatedSeries(aggregatedDatapoints || []).filter(\n          series => series !== undefined\n        ),\n        ...this.getChartSeries(datapointsWithValues, events, alarms, displayOptions)\n      ]\n    };\n  }\n\n  calculateExtendedIntervalInMs(interval: Interval['id'], selectedTimeRange): number {\n    let intervalInMs = INTERVALS.find(i => i.id === interval).timespanInMs;\n    switch (interval) {\n      case 'minutes':\n        intervalInMs = INTERVALS.find(i => i.id === interval).timespanInMs * 60;\n        break;\n      case 'hours':\n        intervalInMs = INTERVALS.find(i => i.id === interval).timespanInMs * 24;\n        break;\n      case 'days':\n        intervalInMs = INTERVALS.find(i => i.id === interval).timespanInMs * 28;\n        break;\n      case 'weeks':\n        intervalInMs = INTERVALS.find(i => i.id === interval).timespanInMs * 12;\n        break;\n      case 'months':\n        intervalInMs = INTERVALS.find(i => i.id === interval).timespanInMs * 12;\n        break;\n      case 'custom':\n        intervalInMs =\n          (new Date(selectedTimeRange.dateTo).valueOf() -\n            new Date(selectedTimeRange.dateFrom).valueOf()) *\n          12;\n        break;\n      default:\n        intervalInMs = INTERVALS.find(i => i.id === interval).timespanInMs;\n        break;\n    }\n    return intervalInMs;\n  }\n\n  getAggregatedSeries(aggregatedDatapoints: DpWithValues[]): SeriesOption[] {\n    const series: SeriesOption[] = [];\n    aggregatedDatapoints.forEach((dp, idx) => {\n      const renderType: DatapointChartRenderType = dp.renderType || 'min';\n      if (renderType === 'area') {\n        series.push(this.getSingleSeries(dp, 'min', idx, true, 'aggr'));\n        series.push(this.getSingleSeries(dp, 'max', idx, true, 'aggr'));\n      } else {\n        series.push(this.getSingleSeries(dp, renderType, idx, false, 'aggr'));\n      }\n    });\n\n    series.forEach((s: any) => {\n      s.datapointId = 'aggregated';\n      s.typeOfSeries = 'fake';\n      s.itemStyle = {\n        ...s.itemStyle,\n        opacity: 0\n      };\n      s.lineStyle = {\n        ...s.lineStyle,\n        opacity: 0\n      };\n    });\n\n    return [series[0]];\n  }\n\n  /**\n   * This method is used to get the series for alarms and events.\n   * @param dp - The data point.\n   * @param renderType - The render type.\n   * @param isMinMaxChart - If the chart is min max chart.\n   * @param items - All alarms or events which should be displayed on the chart.\n   * @param itemType - The item type.\n   * @param id - The id of the device\n   */\n  getAlarmOrEventSeries(\n    dp: DpWithValues,\n    renderType: DatapointChartRenderType,\n    isMinMaxChart = false,\n    items: IAlarm[] | IEvent[] = [],\n    itemType: 'alarm' | 'event' = 'alarm',\n    displayOptions = { displayMarkedLine: true, displayMarkedPoint: true },\n    id?: string | number,\n    idx?: number,\n    realtime?: boolean\n  ): CustomSeriesOptions[] {\n    if (!items.length) {\n      return [];\n    }\n\n    if (!displayOptions.displayMarkedLine && !displayOptions.displayMarkedPoint) {\n      return [];\n    }\n\n    //filter items that are not __hidden\n    const filteredItems: IAlarm[] | IEvent[] = items.filter(item => !item['__hidden']);\n    const itemsByType = this.groupByType(filteredItems, 'type');\n    const isAlarm = itemType === 'alarm';\n\n    return Object.entries(itemsByType).flatMap(\n      ([type, itemsOfType]: [string, (IAlarm | IEvent)[]]) => {\n        // Main series data\n        const mainData = itemsOfType.map(item => [item.creationTime, null, 'markLineFlag']);\n        // Is a specific datapoint template selected for this alarm/event type?\n        let isDpTemplateSelected = false;\n\n        // MarkPoint data\n        const markPointData = itemsOfType.reduce<MarkPointData[]>((acc, item) => {\n          isDpTemplateSelected =\n            item['selectedDatapoint'] &&\n            dp['__target'] &&\n            item['selectedDatapoint']['fragment'] === dp['fragment'] &&\n            item['selectedDatapoint']['series'] === dp['series'] &&\n            item['selectedDatapoint']['target'] === dp['__target']['id'];\n          if (dp.__target?.id === item.source.id) {\n            const isCleared = isAlarm && item.status === AlarmStatus.CLEARED;\n            const isEvent = !isAlarm;\n            return acc.concat(this.createMarkPoint(item, dp, isCleared, isEvent, realtime));\n          } else {\n            if (!item.creationTime) {\n              return [];\n            }\n            return acc.concat([\n              {\n                coord: [item.creationTime, null],\n                name: item.type,\n                itemType: item.type,\n                itemStyle: { color: item['color'] }\n              }\n            ]);\n          }\n        }, []);\n\n        // Construct series with markPoint\n        const seriesWithMarkPoint = {\n          id: `${type}/${dp.__target?.id}+${id ? id : ''}-markPoint`,\n          name: `${type}-markPoint`,\n          typeOfSeries: itemType,\n          data: mainData,\n          isDpTemplateSelected,\n          position: 'bottom',\n          silent: true,\n          markPoint: {\n            showSymbol: true,\n            symbolKeepAspect: true,\n            data: markPointData\n          },\n          yAxisIndex: idx,\n          ...this.chartTypesService.getSeriesOptions(dp, isMinMaxChart, renderType)\n        };\n\n        const markLineData = this.createMarkLine(itemsOfType);\n\n        // Construct series with markLine\n        const seriesWithMarkLine = {\n          id: `${type}/${dp.__target?.id}+${id ? id : ''}-markLine`,\n          name: `${type}-markLine`,\n          typeOfSeries: itemType,\n          isDpTemplateSelected,\n          data: mainData,\n          markLine: {\n            showSymbol: false,\n            symbol: ['none', 'none'], // no symbol at the start/end of the line\n            data: markLineData\n          },\n          ...this.chartTypesService.getSeriesOptions(dp, isMinMaxChart, renderType)\n        };\n\n        //depending on the options return only the required series\n        if (displayOptions.displayMarkedLine && displayOptions.displayMarkedPoint) {\n          return [seriesWithMarkLine, seriesWithMarkPoint];\n        } else if (displayOptions.displayMarkedLine) {\n          return [seriesWithMarkLine];\n        } else if (displayOptions.displayMarkedPoint) {\n          return [seriesWithMarkPoint];\n        } else {\n          return null;\n        }\n      }\n    ) as CustomSeriesOptions[];\n  }\n\n  /**\n   * This method is used to get tooltip formatter for alarms and events.\n   * @param tooltipParams - The tooltip parameters.\n   * @param params - The parameters data.\n   * @param allEvents - All events.\n   * @param allAlarms - All alarms.\n   * @returns The formatted string for the tooltip.\n   */\n  getTooltipFormatterForAlarmAndEvents(\n    tooltipParams: CallbackDataParams,\n    params: { data: { itemType: string } },\n    allEvents: IEvent[],\n    allAlarms: IAlarm[]\n  ): string {\n    if (!Array.isArray(tooltipParams)) {\n      return '';\n    }\n    const XAxisValue: string = tooltipParams[0].data[0];\n    const YAxisReadings: string[] = [];\n    const allSeries = this.echartsInstance?.getOption()['series'] as CustomSeriesOptions[];\n\n    // filter out alarm and event series\n    const allDataPointSeries = allSeries.filter(\n      series => series['typeOfSeries'] !== 'alarm' && series['typeOfSeries'] !== 'event'\n    );\n\n    this.processSeries(allDataPointSeries, XAxisValue, YAxisReadings);\n\n    // find event and alarm of the same type as the hovered markedLine or markedPoint\n    const event = allEvents.find(e => e.type === params.data.itemType);\n    const alarm = allAlarms.find(a => a.type === params.data.itemType);\n\n    let value = '';\n    if (event) {\n      value = this.processEvent(event);\n    }\n\n    if (alarm) {\n      this.processAlarm(alarm).then(alarmVal => {\n        value = alarmVal;\n        YAxisReadings.push(value);\n        const options = this.echartsInstance?.getOption() as EChartsOption;\n        if (!options.tooltip || !Array.isArray(options.tooltip)) {\n          return;\n        }\n        const updatedOptions: Partial<SeriesOption> = {\n          tooltip: options['tooltip'][0]\n        };\n\n        if (!updatedOptions.tooltip) {\n          return;\n        }\n        updatedOptions.tooltip.formatter = `<div style=\"width: ${this.TOOLTIP_WIDTH}px\">${YAxisReadings.join('')}</div>`;\n        updatedOptions.tooltip.transitionDuration = 0;\n        updatedOptions.tooltip.position = this.tooltipPosition();\n        this.echartsInstance?.setOption(updatedOptions);\n        return;\n      });\n    }\n    YAxisReadings.push(value);\n\n    return `<div style=\"width: 300px\">${YAxisReadings.join('')}</div>`;\n  }\n\n  private tooltipPosition(): TooltipPositionCallback {\n    let lastPositionOfTooltip: Partial<Record<'top' | 'bottom' | 'left' | 'right', number>> = {};\n    if (this.tooltipPositionCallback) {\n      return this.tooltipPositionCallback;\n    }\n\n    this.tooltipPositionCallback = (\n      point: [number, number], // position of mouse in chart [X, Y]; 0,0 is top left corner\n      _: any, // tooltip data\n      dom: HTMLElement | unknown, // tooltip element\n      __: any,\n      size: {\n        contentSize: [number, number]; // size of tooltip\n        viewSize: [number, number];\n      } | null // size of chart\n    ) => {\n      const offset = 10;\n      const [mouseX, mouseY] = point;\n      const chartWidth = size?.viewSize[0] || 0;\n      const chartHeight = size?.viewSize[1] || 0;\n      const tooltipWidth = size?.contentSize[0] || 0;\n      const tooltipHeight = size?.contentSize[1] || 0;\n      const tooltipRect = (dom as HTMLElement)?.getBoundingClientRect();\n      const tooltipOverflowsBottomEdge = tooltipRect.bottom > window.innerHeight;\n\n      const tooltipOverflowsRightEdge = tooltipRect.right > window.innerWidth;\n\n      const tooltipWouldOverflowBottomEdgeOnPositionChange =\n        !lastPositionOfTooltip.top &&\n        tooltipRect.bottom + 2 * offset + tooltipHeight > window.innerHeight;\n\n      const tooltipWouldOverflowRightEdgeOnPositionChange =\n        !lastPositionOfTooltip.left &&\n        tooltipRect.right + 2 * offset + tooltipWidth > window.innerWidth;\n\n      let verticalPosition: { top: number } | { bottom: number } = {\n        top: mouseY + offset\n      };\n      let horizontalPosition: { left: number } | { right: number } = {\n        left: mouseX + offset\n      };\n\n      if (tooltipOverflowsBottomEdge || tooltipWouldOverflowBottomEdgeOnPositionChange) {\n        verticalPosition = {\n          bottom: chartHeight - mouseY + offset\n        };\n      }\n      if (tooltipOverflowsRightEdge || tooltipWouldOverflowRightEdgeOnPositionChange) {\n        horizontalPosition = {\n          right: chartWidth - mouseX + offset\n        };\n      }\n\n      lastPositionOfTooltip = {\n        ...verticalPosition,\n        ...horizontalPosition\n      };\n      return lastPositionOfTooltip;\n    };\n    return this.tooltipPositionCallback;\n  }\n\n  /**\n   * This method is used to add the data point info to the tooltip.\n   * @param allDataPointSeries - All the data point series.\n   * @param XAxisValue - The X Axis value.\n   * @param YAxisReadings - The Y Axis readings.\n   */\n  private processSeries(\n    allDataPointSeries: CustomSeriesOptions[],\n    XAxisValue: string,\n    YAxisReadings: string[]\n  ): void {\n    allDataPointSeries.forEach((series: any) => {\n      let value = '';\n      if (series.id.endsWith('/min')) {\n        value = this.processMinSeries(series, allDataPointSeries, XAxisValue);\n      } else if (!series.id.endsWith('/max')) {\n        value = this.processRegularSeries(series, XAxisValue);\n      }\n\n      if (value) {\n        YAxisReadings.push(\n          `<div class=\"d-flex a-i-center p-b-8\"><span class='dlt-c8y-icon-circle m-r-4' style='color: ${series.itemStyle.color};'></span>` + // color circle\n            `<strong>${series.datapointLabel}</strong></div>` + // name\n            `${value}` // single value or min-max range\n        );\n      }\n    });\n  }\n\n  /**\n   * This method is used to process the min series.\n   * @param series - The series.\n   * @param allDataPointSeries - All the data point series.\n   * @param XAxisValue - The X Axis value.\n   * @returns The processed value.\n   */\n  private processMinSeries(\n    series: any,\n    allDataPointSeries: CustomSeriesOptions[],\n    XAxisValue: string\n  ): string {\n    const minValue = this.findValueForExactOrEarlierTimestamp(series.data, XAxisValue);\n    if (!minValue) {\n      return '';\n    }\n    const maxSeries = allDataPointSeries.find(s => s['id'] === series.id.replace('/min', '/max'));\n    const maxValue = this.findValueForExactOrEarlierTimestamp(\n      maxSeries?.['data'] as SeriesValue[],\n      XAxisValue\n    );\n    return (\n      `<div class=\"d-flex a-i-center separator-top p-t-8 p-b-8\"><label class=\"text-12 m-r-8 m-b-0\">${this.datePipe.transform(minValue[0])}</label>` +\n      `<span class=\"m-l-auto text-12\">${minValue[1]} — ${maxValue?.[1]}` +\n      (series.datapointUnit ? ` ${series.datapointUnit}` : '') +\n      `</span></div>`\n    );\n  }\n\n  /**\n   * This method is used to process the regular series.\n   * @param series - The series.\n   * @param XAxisValue - The X Axis value.\n   * @returns The processed value.\n   */\n  private processRegularSeries(series: any, XAxisValue: string): string {\n    const seriesValue = this.findValueForExactOrEarlierTimestamp(series.data, XAxisValue);\n    if (!seriesValue) {\n      return '';\n    }\n    return (\n      `<div class=\"d-flex a-i-center p-t-8 p-b-8 separator-top\">` +\n      `<label class=\"m-b-0 m-r-8 text-12\">${this.datePipe.transform(\n        seriesValue[0]\n      )}</label><span class=\"m-l-auto text-12\">` +\n      seriesValue[1]?.toString() +\n      (series.datapointUnit ? ` ${series.datapointUnit}` : '') +\n      `</span></div>`\n    );\n  }\n\n  /**\n   * This method is used to process the event tooltip.\n   * @param event - The event object.\n   * @returns The processed value.\n   */\n  private processEvent(event: IEvent): string {\n    let value = `<ul class=\"list-unstyled small separator-top\">`;\n    value += `<li class=\"p-t-4 p-b-4 d-flex separator-bottom text-no-wrap\"><label class=\"small m-b-0 m-r-8\">Event type</label><code class=\"m-l-auto\">${event.type}</code></li>`;\n    value += `<li class=\"p-t-4 p-b-4 d-flex separator-bottom text-no-wrap\"><label class=\"small m-b-0 m-r-8\">Event text</label><span class=\"m-l-auto\">${event.text}<span></li>`;\n    value += `<li class=\"p-t-4 p-b-4 d-flex separator-bottom text-no-wrap\"><label class=\"small m-b-0 m-r-8\">Last update</label><span class=\"m-l-auto\">${this.datePipe.transform(event['lastUpdated'])}<span></li>`;\n    value += `</ul>`;\n    return value;\n  }\n\n  /**\n   * This method is used to process the alarm tooltip.\n   * @param alarm - The alarm object.\n   * @returns The processed value.\n   */\n  private async processAlarm(alarm: IAlarm): Promise<string> {\n    let value = `<ul class=\"list-unstyled small separator-top m-0\">`;\n    value += `<li class=\"p-t-4 p-b-4 d-flex a-i-center separator-bottom text-no-wrap\"><label class=\"text-label-small m-b-0 m-r-8\">Alarm Severity</label>`;\n    value += `<span class=\"small d-inline-flex a-i-center gap-4 m-l-auto\"><i class=\"stroked-icon icon-14 status dlt-c8y-icon-${this.severityIconPipe.transform(alarm.severity)} ${alarm.severity.toLowerCase()}\" > </i> ${this.severityLabelPipe.transform(alarm.severity)} </span></li>`;\n    value += `<li class=\"p-t-4 p-b-4 d-flex separator-bottom text-no-wrap\"><label class=\"text-label-small m-b-0 m-r-8\">Alarm Type</label><span class=\"small m-l-auto\"><code>${alarm.type}</code></span></li>`;\n    value += `<li class=\"p-t-4 p-b-4 d-flex separator-bottom text-no-wrap\"><label class=\"text-label-small m-b-0 m-r-8\">Message</label><span class=\"small m-l-auto\" style=\"overflow: hidden; text-overflow: ellipsis;\" title=\"${alarm.text}\">${alarm.text}</span></li>`;\n    value += `<li class=\"p-t-4 p-b-4 d-flex separator-bottom text-no-wrap\"><label class=\"text-label-small m-b-0 m-r-8\">Last Updated</label><span class=\"small m-l-auto\">${this.datePipe.transform(alarm['lastUpdated'])}</span></li>`;\n    const exists = await this.alarmRouteExists();\n    if (exists) {\n      const currentUrl = window.location.href;\n      const baseUrlIndex = currentUrl.indexOf(INDEX_HTML);\n      const baseUrl = currentUrl.substring(0, baseUrlIndex + INDEX_HTML.length);\n      value += `<li class=\"p-t-4 p-b-4 d-flex separator-bottom text-no-wrap\"><label class=\"text-label-small m-b-0 m-r-8\">Link</label><span class=\"small m-l-auto\"><a href=\"${baseUrl}#/alarms/${alarm.id}\">Alarm Details</a></span></li>`;\n    }\n    value += `<li class=\"p-t-4 p-b-4 d-flex text-no-wrap\"><label class=\"text-label-small m-b-0 m-r-8\">Alarm count</label><span class=\"small m-l-auto\"><span class=\"badge badge-info\">${alarm.count}</span></span></li>`;\n    value += `</ul>`;\n    return value;\n  }\n\n  private async alarmRouteExists(): Promise<boolean> {\n    const exists = this.router.config.some(route => {\n      return `${route.path}` === 'alarms';\n    });\n    return exists;\n  }\n\n  private getChartSeries(\n    datapointsWithValues: DpWithValues[],\n    events: IEvent[],\n    alarms: IAlarm[],\n    displayOptions: { displayMarkedLine: boolean; displayMarkedPoint: boolean }\n  ): CustomSeriesOptions[] | SeriesOption[] {\n    const series: SeriesOption[] = [];\n    let eventSeries: CustomSeriesOptions[] = [];\n    let alarmSeries: CustomSeriesOptions[] = [];\n    datapointsWithValues.forEach((dp, idx) => {\n      const renderType: DatapointChartRenderType = dp.renderType || 'min';\n      if (renderType === 'area') {\n        series.push(this.getSingleSeries(dp, 'min', idx, true));\n        series.push(this.getSingleSeries(dp, 'max', idx, true));\n      } else {\n        series.push(this.getSingleSeries(dp, renderType, idx, false));\n      }\n\n      const newEventSeries = this.getAlarmOrEventSeries(\n        dp,\n        renderType,\n        false,\n        events,\n        'event',\n        displayOptions,\n        null,\n        idx\n      );\n      const newAlarmSeries = this.getAlarmOrEventSeries(\n        dp,\n        renderType,\n        false,\n        alarms,\n        'alarm',\n        displayOptions,\n        null,\n        idx\n      );\n      eventSeries = [...eventSeries, ...newEventSeries];\n      alarmSeries = [...alarmSeries, ...newAlarmSeries];\n    });\n    const deduplicateFilterCallback = (\n      obj1: CustomSeriesOptions,\n      i: number,\n      arr: SeriesOption[]\n    ): obj1 is CustomSeriesOptions => {\n      const duplicates = arr.filter(obj2 => obj1['id'] === obj2['id'] && i !== arr.indexOf(obj2));\n\n      if (duplicates.length > 0) {\n        return obj1['isDpTemplateSelected'] as boolean;\n      }\n\n      return true;\n    };\n\n    const deduplicateFilterCallbackFallback = (\n      obj1: SeriesOption,\n      i: number,\n      arr: SeriesOption[]\n    ): obj1 is SeriesOption => arr.findIndex(obj2 => obj2['id'] === obj1['id']) === i;\n\n    let deduplicatedEvents = eventSeries.filter(deduplicateFilterCallback);\n    let deduplicatedAlarms = alarmSeries.filter(deduplicateFilterCallback);\n    if (deduplicatedAlarms.length === 0) {\n      deduplicatedAlarms = alarmSeries.filter(deduplicateFilterCallbackFallback);\n    }\n    if (deduplicatedEvents.length === 0) {\n      deduplicatedEvents = eventSeries.filter(deduplicateFilterCallbackFallback);\n    }\n    return [...series, ...deduplicatedEvents, ...deduplicatedAlarms];\n  }\n\n  private groupByType(\n    items: IAlarm[] | IEvent[],\n    typeField: string\n  ): Record<string, IAlarm[] | IEvent[]> {\n    return items.reduce((grouped, item) => {\n      (grouped[item[typeField]] = grouped[item[typeField]] || []).push(item);\n      return grouped;\n    }, {} as any);\n  }\n\n  /**\n   * This method interpolates between two data points. The goal is to place the markPoint on the chart in the right place.\n   * @param dpValuesArray array of data points\n   * @param targetTime time of the alarm or event\n   * @returns interpolated data point\n   */\n  private interpolateBetweenTwoDps(\n    dpValuesArray: DpValuesItem[],\n    targetTime: number\n  ): DpValuesItem {\n    let maxValue: number;\n    let minValue: number;\n    return dpValuesArray.reduce((acc, curr, idx, arr) => {\n      if (new Date(curr.time).getTime() <= targetTime) {\n        if (idx === arr.length - 1) {\n          return {\n            time: targetTime,\n            values: [{ min: minValue, max: maxValue }]\n          };\n        }\n        const nextDp = arr[idx + 1];\n        if (new Date(nextDp.time).getTime() >= targetTime) {\n          const timeDiff = new Date(nextDp.time).getTime() - new Date(curr.time).getTime();\n          const targetTimeDiff = targetTime - new Date(curr.time).getTime();\n          const minValueDiff = nextDp.values[0]?.min - curr.values[0]?.min;\n          const maxValueDiff = nextDp.values[0]?.max - curr.values[0]?.max;\n          minValue = curr.values[0]?.min + (minValueDiff * targetTimeDiff) / timeDiff;\n          maxValue = curr.values[0]?.max + (maxValueDiff * targetTimeDiff) / timeDiff;\n          return {\n            time: targetTime,\n            values: [{ min: minValue, max: maxValue }]\n          };\n        }\n      }\n      return acc;\n    });\n  }\n\n  private getClosestDpValueToTargetTime(\n    dpValuesArray: DpValuesItem[],\n    targetTime: number\n  ): DpValuesItem {\n    return dpValuesArray.reduce((prev, curr) =>\n      //should take the value closest to the target time, for realtime the current time would always change\n      Math.abs(new Date(curr.time).getTime() - targetTime) <\n      Math.abs(new Date(prev.time).getTime() - targetTime)\n        ? curr\n        : prev\n    );\n  }\n\n  /**\n   * This method creates a markPoint on the chart which represents the icon of the alarm or event.\n   * @param item Single alarm or event\n   * @param dp Data point\n   * @param isCleared If the alarm is cleared in case of alarm\n   * @param isEvent If the item is an event\n   * @param realtime If the chart is in realtime mode\n   * @returns MarkPointDataItemOption[]\n   */\n  private createMarkPoint(\n    item: IAlarm | IEvent,\n    dp: DpWithValues,\n    isCleared: boolean,\n    isEvent: boolean,\n    realtime: boolean\n  ): MarkPointData[] {\n    // check if dp.values object is empty\n    if (!item.creationTime || Object.keys(dp.values).length === 0) {\n      return [];\n    }\n\n    const dpValuesArray: DpValuesItem[] = Object.entries(dp.values).map(([time, values]) => ({\n      time: new Date(time).getTime(),\n      values\n    }));\n    const creationTime = new Date(item.creationTime).getTime();\n    const lastUpdatedTime = new Date(item['lastUpdated']).getTime();\n\n    let coordCreationTime: any;\n    let coordLastUpdatedTime: any;\n\n    if (realtime) {\n      const lastValue = dpValuesArray[dpValuesArray.length - 1];\n      coordCreationTime = [\n        item.creationTime,\n        lastValue?.values[0]?.min ?? lastValue?.values[1] ?? null\n      ];\n      coordLastUpdatedTime = [\n        item['lastUpdated'],\n        lastValue?.values[0]?.min ?? lastValue?.values[1] ?? null\n      ];\n    } else {\n      const closestDpValue = this.interpolateBetweenTwoDps(dpValuesArray, creationTime);\n      const dpValuesForNewAlarms = this.getClosestDpValueToTargetTime(dpValuesArray, creationTime);\n      const closestDpValueLastUpdated = this.interpolateBetweenTwoDps(\n        dpValuesArray,\n        lastUpdatedTime\n      );\n      const dpValuesForNewAlarmsLastUpdated = this.getClosestDpValueToTargetTime(\n        dpValuesArray,\n        lastUpdatedTime\n      );\n\n      coordCreationTime = [\n        item.creationTime,\n        closestDpValue?.values[0]?.min ??\n          closestDpValue?.values[1] ??\n          dpValuesForNewAlarms?.values[0]?.min ??\n          dpValuesForNewAlarms?.values[1] ??\n          null\n      ];\n      coordLastUpdatedTime = [\n        item['lastUpdated'],\n        closestDpValueLastUpdated?.values[0]?.min ??\n          closestDpValueLastUpdated?.values[1] ??\n          dpValuesForNewAlarmsLastUpdated?.values[0]?.min ??\n          dpValuesForNewAlarmsLastUpdated?.values[1] ??\n          null\n      ];\n    }\n\n    if (isEvent) {\n      return [\n        {\n          coord: coordCreationTime,\n          name: item.type,\n          itemType: item.type,\n          itemStyle: {\n            color: item['color']\n          },\n          symbol: 'circle',\n          symbolSize: 24\n        },\n        {\n          coord: coordCreationTime,\n          name: item.type,\n          itemType: item.type,\n          itemStyle: { color: 'white' },\n          symbol: ICONS_MAP.EVENT,\n          symbolSize: 16\n        }\n      ];\n    }\n\n    return isCleared\n      ? [\n          {\n            coord: coordCreationTime,\n            name: item.type,\n            itemType: item.type,\n            itemStyle: {\n              color: item['color']\n            },\n            symbol: 'circle',\n            symbolSize: 24\n          },\n          {\n            coord: coordCreationTime,\n            name: item.type,\n            itemType: item.type,\n            itemStyle: { color: 'white' },\n            symbol: ICONS_MAP[item.severity as SeverityType],\n            symbolSize: 16\n          },\n          {\n            coord: coordLastUpdatedTime,\n            name: item.type,\n            itemType: item.type,\n            itemStyle: {\n              color: item['color']\n            },\n            symbol: 'circle',\n            symbolSize: 24\n          },\n          {\n            coord: coordLastUpdatedTime,\n            name: item.type,\n            itemType: item.type,\n            itemStyle: { color: 'white' },\n            symbol: ICONS_MAP.CLEARED,\n            symbolSize: 16\n          }\n        ]\n      : [\n          {\n            coord: coordCreationTime,\n            name: item.type,\n            itemType: item.type,\n            itemStyle: {\n              color: item['color']\n            },\n            symbol: 'circle',\n            symbolSize: 24\n          },\n          {\n            coord: coordCreationTime,\n            name: item.type,\n            itemType: item.type,\n            itemStyle: { color: 'white' },\n            symbol: ICONS_MAP[item.severity as SeverityType],\n            symbolSize: 16\n          },\n          {\n            coord: coordLastUpdatedTime,\n            name: item.type,\n            itemType: item.type,\n            itemStyle: {\n              color: item['color']\n            },\n            symbol: 'circle',\n            symbolSize: 24\n          },\n          {\n            coord: coordLastUpdatedTime,\n            name: item.type,\n            itemType: item.type,\n            itemStyle: { color: 'white' },\n            symbol: ICONS_MAP[item.severity as SeverityType],\n            symbolSize: 16\n          }\n        ];\n  }\n\n  /**\n   * This method creates a markLine on the chart which represents the line between every alarm or event on the chart.\n   * @param items Array of alarms or events\n   * @returns MarkLineDataItemOptionBase[]\n   */\n  private createMarkLine<T extends IAlarm | IEvent>(items: T[]): MarkLineData[] {\n    return items.reduce<MarkLineData[]>((acc, item) => {\n      if (!item.creationTime) {\n        return acc;\n      }\n      if (item.creationTime === item['lastUpdated']) {\n        return acc.concat([\n          {\n            xAxis: item.creationTime,\n            itemType: item.type,\n            label: { show: false, formatter: () => item.type },\n            itemStyle: { color: item['color'] }\n          }\n        ]);\n      } else {\n        return acc.concat([\n          {\n            xAxis: item.creationTime,\n            itemType: item.type,\n            label: { show: false, formatter: () => item.type },\n            itemStyle: { color: item['color'] }\n          },\n          {\n            xAxis: item['lastUpdated'],\n            itemType: item.type,\n            label: { show: false, formatter: () => item.type },\n            itemStyle: { color: item['color'] }\n          }\n        ]);\n      }\n    }, []);\n  }\n\n  private getSingleSeries(\n    dp: DpWithValues,\n    renderType: Exclude<DatapointChartRenderType, 'area'>,\n    idx: number,\n    isMinMaxChart = false,\n    seriesType = ''\n  ): SeriesOption & SeriesDatapointInfo {\n    const datapointId = dp.__target?.id + dp.fragment + dp.series;\n    return {\n      datapointId,\n      datapointUnit: dp.unit || '',\n      // 'id' property is needed as 'seriesId' in tooltip formatter\n      id: isMinMaxChart\n        ? `${datapointId}/${renderType}${seriesType}`\n        : `${datapointId}${seriesType}`,\n      name: `${dp.label} (${dp.__target?.['name']})`,\n      // datapointLabel used to proper display of tooltip\n      datapointLabel: dp.label || '',\n      data: Object.entries(dp.values).map(([dateString, values]) => {\n        return [dateString, values[0][renderType]];\n      }),\n      yAxisIndex: idx,\n      ...this.chartTypesService.getSeriesOptions(dp, isMinMaxChart, renderType)\n    };\n  }\n\n  /**\n   * This method creates a general tooltip formatter for the chart.\n   * @returns TooltipFormatterCallback<TopLevelFormatterParams>\n   */\n  private getTooltipFormatter(): TooltipFormatterCallback<TopLevelFormatterParams> {\n    return params => {\n      if (!Array.isArray(params) || !params[0]?.data) {\n        return '';\n      }\n      const data = params[0].data as [string, ...any[]];\n      const XAxisValue: string = data[0];\n      const YAxisReadings: string[] = [];\n      const allSeries = this.echartsInstance?.getOption()['series'] as CustomSeriesOptions[];\n\n      const allDataPointSeries = allSeries.filter(\n        series =>\n          series['typeOfSeries'] !== 'alarm' &&\n          series['typeOfSeries'] !== 'event' &&\n          series['typeOfSeries'] !== 'fake'\n      );\n\n      allDataPointSeries.forEach((series: CustomSeriesOptions) => {\n        let value: string;\n        const id = series['id'] as string;\n        if (id.endsWith('/min')) {\n          const minValue = this.findValueForExactOrEarlierTimestamp(\n            series['data'] as SeriesValue[],\n            XAxisValue\n          );\n          if (!minValue) {\n            return;\n          }\n          const maxSeries = allDataPointSeries.find(s => s['id'] === id.replace('/min', '/max'));\n          if (!maxSeries) {\n            return;\n          }\n          const maxValue = this.findValueForExactOrEarlierTimestamp(\n            maxSeries['data'] as SeriesValue[],\n            XAxisValue\n          );\n          if (maxValue === null) {\n            return;\n          }\n          value =\n            `<div class=\"d-flex a-i-center separator-top p-t-8 p-b-8\">` +\n            `<label class=\"text-12 m-r-8 m-b-0\">${this.datePipe.transform(minValue[0])}</label>` +\n            `<div class=\"m-l-auto text-12\" >${minValue[1]} — ${maxValue[1]}` +\n            (series['datapointUnit'] ? ` ${series['datapointUnit']}` : '') +\n            `</div></div>`;\n        } else if (id.endsWith('/max')) {\n          // do nothing, value is handled  in 'min' case\n          return;\n        } else {\n          const seriesValue = this.findValueForExactOrEarlierTimestamp(\n            series['data'] as SeriesValue[],\n            XAxisValue\n          );\n          if (!seriesValue) {\n            return;\n          }\n          value =\n            `<div class=\"d-flex a-i-center separator-top p-t-8 p-b-8\">` +\n            `<label class=\"text-12 m-r-8 m-b-0\">${this.datePipe.transform(seriesValue[0])}</label>` +\n            `<div class=\"m-l-auto text-12\" >${seriesValue[1]?.toString()}` +\n            (series['datapointUnit'] ? ` ${series['datapointUnit']}` : '') +\n            `</div></div>`;\n        }\n\n        const itemStyle = series['itemStyle'] as { color: string };\n\n        YAxisReadings.push(\n          `<div class=\"d-flex a-i-center p-b-8\"><span class='dlt-c8y-icon-circle m-r-4' style='color: ${itemStyle.color}'></span>` + // color circle\n            `<strong>${series['datapointLabel']} </strong></div>` + // name\n            `${value}` // single value or min-max range\n        );\n      });\n\n      return `<div style=\"width: ${this.TOOLTIP_WIDTH}px\">${YAxisReadings.join('')}</div>`;\n    };\n  }\n\n  private findValueForExactOrEarlierTimestamp(\n    values: SeriesValue[],\n    timestampString: DateString\n  ): SeriesValue | null {\n    const timestamp = new Date(timestampString).valueOf();\n    return values.reduce((acc: SeriesValue | null, curr: SeriesValue) => {\n      if (new Date(curr[0]).valueOf() <= timestamp) {\n        if (\n          acc === null ||\n          Math.abs(new Date(curr[0]).valueOf() - timestamp) <\n            Math.abs(new Date(acc[0]).valueOf() - timestamp)\n        ) {\n          return curr;\n        }\n      }\n      return acc;\n    }, null);\n  }\n}\n"]}