@provoly/dashboard 1.0.0 → 1.1.1

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 (73) hide show
  1. package/assets/svgs/analytic.svg +1 -0
  2. package/esm2022/admin/components/admin-dataset/admin-select-dataset/admin-select-dataset.component.mjs +5 -4
  3. package/esm2022/import/components/import.component.mjs +22 -10
  4. package/esm2022/import/i18n/en.translations.mjs +3 -1
  5. package/esm2022/import/i18n/fr.translations.mjs +3 -1
  6. package/esm2022/lib/core/access/access.service.mjs +4 -1
  7. package/esm2022/lib/core/i18n/en.translations.mjs +2 -1
  8. package/esm2022/lib/core/i18n/fr.translations.mjs +2 -1
  9. package/esm2022/lib/core/model/public-api.mjs +2 -1
  10. package/esm2022/lib/core/model/widget-analytic-manifest.interface.mjs +2 -0
  11. package/esm2022/lib/core/store/data-source/data-source.effects.mjs +5 -2
  12. package/esm2022/lib/core/toolbox/toolbox-manifest.service.mjs +12 -1
  13. package/esm2022/lib/core/toolbox/toolbox-menu.service.mjs +4 -3
  14. package/esm2022/lib/dashboard/store/dashboard.actions.mjs +2 -2
  15. package/esm2022/lib/dashboard/store/dashboard.effects.mjs +6 -4
  16. package/esm2022/lib/dashboard/store/dashboard.reducers.mjs +17 -9
  17. package/esm2022/lib/dashboard/store/dashboard.selectors.mjs +8 -2
  18. package/esm2022/presentation/components/presentation.component.mjs +18 -9
  19. package/esm2022/widgets/widget-analytic/component/widget-analytic.component.mjs +108 -0
  20. package/esm2022/widgets/widget-analytic/i18n/en.translations.mjs +13 -0
  21. package/esm2022/widgets/widget-analytic/i18n/fr.translations.mjs +13 -0
  22. package/esm2022/widgets/widget-analytic/provoly-dashboard-widgets-widget-analytic.mjs +5 -0
  23. package/esm2022/widgets/widget-analytic/public-api.mjs +3 -0
  24. package/esm2022/widgets/widget-analytic/style/css.component.mjs +11 -0
  25. package/esm2022/widgets/widget-analytic/widget-analytic.module.mjs +70 -0
  26. package/esm2022/widgets/widget-map/component/widget-map.component.mjs +14 -137
  27. package/esm2022/widgets/widget-map/interaction/interaction-manager.class.mjs +2 -34
  28. package/esm2022/widgets/widget-map/interaction/tooltip-manager.class.mjs +191 -0
  29. package/esm2022/widgets/widget-map/public-api.mjs +2 -1
  30. package/esm2022/widgets/widget-tile/component/widget-tile.component.mjs +24 -5
  31. package/fesm2022/provoly-dashboard-admin.mjs +3 -2
  32. package/fesm2022/provoly-dashboard-admin.mjs.map +1 -1
  33. package/fesm2022/provoly-dashboard-import.mjs +25 -9
  34. package/fesm2022/provoly-dashboard-import.mjs.map +1 -1
  35. package/fesm2022/provoly-dashboard-presentation.mjs +17 -8
  36. package/fesm2022/provoly-dashboard-presentation.mjs.map +1 -1
  37. package/fesm2022/provoly-dashboard-widgets-widget-analytic.mjs +211 -0
  38. package/fesm2022/provoly-dashboard-widgets-widget-analytic.mjs.map +1 -0
  39. package/fesm2022/provoly-dashboard-widgets-widget-map.mjs +201 -169
  40. package/fesm2022/provoly-dashboard-widgets-widget-map.mjs.map +1 -1
  41. package/fesm2022/provoly-dashboard-widgets-widget-tile.mjs +23 -4
  42. package/fesm2022/provoly-dashboard-widgets-widget-tile.mjs.map +1 -1
  43. package/fesm2022/provoly-dashboard.mjs +52 -16
  44. package/fesm2022/provoly-dashboard.mjs.map +1 -1
  45. package/import/components/import.component.d.ts +4 -2
  46. package/import/i18n/en.translations.d.ts +2 -0
  47. package/import/i18n/fr.translations.d.ts +2 -0
  48. package/lib/core/access/access.service.d.ts +3 -0
  49. package/lib/core/i18n/en.translations.d.ts +1 -0
  50. package/lib/core/i18n/fr.translations.d.ts +1 -0
  51. package/lib/core/model/public-api.d.ts +1 -0
  52. package/lib/core/model/widget-analytic-manifest.interface.d.ts +6 -0
  53. package/lib/core/store/data-source/data-source.effects.d.ts +4 -2
  54. package/lib/dashboard/store/dashboard.actions.d.ts +11 -1
  55. package/lib/dashboard/store/dashboard.effects.d.ts +2 -0
  56. package/lib/dashboard/store/dashboard.reducers.d.ts +3 -1
  57. package/lib/dashboard/store/dashboard.selectors.d.ts +1 -1
  58. package/package.json +54 -48
  59. package/presentation/components/presentation.component.d.ts +5 -3
  60. package/styles/components/_o-pry-admin-classes-customize.scss +1 -0
  61. package/widgets/widget-analytic/component/widget-analytic.component.d.ts +47 -0
  62. package/widgets/widget-analytic/i18n/en.translations.d.ts +12 -0
  63. package/widgets/widget-analytic/i18n/fr.translations.d.ts +12 -0
  64. package/widgets/widget-analytic/index.d.ts +5 -0
  65. package/widgets/widget-analytic/public-api.d.ts +2 -0
  66. package/widgets/widget-analytic/style/_o-widget-analytic.scss +43 -0
  67. package/widgets/widget-analytic/style/css.component.d.ts +5 -0
  68. package/widgets/widget-analytic/widget-analytic.module.d.ts +19 -0
  69. package/widgets/widget-map/component/widget-map.component.d.ts +9 -22
  70. package/widgets/widget-map/interaction/interaction-manager.class.d.ts +0 -5
  71. package/widgets/widget-map/interaction/tooltip-manager.class.d.ts +43 -0
  72. package/widgets/widget-map/public-api.d.ts +1 -0
  73. package/widgets/widget-tile/component/widget-tile.component.d.ts +4 -1
@@ -1,5 +1,4 @@
1
- import { ContextMenuActions, TooltipMode } from '@provoly/dashboard';
2
- import { Feature, Overlay } from 'ol';
1
+ import { ContextMenuActions } from '@provoly/dashboard';
3
2
  import { ScaleLine } from 'ol/control';
4
3
  import { toLonLat } from 'ol/proj';
5
4
  import { SelectionInteraction } from './selection-interaction.class';
@@ -24,30 +23,12 @@ export class InteractionManager {
24
23
  }));
25
24
  e.preventDefault();
26
25
  }
27
- },
28
- popup: (event) => {
29
- const values = this.component.map.getFeaturesAtPixel(event.pixel, { hitTolerance: 5 });
30
- const valuesToDisplay = [...new Set(values.map((value) => value.get('id')).filter((id) => !!id))]
31
- .map((id) => values.find((value) => value.get('id') === id))
32
- .map((value) => ({
33
- oClass: value?.get('oClass'),
34
- coordinates: event.coordinate,
35
- values: value instanceof Feature ? value.getKeys().reduce((p, c) => ({ ...p, [c]: value?.get(c) }), {}) : undefined
36
- }))
37
- .filter((value) => value.oClass);
38
- this.component.classicFeatures$.next(valuesToDisplay);
39
- const wmsLayers = this.component.getWMSLayers();
40
- if (wmsLayers) {
41
- this.component.getFeatureFromServer(wmsLayers, event.coordinate);
42
- }
43
- this.tooltipEvent = event;
44
26
  }
45
27
  };
46
28
  }
47
29
  init(component) {
48
30
  this.component = component;
49
31
  this.addContextMenuInteraction();
50
- this.addOverlayOnInteraction();
51
32
  this.addScaleControl();
52
33
  this.addUpdateZoomAndCenter();
53
34
  this.component.map.addInteraction(this.selectionInteraction.init(this.component.store));
@@ -56,19 +37,6 @@ export class InteractionManager {
56
37
  addContextMenuInteraction() {
57
38
  this.component.map.getViewport().addEventListener('contextmenu', this.listeners.contextMenu);
58
39
  }
59
- addOverlayOnInteraction() {
60
- if (!this.interactionPopupOverlay) {
61
- this.interactionPopupOverlay = new Overlay({
62
- element: this.component.popup.nativeElement,
63
- offset: [9, 9]
64
- });
65
- this.component.map.addOverlay(this.interactionPopupOverlay);
66
- }
67
- if (this.component.optionsCopy.tooltipMode !== TooltipMode.NONE) {
68
- this.component.map.un('singleclick', this.listeners.popup);
69
- this.component.map.on('singleclick', this.listeners.popup);
70
- }
71
- }
72
40
  addScaleControl() {
73
41
  if (!this.scaleControl) {
74
42
  this.scaleControl = new ScaleLine({
@@ -100,4 +68,4 @@ export class InteractionManager {
100
68
  });
101
69
  }
102
70
  }
103
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"interaction-manager.class.js","sourceRoot":"","sources":["../../../../../../../projects/provoly/dashboard/widgets/widget-map/interaction/interaction-manager.class.ts"],"names":[],"mappings":"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","sourcesContent":["import { ContextMenuActions, TooltipMode } from '@provoly/dashboard';\nimport { Feature, MapBrowserEvent, Overlay } from 'ol';\nimport { ScaleLine } from 'ol/control';\nimport { Circle, Point } from 'ol/geom';\nimport { toLonLat } from 'ol/proj';\n// This is not a circular dependency\nimport type { WidgetMapComponent } from '../component/widget-map.component';\nimport { SelectionInteraction } from './selection-interaction.class';\nimport { FeatureLike } from 'ol/Feature';\n\nexport class InteractionManager {\n  interactionPopupOverlay?: Overlay;\n  scaleControl?: ScaleLine;\n  selectionInteraction = new SelectionInteraction();\n  public tooltipEvent?: MapBrowserEvent<MouseEvent>;\n  private component!: WidgetMapComponent;\n\n  init(component: WidgetMapComponent) {\n    this.component = component;\n    this.addContextMenuInteraction();\n    this.addOverlayOnInteraction();\n    this.addScaleControl();\n    this.addUpdateZoomAndCenter();\n    this.component.map.addInteraction(this.selectionInteraction.init(this.component.store));\n    return this;\n  }\n\n  listeners = {\n    contextMenu: (e: MouseEvent) => {\n      if (this.component.manifest.contextMenu !== false) {\n        const mapCoordinates = this.component.map.getEventPixel(e);\n        const itemId = this.component.map.forEachFeatureAtPixel(mapCoordinates, (feature: any) => feature.get('id'));\n        this.component.store.dispatch(\n          ContextMenuActions.open({\n            itemId,\n            clientX: e.clientX,\n            clientY: e.clientY,\n            resultSet: Array.isArray(this.component.manifest.datasource)\n              ? this.component.manifest.datasource[0]\n              : this.component.manifest.datasource,\n            from: 'map',\n            coordinates: toLonLat(\n              this.component.map.getCoordinateFromPixel(mapCoordinates),\n              this.component.map.getView().getProjection()\n            ).reverse(),\n            allowObjectCreation: false\n          })\n        );\n        e.preventDefault();\n      }\n    },\n    popup: (event: MapBrowserEvent<MouseEvent>) => {\n      const values = this.component.map.getFeaturesAtPixel(event.pixel, { hitTolerance: 5 });\n      const valuesToDisplay = [...new Set(values.map((value) => value.get('id')).filter((id) => !!id))]\n        .map((id) => values.find((value) => value.get('id') === id))\n        .map((value) => ({\n          oClass: value?.get('oClass'),\n          coordinates: event.coordinate,\n          values:\n            value instanceof Feature ? value.getKeys().reduce((p, c) => ({ ...p, [c]: value?.get(c) }), {}) : undefined\n        }))\n        .filter((value) => value.oClass);\n      this.component.classicFeatures$.next(valuesToDisplay);\n      const wmsLayers = this.component.getWMSLayers();\n      if (wmsLayers) {\n        this.component.getFeatureFromServer(wmsLayers, event.coordinate);\n      }\n      this.tooltipEvent = event;\n    }\n  };\n\n  private addContextMenuInteraction() {\n    this.component.map.getViewport().addEventListener('contextmenu', this.listeners.contextMenu);\n  }\n\n  private addOverlayOnInteraction() {\n    if (!this.interactionPopupOverlay) {\n      this.interactionPopupOverlay = new Overlay({\n        element: this.component.popup.nativeElement,\n        offset: [9, 9]\n      });\n      this.component.map.addOverlay(this.interactionPopupOverlay);\n    }\n    if (this.component.optionsCopy.tooltipMode !== TooltipMode.NONE) {\n      this.component.map.un('singleclick', this.listeners.popup);\n      this.component.map.on('singleclick', this.listeners.popup);\n    }\n  }\n\n  private addScaleControl() {\n    if (!this.scaleControl) {\n      this.scaleControl = new ScaleLine({\n        units: 'metric',\n        bar: true,\n        steps: 4,\n        text: true,\n        minWidth: 140,\n        dpi: 90\n      });\n      this.component.map.addControl(this.scaleControl);\n    }\n  }\n\n  private addUpdateZoomAndCenter() {\n    this.component.map.on('moveend', () => {\n      this.component.optionsCopy.zoom = {\n        start:\n          this.component.map.getView().getZoom() ??\n          this.component.optionsCopy.zoom?.start ??\n          this.component.map.getView().getMinZoom(),\n        min: this.component.map.getView().getMinZoom(),\n        max: this.component.map.getView().getMaxZoom()\n      };\n      this.component.optionsCopy.center =\n        toLonLat(this.component.map.getView().getCenter() ?? [0, 0]) ?? this.component.optionsCopy.center;\n      this.component.zoomAndCenter$.next({\n        zoom: this.component.optionsCopy.zoom,\n        center: this.component.optionsCopy.center\n      });\n    });\n  }\n}\n"]}
71
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"interaction-manager.class.js","sourceRoot":"","sources":["../../../../../../../projects/provoly/dashboard/widgets/widget-map/interaction/interaction-manager.class.ts"],"names":[],"mappings":"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","sourcesContent":["import { ContextMenuActions } from '@provoly/dashboard';\nimport { ScaleLine } from 'ol/control';\nimport { toLonLat } from 'ol/proj';\n// This is not a circular dependency\nimport type { WidgetMapComponent } from '../component/widget-map.component';\nimport { SelectionInteraction } from './selection-interaction.class';\n\nexport class InteractionManager {\n  scaleControl?: ScaleLine;\n  selectionInteraction = new SelectionInteraction();\n  private component!: WidgetMapComponent;\n\n  init(component: WidgetMapComponent) {\n    this.component = component;\n    this.addContextMenuInteraction();\n    this.addScaleControl();\n    this.addUpdateZoomAndCenter();\n    this.component.map.addInteraction(this.selectionInteraction.init(this.component.store));\n    return this;\n  }\n\n  listeners = {\n    contextMenu: (e: MouseEvent) => {\n      if (this.component.manifest.contextMenu !== false) {\n        const mapCoordinates = this.component.map.getEventPixel(e);\n        const itemId = this.component.map.forEachFeatureAtPixel(mapCoordinates, (feature: any) => feature.get('id'));\n        this.component.store.dispatch(\n          ContextMenuActions.open({\n            itemId,\n            clientX: e.clientX,\n            clientY: e.clientY,\n            resultSet: Array.isArray(this.component.manifest.datasource)\n              ? this.component.manifest.datasource[0]\n              : this.component.manifest.datasource,\n            from: 'map',\n            coordinates: toLonLat(\n              this.component.map.getCoordinateFromPixel(mapCoordinates),\n              this.component.map.getView().getProjection()\n            ).reverse(),\n            allowObjectCreation: false\n          })\n        );\n        e.preventDefault();\n      }\n    }\n  };\n\n  private addContextMenuInteraction() {\n    this.component.map.getViewport().addEventListener('contextmenu', this.listeners.contextMenu);\n  }\n\n  private addScaleControl() {\n    if (!this.scaleControl) {\n      this.scaleControl = new ScaleLine({\n        units: 'metric',\n        bar: true,\n        steps: 4,\n        text: true,\n        minWidth: 140,\n        dpi: 90\n      });\n      this.component.map.addControl(this.scaleControl);\n    }\n  }\n\n  private addUpdateZoomAndCenter() {\n    this.component.map.on('moveend', () => {\n      this.component.optionsCopy.zoom = {\n        start:\n          this.component.map.getView().getZoom() ??\n          this.component.optionsCopy.zoom?.start ??\n          this.component.map.getView().getMinZoom(),\n        min: this.component.map.getView().getMinZoom(),\n        max: this.component.map.getView().getMaxZoom()\n      };\n      this.component.optionsCopy.center =\n        toLonLat(this.component.map.getView().getCenter() ?? [0, 0]) ?? this.component.optionsCopy.center;\n      this.component.zoomAndCenter$.next({\n        zoom: this.component.optionsCopy.zoom,\n        center: this.component.optionsCopy.center\n      });\n    });\n  }\n}\n"]}
@@ -0,0 +1,191 @@
1
+ import { DashboardActions, DashboardSelectors, TooltipMode, WIDGET_HEADER_HEIGHT } from '@provoly/dashboard';
2
+ import { BehaviorSubject, combineLatest, debounceTime, withLatestFrom } from 'rxjs';
3
+ import { Feature, Overlay } from 'ol';
4
+ import { TOOLTIP_PADDING } from '../component/widget-map.component';
5
+ export class TooltipManager {
6
+ constructor() {
7
+ this.classicFeatures$ = new BehaviorSubject([]);
8
+ this.tooltipIndex = 0;
9
+ this.tooltipNumber = 0;
10
+ }
11
+ init(tooltipFactoryService, injector, store, subscriptions, widgetSize$, displayHeader$, map, popup, popupContent, interactionManager, parent) {
12
+ this.tooltipFactoryService = tooltipFactoryService;
13
+ this.injector = injector;
14
+ this.store = store;
15
+ this.parent = parent;
16
+ this.map = map;
17
+ this.popupContent = popupContent;
18
+ this.popup = popup;
19
+ this.interactionManager = interactionManager;
20
+ this.wmsFeatures$ = this.store.select(DashboardSelectors.wmsFeatures(this.parent.id));
21
+ subscriptions.add(combineLatest([this.wmsFeatures$, this.classicFeatures$])
22
+ .pipe(debounceTime(50), withLatestFrom(widgetSize$, displayHeader$))
23
+ .subscribe(([[wmsFeatures, classicFeatures], size, header]) => {
24
+ if (map) {
25
+ this.tooltipIndex = 0;
26
+ if (this.popupOverlay !== null) {
27
+ this.clearTooltip();
28
+ }
29
+ this.popupOverlay = new Overlay({
30
+ element: popup.nativeElement,
31
+ offset: [9, 9]
32
+ });
33
+ this.map.addOverlay(this.popupOverlay);
34
+ this.popupContent.clear();
35
+ const componentPromises = [];
36
+ wmsFeatures.forEach((value) => {
37
+ componentPromises.push(this.generateTooltip(value));
38
+ });
39
+ classicFeatures.forEach((value) => {
40
+ componentPromises.push(this.generateTooltip(value.values));
41
+ });
42
+ Promise.all(componentPromises).then((promises) => {
43
+ const realComponents = promises.filter((component) => !!component);
44
+ this.tooltipNumber = realComponents.length;
45
+ if (realComponents.length <= 0) {
46
+ popup.nativeElement.hidden = true;
47
+ this.popupOverlay?.setPosition([-9999999999, -9999999999]);
48
+ }
49
+ else {
50
+ popup.nativeElement.hidden = false;
51
+ // Do not go outside the map
52
+ const rect = {
53
+ height: 200 + (header ? WIDGET_HEADER_HEIGHT : 0) + TOOLTIP_PADDING,
54
+ width: 300 + TOOLTIP_PADDING
55
+ };
56
+ let pixel = this.tooltipEvent?.pixel;
57
+ if (!!pixel && pixel[0] + rect.width > size.width) {
58
+ pixel[0] = size.width - rect.width;
59
+ }
60
+ if (!!pixel && pixel[1] + rect.height > size.height) {
61
+ pixel[1] = size.height - rect.height;
62
+ }
63
+ setTimeout(() => {
64
+ this.tooltipMove(0);
65
+ }, 10);
66
+ this.popupOverlay?.setPosition(this.map.getCoordinateFromPixel(pixel ?? [0, 0]));
67
+ }
68
+ });
69
+ }
70
+ }));
71
+ window.addEventListener('forceCloseMapTooltip', () => {
72
+ this.clearTooltip();
73
+ });
74
+ this.addOverlayOnInteraction();
75
+ return this;
76
+ }
77
+ addOverlayOnInteraction() {
78
+ if (!this.interactionPopupOverlay) {
79
+ this.interactionPopupOverlay = new Overlay({
80
+ element: this.popup.nativeElement,
81
+ offset: [9, 9]
82
+ });
83
+ this.map.addOverlay(this.interactionPopupOverlay);
84
+ }
85
+ const popupListener = (event) => {
86
+ const values = this.map.getFeaturesAtPixel(event.pixel, { hitTolerance: 5 });
87
+ const valuesToDisplay = [...new Set(values.map((value) => value.get('id')).filter((id) => !!id))]
88
+ .map((id) => values.find((value) => value.get('id') === id))
89
+ .map((value) => ({
90
+ oClass: value?.get('oClass'),
91
+ coordinates: event.coordinate,
92
+ values: value instanceof Feature ? value.getKeys().reduce((p, c) => ({ ...p, [c]: value?.get(c) }), {}) : undefined
93
+ }))
94
+ .filter((value) => value.oClass);
95
+ this.classicFeatures$.next(valuesToDisplay);
96
+ const wmsLayers = this.parent.getWMSLayers();
97
+ if (wmsLayers) {
98
+ this.getFeatureFromServer(wmsLayers, event.coordinate);
99
+ }
100
+ this.tooltipEvent = event;
101
+ };
102
+ if (this.parent.optionsCopy.tooltipMode !== TooltipMode.NONE) {
103
+ this.parent.map.un('singleclick', popupListener);
104
+ this.parent.map.on('singleclick', popupListener);
105
+ }
106
+ }
107
+ generateTooltip(value) {
108
+ return new Promise((resolve) => {
109
+ const component$ = this.tooltipFactoryService.generateTooltip(value.oClass, this.injector, this.popupContent);
110
+ this.previousTooltipSub?.unsubscribe();
111
+ this.previousTooltipSub = component$.subscribe((component) => {
112
+ if (component) {
113
+ if (this.previousTooltip) {
114
+ this.previousTooltip.destroy();
115
+ }
116
+ component.instance.data.item = value;
117
+ component.instance.popup = this.popup;
118
+ resolve(component);
119
+ }
120
+ resolve(null);
121
+ });
122
+ });
123
+ }
124
+ clearTooltip(empty = false) {
125
+ if (this.popupOverlay)
126
+ this.map?.removeOverlay(this.popupOverlay);
127
+ this.popupContent?.clear();
128
+ this.previousTooltip?.destroy();
129
+ this.previousTooltipSub?.unsubscribe();
130
+ if (empty) {
131
+ this.store.dispatch(DashboardActions.resetWmsFeatures({ componentId: this.parent.id }));
132
+ this.classicFeatures$.next([]);
133
+ }
134
+ }
135
+ tooltipMove(number) {
136
+ if (this.tooltipCanMove(number)) {
137
+ this.setStyleAsync(this.tooltipIndex, 'none');
138
+ this.tooltipIndex = this.tooltipIndex + number;
139
+ this.setStyleAsync(this.tooltipIndex, 'block');
140
+ }
141
+ }
142
+ setStyleAsync(idx, value) {
143
+ setTimeout(() => {
144
+ const targetContainer = this.popupContent.get(idx);
145
+ if (targetContainer) {
146
+ // @ts-ignore
147
+ targetContainer.rootNodes[0].style.display = value;
148
+ }
149
+ }, 100);
150
+ }
151
+ tooltipCanMove(number) {
152
+ return this.tooltipIndex + number >= 0 && this.tooltipIndex + number < this.tooltipNumber;
153
+ }
154
+ getFeatureFromServer(wmsLayers, pixel) {
155
+ for (const layer of wmsLayers) {
156
+ let url = layer
157
+ .getSource()
158
+ .getFeatureInfoUrl(pixel, this.map.getView().getResolution(), this.map.getView().getProjection(), {
159
+ INFO_FORMAT: 'application/json',
160
+ FEATURE_COUNT: 10,
161
+ ...layer.getProperties()['getFeatureInfoAdditionalParameters']
162
+ });
163
+ if (this.parent.optionsCopy.layers?.find((l) => {
164
+ return l.title === layer.getProperties()['title'];
165
+ })?.getWfsFeaturesForTooltips) {
166
+ this.store.dispatch(DashboardActions.getWfsFeaturesForPointStackTooltips({
167
+ url,
168
+ oClass: layer.getProperties()['oClass'] ?? '',
169
+ coordinates: pixel,
170
+ componentId: this.parent.id
171
+ }));
172
+ }
173
+ else {
174
+ if (url && url.indexOf('I=-1') === -1 && url.indexOf('J=-1') === -1) {
175
+ this.store.dispatch(DashboardActions.getWmsFeatures({
176
+ url,
177
+ oClass: layer.getProperties()['oClass'] ?? '',
178
+ coordinates: pixel,
179
+ componentId: this.parent.id
180
+ }));
181
+ }
182
+ }
183
+ }
184
+ }
185
+ destroy() {
186
+ this.store.dispatch(DashboardActions.resetWmsFeatures({ componentId: this.parent.id }));
187
+ this.previousTooltipSub?.unsubscribe();
188
+ this.previousTooltip?.destroy();
189
+ }
190
+ }
191
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tooltip-manager.class.js","sourceRoot":"","sources":["../../../../../../../projects/provoly/dashboard/widgets/widget-map/interaction/tooltip-manager.class.ts"],"names":[],"mappings":"AAEA,OAAO,EAEL,gBAAgB,EAChB,kBAAkB,EAIlB,WAAW,EACX,oBAAoB,EACrB,MAAM,oBAAoB,CAAC;AAC5B,OAAO,EAAE,eAAe,EAAE,aAAa,EAAE,YAAY,EAA4B,cAAc,EAAE,MAAM,MAAM,CAAC;AAC9G,OAAO,EAAE,OAAO,EAAwB,OAAO,EAAE,MAAM,IAAI,CAAC;AAE5D,OAAO,EAAE,eAAe,EAAsB,MAAM,mCAAmC,CAAC;AAIxF,MAAM,OAAO,cAAc;IAA3B;QAKE,qBAAgB,GAAyB,IAAI,eAAe,CAAC,EAAE,CAAC,CAAC;QAC1D,iBAAY,GAAG,CAAC,CAAC;QAKjB,kBAAa,GAAG,CAAC,CAAC;IAkO3B,CAAC;IA1NC,IAAI,CACF,qBAA4C,EAC5C,QAAkB,EAClB,KAAY,EACZ,aAA2B,EAC3B,WAA6B,EAC7B,cAA8D,EAC9D,GAAQ,EACR,KAAiB,EACjB,YAA8B,EAC9B,kBAAsC,EACtC,MAA0B;QAE1B,IAAI,CAAC,qBAAqB,GAAG,qBAAqB,CAAC;QACnD,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAC;QACzB,IAAI,CAAC,KAAK,GAAG,KAAK,CAAC;QACnB,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC;QACrB,IAAI,CAAC,GAAG,GAAG,GAAG,CAAC;QACf,IAAI,CAAC,YAAY,GAAG,YAAY,CAAC;QACjC,IAAI,CAAC,KAAK,GAAG,KAAK,CAAC;QACnB,IAAI,CAAC,kBAAkB,GAAG,kBAAkB,CAAC;QAC7C,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,kBAAkB,CAAC,WAAW,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;QAEtF,aAAa,CAAC,GAAG,CACf,aAAa,CAAC,CAAC,IAAI,CAAC,YAAY,EAAE,IAAI,CAAC,gBAAgB,CAAC,CAAC;aACtD,IAAI,CAAC,YAAY,CAAC,EAAE,CAAC,EAAE,cAAc,CAAC,WAAW,EAAE,cAAc,CAAC,CAAC;aACnE,SAAS,CAAC,CAAC,CAAC,CAAC,WAAW,EAAE,eAAe,CAAC,EAAE,IAAI,EAAE,MAAM,CAAC,EAAE,EAAE;YAC5D,IAAI,GAAG,EAAE,CAAC;gBACR,IAAI,CAAC,YAAY,GAAG,CAAC,CAAC;gBACtB,IAAI,IAAI,CAAC,YAAY,KAAK,IAAI,EAAE,CAAC;oBAC/B,IAAI,CAAC,YAAY,EAAE,CAAC;gBACtB,CAAC;gBACD,IAAI,CAAC,YAAY,GAAG,IAAI,OAAO,CAAC;oBAC9B,OAAO,EAAE,KAAK,CAAC,aAAa;oBAC5B,MAAM,EAAE,CAAC,CAAC,EAAE,CAAC,CAAC;iBACf,CAAC,CAAC;gBACH,IAAI,CAAC,GAAG,CAAC,UAAU,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;gBACvC,IAAI,CAAC,YAAY,CAAC,KAAK,EAAE,CAAC;gBAC1B,MAAM,iBAAiB,GAAG,EAA0D,CAAC;gBACrF,WAAW,CAAC,OAAO,CAAC,CAAC,KAAK,EAAE,EAAE;oBAC5B,iBAAiB,CAAC,IAAI,CAAC,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC,CAAC;gBACtD,CAAC,CAAC,CAAC;gBACH,eAAe,CAAC,OAAO,CAAC,CAAC,KAAU,EAAE,EAAE;oBACrC,iBAAiB,CAAC,IAAI,CAAC,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC;gBAC7D,CAAC,CAAC,CAAC;gBAEH,OAAO,CAAC,GAAG,CAAC,iBAAiB,CAAC,CAAC,IAAI,CAAC,CAAC,QAAQ,EAAE,EAAE;oBAC/C,MAAM,cAAc,GAAG,QAAQ,CAAC,MAAM,CAAC,CAAC,SAAS,EAAE,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC;oBACnE,IAAI,CAAC,aAAa,GAAG,cAAc,CAAC,MAAM,CAAC;oBAC3C,IAAI,cAAc,CAAC,MAAM,IAAI,CAAC,EAAE,CAAC;wBAC/B,KAAK,CAAC,aAAa,CAAC,MAAM,GAAG,IAAI,CAAC;wBAClC,IAAI,CAAC,YAAY,EAAE,WAAW,CAAC,CAAC,CAAC,UAAU,EAAE,CAAC,UAAU,CAAC,CAAC,CAAC;oBAC7D,CAAC;yBAAM,CAAC;wBACN,KAAK,CAAC,aAAa,CAAC,MAAM,GAAG,KAAK,CAAC;wBACnC,4BAA4B;wBAC5B,MAAM,IAAI,GAAG;4BACX,MAAM,EAAE,GAAG,GAAG,CAAC,MAAM,CAAC,CAAC,CAAC,oBAAoB,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,eAAe;4BACnE,KAAK,EAAE,GAAG,GAAG,eAAe;yBAC7B,CAAC;wBACF,IAAI,KAAK,GAAG,IAAI,CAAC,YAAY,EAAE,KAAK,CAAC;wBACrC,IAAI,CAAC,CAAC,KAAK,IAAI,KAAK,CAAC,CAAC,CAAC,GAAG,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,KAAK,EAAE,CAAC;4BAClD,KAAK,CAAC,CAAC,CAAC,GAAG,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC;wBACrC,CAAC;wBACD,IAAI,CAAC,CAAC,KAAK,IAAI,KAAK,CAAC,CAAC,CAAC,GAAG,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,MAAM,EAAE,CAAC;4BACpD,KAAK,CAAC,CAAC,CAAC,GAAG,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC;wBACvC,CAAC;wBACD,UAAU,CAAC,GAAG,EAAE;4BACd,IAAI,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC;wBACtB,CAAC,EAAE,EAAE,CAAC,CAAC;wBACP,IAAI,CAAC,YAAY,EAAE,WAAW,CAAC,IAAI,CAAC,GAAG,CAAC,sBAAsB,CAAC,KAAK,IAAI,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;oBACnF,CAAC;gBACH,CAAC,CAAC,CAAC;YACL,CAAC;QACH,CAAC,CAAC,CACL,CAAC;QAEF,MAAM,CAAC,gBAAgB,CAAC,sBAAsB,EAAE,GAAG,EAAE;YACnD,IAAI,CAAC,YAAY,EAAE,CAAC;QACtB,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,uBAAuB,EAAE,CAAC;QAE/B,OAAO,IAAI,CAAC;IACd,CAAC;IAEO,uBAAuB;QAC7B,IAAI,CAAC,IAAI,CAAC,uBAAuB,EAAE,CAAC;YAClC,IAAI,CAAC,uBAAuB,GAAG,IAAI,OAAO,CAAC;gBACzC,OAAO,EAAE,IAAI,CAAC,KAAK,CAAC,aAAa;gBACjC,MAAM,EAAE,CAAC,CAAC,EAAE,CAAC,CAAC;aACf,CAAC,CAAC;YACH,IAAI,CAAC,GAAG,CAAC,UAAU,CAAC,IAAI,CAAC,uBAAuB,CAAC,CAAC;QACpD,CAAC;QAED,MAAM,aAAa,GAAG,CAAC,KAAkC,EAAE,EAAE;YAC3D,MAAM,MAAM,GAAG,IAAI,CAAC,GAAG,CAAC,kBAAkB,CAAC,KAAK,CAAC,KAAK,EAAE,EAAE,YAAY,EAAE,CAAC,EAAE,CAAC,CAAC;YAC7E,MAAM,eAAe,GAAG,CAAC,GAAG,IAAI,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,EAAE,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;iBAC9F,GAAG,CAAC,CAAC,EAAE,EAAE,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC;iBAC3D,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;gBACf,MAAM,EAAE,KAAK,EAAE,GAAG,CAAC,QAAQ,CAAC;gBAC5B,WAAW,EAAE,KAAK,CAAC,UAAU;gBAC7B,MAAM,EACJ,KAAK,YAAY,OAAO,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,EAAE,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,EAAE,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,SAAS;aAC9G,CAAC,CAAC;iBACF,MAAM,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;YACnC,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;YAC5C,MAAM,SAAS,GAAG,IAAI,CAAC,MAAM,CAAC,YAAY,EAAE,CAAC;YAC7C,IAAI,SAAS,EAAE,CAAC;gBACd,IAAI,CAAC,oBAAoB,CAAC,SAAS,EAAE,KAAK,CAAC,UAAU,CAAC,CAAC;YACzD,CAAC;YACD,IAAI,CAAC,YAAY,GAAG,KAAK,CAAC;QAC5B,CAAC,CAAC;QAEF,IAAI,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,WAAW,KAAK,WAAW,CAAC,IAAI,EAAE,CAAC;YAC7D,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,aAAa,EAAE,aAAa,CAAC,CAAC;YACjD,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,aAAa,EAAE,aAAa,CAAC,CAAC;QACnD,CAAC;IACH,CAAC;IAED,eAAe,CAAC,KAIf;QACC,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE;YAC7B,MAAM,UAAU,GAAG,IAAI,CAAC,qBAAqB,CAAC,eAAe,CAAC,KAAK,CAAC,MAAM,EAAE,IAAI,CAAC,QAAQ,EAAE,IAAI,CAAC,YAAY,CAAC,CAAC;YAC9G,IAAI,CAAC,kBAAkB,EAAE,WAAW,EAAE,CAAC;YACvC,IAAI,CAAC,kBAAkB,GAAG,UAAU,CAAC,SAAS,CAAC,CAAC,SAAS,EAAE,EAAE;gBAC3D,IAAI,SAAS,EAAE,CAAC;oBACd,IAAI,IAAI,CAAC,eAAe,EAAE,CAAC;wBACzB,IAAI,CAAC,eAAe,CAAC,OAAO,EAAE,CAAC;oBACjC,CAAC;oBACD,SAAS,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,GAAG,KAAK,CAAC;oBACrC,SAAS,CAAC,QAAQ,CAAC,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC;oBACtC,OAAO,CAAC,SAAS,CAAC,CAAC;gBACrB,CAAC;gBACD,OAAO,CAAC,IAAI,CAAC,CAAC;YAChB,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC;IAEM,YAAY,CAAC,KAAK,GAAG,KAAK;QAC/B,IAAI,IAAI,CAAC,YAAY;YAAE,IAAI,CAAC,GAAG,EAAE,aAAa,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;QAClE,IAAI,CAAC,YAAY,EAAE,KAAK,EAAE,CAAC;QAC3B,IAAI,CAAC,eAAe,EAAE,OAAO,EAAE,CAAC;QAChC,IAAI,CAAC,kBAAkB,EAAE,WAAW,EAAE,CAAC;QACvC,IAAI,KAAK,EAAE,CAAC;YACV,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,gBAAgB,CAAC,gBAAgB,CAAC,EAAE,WAAW,EAAE,IAAI,CAAC,MAAM,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC;YACxF,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACjC,CAAC;IACH,CAAC;IAEM,WAAW,CAAC,MAAc;QAC/B,IAAI,IAAI,CAAC,cAAc,CAAC,MAAM,CAAC,EAAE,CAAC;YAChC,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,YAAY,EAAE,MAAM,CAAC,CAAC;YAC9C,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,YAAY,GAAG,MAAM,CAAC;YAC/C,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,YAAY,EAAE,OAAO,CAAC,CAAC;QACjD,CAAC;IACH,CAAC;IAEO,aAAa,CAAC,GAAW,EAAE,KAAa;QAC9C,UAAU,CAAC,GAAG,EAAE;YACd,MAAM,eAAe,GAAG,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;YACnD,IAAI,eAAe,EAAE,CAAC;gBACpB,aAAa;gBACb,eAAe,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,GAAG,KAAK,CAAC;YACrD,CAAC;QACH,CAAC,EAAE,GAAG,CAAC,CAAC;IACV,CAAC;IAEM,cAAc,CAAC,MAAc;QAClC,OAAO,IAAI,CAAC,YAAY,GAAG,MAAM,IAAI,CAAC,IAAI,IAAI,CAAC,YAAY,GAAG,MAAM,GAAG,IAAI,CAAC,aAAa,CAAC;IAC5F,CAAC;IAED,oBAAoB,CAAC,SAAuB,EAAE,KAAY;QACxD,KAAK,MAAM,KAAK,IAAI,SAAS,EAAE,CAAC;YAC9B,IAAI,GAAG,GAAG,KAAK;iBACZ,SAAS,EAAE;iBACX,iBAAiB,CAAC,KAAK,EAAE,IAAI,CAAC,GAAG,CAAC,OAAO,EAAE,CAAC,aAAa,EAAE,EAAE,IAAI,CAAC,GAAG,CAAC,OAAO,EAAE,CAAC,aAAa,EAAE,EAAE;gBAChG,WAAW,EAAE,kBAAkB;gBAC/B,aAAa,EAAE,EAAE;gBACjB,GAAG,KAAK,CAAC,aAAa,EAAE,CAAC,oCAAoC,CAAC;aAC/D,CAAC,CAAC;YACL,IAEI,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,MAAM,EAAE,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE;gBACzC,OAAO,CAAC,CAAC,KAAK,KAAK,KAAK,CAAC,aAAa,EAAE,CAAC,OAAO,CAAC,CAAC;YACpD,CAAC,CACF,EAAE,yBAAyB,EAC5B,CAAC;gBACD,IAAI,CAAC,KAAK,CAAC,QAAQ,CACjB,gBAAgB,CAAC,mCAAmC,CAAC;oBACnD,GAAG;oBACH,MAAM,EAAE,KAAK,CAAC,aAAa,EAAE,CAAC,QAAQ,CAAC,IAAI,EAAE;oBAC7C,WAAW,EAAE,KAAK;oBAClB,WAAW,EAAE,IAAI,CAAC,MAAM,CAAC,EAAE;iBAC5B,CAAC,CACH,CAAC;YACJ,CAAC;iBAAM,CAAC;gBACN,IAAI,GAAG,IAAI,GAAG,CAAC,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,IAAI,GAAG,CAAC,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC;oBACpE,IAAI,CAAC,KAAK,CAAC,QAAQ,CACjB,gBAAgB,CAAC,cAAc,CAAC;wBAC9B,GAAG;wBACH,MAAM,EAAE,KAAK,CAAC,aAAa,EAAE,CAAC,QAAQ,CAAC,IAAI,EAAE;wBAC7C,WAAW,EAAE,KAAK;wBAClB,WAAW,EAAE,IAAI,CAAC,MAAM,CAAC,EAAE;qBAC5B,CAAC,CACH,CAAC;gBACJ,CAAC;YACH,CAAC;QACH,CAAC;IACH,CAAC;IAED,OAAO;QACL,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,gBAAgB,CAAC,gBAAgB,CAAC,EAAE,WAAW,EAAE,IAAI,CAAC,MAAM,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC;QACxF,IAAI,CAAC,kBAAkB,EAAE,WAAW,EAAE,CAAC;QACvC,IAAI,CAAC,eAAe,EAAE,OAAO,EAAE,CAAC;IAClC,CAAC;CACF","sourcesContent":["import { Pixel } from 'ol/pixel';\nimport { ComponentRef, ElementRef, Injector, ViewContainerRef } from '@angular/core';\nimport {\n  BaseTooltipComponent,\n  DashboardActions,\n  DashboardSelectors,\n  MapGeoServerLayerOptions,\n  Size,\n  TooltipFactoryService,\n  TooltipMode,\n  WIDGET_HEADER_HEIGHT\n} from '@provoly/dashboard';\nimport { BehaviorSubject, combineLatest, debounceTime, Observable, Subscription, withLatestFrom } from 'rxjs';\nimport { Feature, Map, MapBrowserEvent, Overlay } from 'ol';\nimport { Store } from '@ngrx/store';\nimport { TOOLTIP_PADDING, WidgetMapComponent } from '../component/widget-map.component';\nimport { InteractionManager } from './interaction-manager.class';\nimport { Layer } from 'ol/layer';\n\nexport class TooltipManager {\n  private tooltipFactoryService!: TooltipFactoryService;\n  private injector!: Injector;\n  private store!: Store;\n  wmsFeatures$!: Observable<any[]>;\n  classicFeatures$: BehaviorSubject<any> = new BehaviorSubject([]);\n  public tooltipIndex = 0;\n  popupOverlay?: Overlay;\n  map!: Map;\n  popup!: ElementRef;\n  popupContent!: ViewContainerRef;\n  public tooltipNumber = 0;\n  interactionManager!: InteractionManager;\n  previousTooltip?: ComponentRef<BaseTooltipComponent>;\n  previousTooltipSub?: Subscription;\n  public tooltipEvent?: MapBrowserEvent<MouseEvent>;\n  parent!: WidgetMapComponent;\n  interactionPopupOverlay!: Overlay;\n\n  init(\n    tooltipFactoryService: TooltipFactoryService,\n    injector: Injector,\n    store: Store,\n    subscriptions: Subscription,\n    widgetSize$: Observable<Size>,\n    displayHeader$: Observable<boolean | { [p: string]: boolean }>,\n    map: Map,\n    popup: ElementRef,\n    popupContent: ViewContainerRef,\n    interactionManager: InteractionManager,\n    parent: WidgetMapComponent\n  ) {\n    this.tooltipFactoryService = tooltipFactoryService;\n    this.injector = injector;\n    this.store = store;\n    this.parent = parent;\n    this.map = map;\n    this.popupContent = popupContent;\n    this.popup = popup;\n    this.interactionManager = interactionManager;\n    this.wmsFeatures$ = this.store.select(DashboardSelectors.wmsFeatures(this.parent.id));\n\n    subscriptions.add(\n      combineLatest([this.wmsFeatures$, this.classicFeatures$])\n        .pipe(debounceTime(50), withLatestFrom(widgetSize$, displayHeader$))\n        .subscribe(([[wmsFeatures, classicFeatures], size, header]) => {\n          if (map) {\n            this.tooltipIndex = 0;\n            if (this.popupOverlay !== null) {\n              this.clearTooltip();\n            }\n            this.popupOverlay = new Overlay({\n              element: popup.nativeElement,\n              offset: [9, 9]\n            });\n            this.map.addOverlay(this.popupOverlay);\n            this.popupContent.clear();\n            const componentPromises = [] as Promise<ComponentRef<BaseTooltipComponent> | null>[];\n            wmsFeatures.forEach((value) => {\n              componentPromises.push(this.generateTooltip(value));\n            });\n            classicFeatures.forEach((value: any) => {\n              componentPromises.push(this.generateTooltip(value.values));\n            });\n\n            Promise.all(componentPromises).then((promises) => {\n              const realComponents = promises.filter((component) => !!component);\n              this.tooltipNumber = realComponents.length;\n              if (realComponents.length <= 0) {\n                popup.nativeElement.hidden = true;\n                this.popupOverlay?.setPosition([-9999999999, -9999999999]);\n              } else {\n                popup.nativeElement.hidden = false;\n                // Do not go outside the map\n                const rect = {\n                  height: 200 + (header ? WIDGET_HEADER_HEIGHT : 0) + TOOLTIP_PADDING,\n                  width: 300 + TOOLTIP_PADDING\n                };\n                let pixel = this.tooltipEvent?.pixel;\n                if (!!pixel && pixel[0] + rect.width > size.width) {\n                  pixel[0] = size.width - rect.width;\n                }\n                if (!!pixel && pixel[1] + rect.height > size.height) {\n                  pixel[1] = size.height - rect.height;\n                }\n                setTimeout(() => {\n                  this.tooltipMove(0);\n                }, 10);\n                this.popupOverlay?.setPosition(this.map.getCoordinateFromPixel(pixel ?? [0, 0]));\n              }\n            });\n          }\n        })\n    );\n\n    window.addEventListener('forceCloseMapTooltip', () => {\n      this.clearTooltip();\n    });\n\n    this.addOverlayOnInteraction();\n\n    return this;\n  }\n\n  private addOverlayOnInteraction() {\n    if (!this.interactionPopupOverlay) {\n      this.interactionPopupOverlay = new Overlay({\n        element: this.popup.nativeElement,\n        offset: [9, 9]\n      });\n      this.map.addOverlay(this.interactionPopupOverlay);\n    }\n\n    const popupListener = (event: MapBrowserEvent<MouseEvent>) => {\n      const values = this.map.getFeaturesAtPixel(event.pixel, { hitTolerance: 5 });\n      const valuesToDisplay = [...new Set(values.map((value) => value.get('id')).filter((id) => !!id))]\n        .map((id) => values.find((value) => value.get('id') === id))\n        .map((value) => ({\n          oClass: value?.get('oClass'),\n          coordinates: event.coordinate,\n          values:\n            value instanceof Feature ? value.getKeys().reduce((p, c) => ({ ...p, [c]: value?.get(c) }), {}) : undefined\n        }))\n        .filter((value) => value.oClass);\n      this.classicFeatures$.next(valuesToDisplay);\n      const wmsLayers = this.parent.getWMSLayers();\n      if (wmsLayers) {\n        this.getFeatureFromServer(wmsLayers, event.coordinate);\n      }\n      this.tooltipEvent = event;\n    };\n\n    if (this.parent.optionsCopy.tooltipMode !== TooltipMode.NONE) {\n      this.parent.map.un('singleclick', popupListener);\n      this.parent.map.on('singleclick', popupListener);\n    }\n  }\n\n  generateTooltip(value: {\n    oClass: string;\n    coordinates: Pixel;\n    values: any;\n  }): Promise<ComponentRef<BaseTooltipComponent> | null> {\n    return new Promise((resolve) => {\n      const component$ = this.tooltipFactoryService.generateTooltip(value.oClass, this.injector, this.popupContent);\n      this.previousTooltipSub?.unsubscribe();\n      this.previousTooltipSub = component$.subscribe((component) => {\n        if (component) {\n          if (this.previousTooltip) {\n            this.previousTooltip.destroy();\n          }\n          component.instance.data.item = value;\n          component.instance.popup = this.popup;\n          resolve(component);\n        }\n        resolve(null);\n      });\n    });\n  }\n\n  public clearTooltip(empty = false) {\n    if (this.popupOverlay) this.map?.removeOverlay(this.popupOverlay);\n    this.popupContent?.clear();\n    this.previousTooltip?.destroy();\n    this.previousTooltipSub?.unsubscribe();\n    if (empty) {\n      this.store.dispatch(DashboardActions.resetWmsFeatures({ componentId: this.parent.id }));\n      this.classicFeatures$.next([]);\n    }\n  }\n\n  public tooltipMove(number: number) {\n    if (this.tooltipCanMove(number)) {\n      this.setStyleAsync(this.tooltipIndex, 'none');\n      this.tooltipIndex = this.tooltipIndex + number;\n      this.setStyleAsync(this.tooltipIndex, 'block');\n    }\n  }\n\n  private setStyleAsync(idx: number, value: string) {\n    setTimeout(() => {\n      const targetContainer = this.popupContent.get(idx);\n      if (targetContainer) {\n        // @ts-ignore\n        targetContainer.rootNodes[0].style.display = value;\n      }\n    }, 100);\n  }\n\n  public tooltipCanMove(number: number) {\n    return this.tooltipIndex + number >= 0 && this.tooltipIndex + number < this.tooltipNumber;\n  }\n\n  getFeatureFromServer(wmsLayers: Layer<any>[], pixel: Pixel) {\n    for (const layer of wmsLayers) {\n      let url = layer\n        .getSource()\n        .getFeatureInfoUrl(pixel, this.map.getView().getResolution(), this.map.getView().getProjection(), {\n          INFO_FORMAT: 'application/json',\n          FEATURE_COUNT: 10,\n          ...layer.getProperties()['getFeatureInfoAdditionalParameters']\n        });\n      if (\n        (\n          this.parent.optionsCopy.layers?.find((l) => {\n            return l.title === layer.getProperties()['title'];\n          }) as MapGeoServerLayerOptions\n        )?.getWfsFeaturesForTooltips\n      ) {\n        this.store.dispatch(\n          DashboardActions.getWfsFeaturesForPointStackTooltips({\n            url,\n            oClass: layer.getProperties()['oClass'] ?? '',\n            coordinates: pixel,\n            componentId: this.parent.id\n          })\n        );\n      } else {\n        if (url && url.indexOf('I=-1') === -1 && url.indexOf('J=-1') === -1) {\n          this.store.dispatch(\n            DashboardActions.getWmsFeatures({\n              url,\n              oClass: layer.getProperties()['oClass'] ?? '',\n              coordinates: pixel,\n              componentId: this.parent.id\n            })\n          );\n        }\n      }\n    }\n  }\n\n  destroy() {\n    this.store.dispatch(DashboardActions.resetWmsFeatures({ componentId: this.parent.id }));\n    this.previousTooltipSub?.unsubscribe();\n    this.previousTooltip?.destroy();\n  }\n}\n"]}
@@ -6,8 +6,9 @@ export * from './pipe/widget-map-legend-url.pipe';
6
6
  export * from './interaction/interaction-manager.class';
7
7
  export * from './interaction/selection-interaction.class';
8
8
  export * from './interaction/layer-slider.class';
9
+ export * from './interaction/tooltip-manager.class';
9
10
  export * from './utils/widget-map.utils';
10
11
  export * from './utils/xml-utils.class';
11
12
  export * from './utils/cql-utils.class';
12
13
  export * from './style/css.component';
13
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3Byb3ZvbHkvZGFzaGJvYXJkL3dpZGdldHMvd2lkZ2V0LW1hcC9wdWJsaWMtYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMscUJBQXFCLENBQUM7QUFDcEMsY0FBYyxrQ0FBa0MsQ0FBQztBQUNqRCxjQUFjLHNDQUFzQyxDQUFDO0FBQ3JELGNBQWMsNENBQTRDLENBQUM7QUFDM0QsY0FBYyxtQ0FBbUMsQ0FBQztBQUNsRCxjQUFjLHlDQUF5QyxDQUFDO0FBQ3hELGNBQWMsMkNBQTJDLENBQUM7QUFDMUQsY0FBYyxrQ0FBa0MsQ0FBQztBQUNqRCxjQUFjLDBCQUEwQixDQUFDO0FBQ3pDLGNBQWMseUJBQXlCLENBQUM7QUFDeEMsY0FBYyx5QkFBeUIsQ0FBQztBQUN4QyxjQUFjLHVCQUF1QixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi93aWRnZXQtbWFwLm1vZHVsZSc7XG5leHBvcnQgKiBmcm9tICcuL2NvbXBvbmVudC93aWRnZXQtbWFwLmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL2NvbXBvbmVudC93aWRnZXQtbWFwLWxheWVyLnNlcnZpY2UnO1xuZXhwb3J0ICogZnJvbSAnLi9waXBlL3dpZGdldC1tYXAtZ2VvbWV0cnktZmllbGRzLWZvci5waXBlJztcbmV4cG9ydCAqIGZyb20gJy4vcGlwZS93aWRnZXQtbWFwLWxlZ2VuZC11cmwucGlwZSc7XG5leHBvcnQgKiBmcm9tICcuL2ludGVyYWN0aW9uL2ludGVyYWN0aW9uLW1hbmFnZXIuY2xhc3MnO1xuZXhwb3J0ICogZnJvbSAnLi9pbnRlcmFjdGlvbi9zZWxlY3Rpb24taW50ZXJhY3Rpb24uY2xhc3MnO1xuZXhwb3J0ICogZnJvbSAnLi9pbnRlcmFjdGlvbi9sYXllci1zbGlkZXIuY2xhc3MnO1xuZXhwb3J0ICogZnJvbSAnLi91dGlscy93aWRnZXQtbWFwLnV0aWxzJztcbmV4cG9ydCAqIGZyb20gJy4vdXRpbHMveG1sLXV0aWxzLmNsYXNzJztcbmV4cG9ydCAqIGZyb20gJy4vdXRpbHMvY3FsLXV0aWxzLmNsYXNzJztcbmV4cG9ydCAqIGZyb20gJy4vc3R5bGUvY3NzLmNvbXBvbmVudCc7XG4iXX0=
14
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3Byb3ZvbHkvZGFzaGJvYXJkL3dpZGdldHMvd2lkZ2V0LW1hcC9wdWJsaWMtYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMscUJBQXFCLENBQUM7QUFDcEMsY0FBYyxrQ0FBa0MsQ0FBQztBQUNqRCxjQUFjLHNDQUFzQyxDQUFDO0FBQ3JELGNBQWMsNENBQTRDLENBQUM7QUFDM0QsY0FBYyxtQ0FBbUMsQ0FBQztBQUNsRCxjQUFjLHlDQUF5QyxDQUFDO0FBQ3hELGNBQWMsMkNBQTJDLENBQUM7QUFDMUQsY0FBYyxrQ0FBa0MsQ0FBQztBQUNqRCxjQUFjLHFDQUFxQyxDQUFDO0FBQ3BELGNBQWMsMEJBQTBCLENBQUM7QUFDekMsY0FBYyx5QkFBeUIsQ0FBQztBQUN4QyxjQUFjLHlCQUF5QixDQUFDO0FBQ3hDLGNBQWMsdUJBQXVCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL3dpZGdldC1tYXAubW9kdWxlJztcbmV4cG9ydCAqIGZyb20gJy4vY29tcG9uZW50L3dpZGdldC1tYXAuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vY29tcG9uZW50L3dpZGdldC1tYXAtbGF5ZXIuc2VydmljZSc7XG5leHBvcnQgKiBmcm9tICcuL3BpcGUvd2lkZ2V0LW1hcC1nZW9tZXRyeS1maWVsZHMtZm9yLnBpcGUnO1xuZXhwb3J0ICogZnJvbSAnLi9waXBlL3dpZGdldC1tYXAtbGVnZW5kLXVybC5waXBlJztcbmV4cG9ydCAqIGZyb20gJy4vaW50ZXJhY3Rpb24vaW50ZXJhY3Rpb24tbWFuYWdlci5jbGFzcyc7XG5leHBvcnQgKiBmcm9tICcuL2ludGVyYWN0aW9uL3NlbGVjdGlvbi1pbnRlcmFjdGlvbi5jbGFzcyc7XG5leHBvcnQgKiBmcm9tICcuL2ludGVyYWN0aW9uL2xheWVyLXNsaWRlci5jbGFzcyc7XG5leHBvcnQgKiBmcm9tICcuL2ludGVyYWN0aW9uL3Rvb2x0aXAtbWFuYWdlci5jbGFzcyc7XG5leHBvcnQgKiBmcm9tICcuL3V0aWxzL3dpZGdldC1tYXAudXRpbHMnO1xuZXhwb3J0ICogZnJvbSAnLi91dGlscy94bWwtdXRpbHMuY2xhc3MnO1xuZXhwb3J0ICogZnJvbSAnLi91dGlscy9jcWwtdXRpbHMuY2xhc3MnO1xuZXhwb3J0ICogZnJvbSAnLi9zdHlsZS9jc3MuY29tcG9uZW50JztcbiJdfQ==
@@ -41,10 +41,29 @@ export class WidgetTileComponent extends DataWidgetComponent {
41
41
  ]).pipe(map(([classes, tileDefs, maxTilesAttributes]) => ({
42
42
  ...classes
43
43
  .map((clazz) => ({
44
- [clazz.id]: [...clazz.attributes.map((attr) => attr.name).filter((_, i) => i < maxTilesAttributes)]
44
+ [clazz.id]: [
45
+ ...clazz.attributes
46
+ .map((attr) => ({
47
+ label: attr.name,
48
+ prop: attr.technicalName
49
+ }))
50
+ .filter((_, i) => i < maxTilesAttributes)
51
+ ]
45
52
  }))
46
53
  .reduce((p, c) => ({ ...p, ...c }), {}),
47
- ...tileDefs
54
+ ...classes
55
+ .filter((clazz) => !!tileDefs[clazz.id])
56
+ .map((clazz) => ({
57
+ [clazz.id]: [
58
+ ...clazz.attributes
59
+ .filter((attr) => tileDefs[clazz.id].includes(attr.name))
60
+ .map((attr) => ({
61
+ label: attr.name,
62
+ prop: attr.technicalName
63
+ }))
64
+ ]
65
+ }))
66
+ .reduce((p, c) => ({ ...p, ...c }), {})
48
67
  })));
49
68
  this.subscriptions.add(this.store.select(FieldSelectors.fields).subscribe((fields) => (this.fields = fields)));
50
69
  this.subscriptions.add(this.classes$.subscribe((classes) => (this.classes = classes)));
@@ -106,10 +125,10 @@ export class WidgetTileComponent extends DataWidgetComponent {
106
125
  return Object.keys(value?.metadata ?? {}).length > 0;
107
126
  }
108
127
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.1.3", ngImport: i0, type: WidgetTileComponent, deps: [{ token: i1.Store }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component }); }
109
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.1.3", type: WidgetTileComponent, selector: "pry-widget-tile", usesInheritance: true, ngImport: i0, template: "<pry-widget-tile-css></pry-widget-tile-css>\n<div class=\"o-widget o-widget--tile\">\n <pry-widget-header\n [datasourceIds]=\"(datasourceIds$ | async) ?? []\"\n *ngIf=\"displayHeader$ | async\"\n [widgetIndex]=\"widgetIndex\"\n [manifest]=\"manifest\"\n (manifestModified)=\"manifestModified.emit($event)\"\n [headerOptions]=\"(displayHeader$ | async) ?? {}\"\n #header\n >\n <pry-settings\n (saveTriggered)=\"emitManifest()\"\n (changeTitle)=\"changeWidgetTitle($event)\"\n [headerPresent]=\"displayHeader$ | async\"\n [widgetIndex]=\"widgetIndex\"\n [header]=\"header\"\n [open$]=\"open$\"\n class=\"o-settings\"\n >\n <div class=\"m-form-label-field\">\n <label class=\"a-label\" for=\"tile_classes\">{{ '@pry.widget.tile.classes' | i18n }}</label>\n <pry-select\n (ngModelChange)=\"changeClasses($event)\"\n [items]=\"classes$ | async\"\n [multiple]=\"true\"\n [closeOnSelect]=\"false\"\n [ngModel]=\"optionsCopy.classes\"\n bindLabel=\"name\"\n bindValue=\"id\"\n id=\"tile_classes\"\n ></pry-select>\n </div>\n </pry-settings>\n </pry-widget-header>\n <div class=\"o-widget--tile-wrapper\" [style.top]=\"(displayHeader$ | async) ? WIDGET_HEADER_HEIGHT + 'px' : 0\">\n <div class=\"no-result\" *ngIf=\"(items$ | async)?.length === 0\">\n <img\n class=\"no-result__search\"\n src=\"../../../assets/svgs/pry_recherche_pas_de_resultat.svg\"\n alt=\"{{ '@pry.widget.tile.empty' | i18n }}\"\n aria-hidden=\"true\"\n />\n <span class=\"no-result__text\">{{ '@pry.widget.tile.empty' | i18n }}</span>\n </div>\n <ul class=\"o-widget--tile__list\">\n <li\n class=\"o-widget--tile__list__item\"\n *ngFor=\"let item of items$ | async; let index = index\"\n (contextmenu)=\"contextMenu($event, item)\"\n >\n <div class=\"o-widget--tile__list__item__header\">\n <img\n [height]=\"25\"\n [width]=\"25\"\n [src]=\"item | translateItemToSymbol | async\"\n [alt]=\"item.oClass | translateId : { type: 'class', output: 'name' } | async\"\n />\n <h3 class=\"o-widget--tile__list__item__header__title\">\n {{\n item.oClass\n | translateId\n : {\n type: 'class',\n output: 'name'\n }\n | async\n }}\n </h3>\n <pry-checkbox\n [ngModel]=\"(selectedIds$ | async).includes(item.id)\"\n (click)=\"select($event, item)\"\n ></pry-checkbox>\n </div>\n <div class=\"o-properties\">\n <ul class=\"o-properties__list\">\n <ng-container *ngFor=\"let property of (propertiesToDisplay$ | async)![item.oClass]\">\n <li\n *ngFor=\"let value of getValue(item, property); let i = index; let c = count\"\n class=\"o-properties__list__item\"\n >\n <p class=\"o-properties__list__item__label\" *ngIf=\"i === 0\">{{ property }}:</p>\n <p class=\"o-properties__list__item__value\" *ngIf=\"!isRaw(property, item.oClass)\">{{ value?.value }}</p>\n <p class=\"o-properties__list__item__raw\" *ngIf=\"isRaw(property, item.oClass)\">\n <pry-vizualize-raw [item]=\"item\" [propertyName]=\"property\"></pry-vizualize-raw>\n </p>\n <ng-container *ngIf=\"hasMetadata(value)\">\n <pry-metadata [item]=\"value\"></pry-metadata>\n </ng-container>\n </li>\n </ng-container>\n </ul>\n </div>\n <div (click)=\"toggleMap(index)\" class=\"o-widget--tile__list__item__footer\">\n <button\n type=\"button\"\n class=\"a-btn a-btn--icon-text -link-like\"\n aria-controls=\"tile-map-xxx\"\n aria-expanded=\"false\"\n >\n {{ '@pry.widget.tile.map' | i18n }}\n </button>\n </div>\n <div class=\"o-widget--tile__list__item__map\" *ngIf=\"maps[index]\" id=\"tile-map-xxx\">\n <pry-widget-map [staticManifest]=\"mapManifest\" [staticResultSet]=\"resultSetWithItem(item)\"></pry-widget-map>\n </div>\n </li>\n </ul>\n </div>\n</div>\n", dependencies: [{ kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i3.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: i4.SettingsComponent, selector: "pry-settings", inputs: ["widgetIndex", "isDisable", "headerPresent", "open$", "header"], outputs: ["triggerClick", "saveTriggered", "changeTitle"] }, { kind: "component", type: i4.MetadataComponent, selector: "pry-metadata", inputs: ["item"] }, { kind: "component", type: i4.PryWidgetHeaderComponent, selector: "pry-widget-header", inputs: ["manifest", "openData$", "additionalOptions", "headerOptions", "displayCount", "datasourceIds", "widgetIndex"], outputs: ["manifestModified"] }, { kind: "component", type: i4.VizualizeRawComponent, selector: "pry-vizualize-raw", inputs: ["item", "propertyName", "alt"] }, { kind: "component", type: i4.PrySelectComponent, selector: "pry-select", inputs: ["items", "clearable", "multiple", "closeOnSelect", "placeholder", "isForm", "required", "name", "readonly", "autocomplete", "alwaysShowAutosuggestedValues", "externalAutocompleteService", "bindValue", "bindLabel", "iconSize", "bindIcon", "template", "i18nPrefix", "bindClasses", "loading", "elementRef"], outputs: ["searched", "cleared", "clicked"] }, { kind: "component", type: i5.PryCheckboxComponent, selector: "pry-checkbox", inputs: ["circle"] }, { kind: "component", type: i6.WidgetMapComponent, selector: "pry-widget-map" }, { kind: "component", type: i7.PryWidgetTileCssComponent, selector: "pry-widget-tile-css" }, { kind: "pipe", type: i2.AsyncPipe, name: "async" }, { kind: "pipe", type: i4.TranslateIdPipe, name: "translateId" }, { kind: "pipe", type: i4.TranslateItemToSymbolPipe, name: "translateItemToSymbol" }, { kind: "pipe", type: i4.I18nPipe, name: "i18n" }] }); }
128
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.1.3", type: WidgetTileComponent, selector: "pry-widget-tile", usesInheritance: true, ngImport: i0, template: "<pry-widget-tile-css></pry-widget-tile-css>\n<div class=\"o-widget o-widget--tile\">\n <pry-widget-header\n [datasourceIds]=\"(datasourceIds$ | async) ?? []\"\n *ngIf=\"displayHeader$ | async\"\n [widgetIndex]=\"widgetIndex\"\n [manifest]=\"manifest\"\n (manifestModified)=\"manifestModified.emit($event)\"\n [headerOptions]=\"(displayHeader$ | async) ?? {}\"\n #header\n >\n <pry-settings\n (saveTriggered)=\"emitManifest()\"\n (changeTitle)=\"changeWidgetTitle($event)\"\n [headerPresent]=\"displayHeader$ | async\"\n [widgetIndex]=\"widgetIndex\"\n [header]=\"header\"\n [open$]=\"open$\"\n class=\"o-settings\"\n >\n <div class=\"m-form-label-field\">\n <label class=\"a-label\" for=\"tile_classes\">{{ '@pry.widget.tile.classes' | i18n }}</label>\n <pry-select\n (ngModelChange)=\"changeClasses($event)\"\n [items]=\"classes$ | async\"\n [multiple]=\"true\"\n [closeOnSelect]=\"false\"\n [ngModel]=\"optionsCopy.classes\"\n bindLabel=\"name\"\n bindValue=\"id\"\n id=\"tile_classes\"\n ></pry-select>\n </div>\n </pry-settings>\n </pry-widget-header>\n <div class=\"o-widget--tile-wrapper\" [style.top]=\"(displayHeader$ | async) ? WIDGET_HEADER_HEIGHT + 'px' : 0\">\n <div class=\"no-result\" *ngIf=\"(items$ | async)?.length === 0\">\n <img\n class=\"no-result__search\"\n src=\"../../../assets/svgs/pry_recherche_pas_de_resultat.svg\"\n alt=\"{{ '@pry.widget.tile.empty' | i18n }}\"\n aria-hidden=\"true\"\n />\n <span class=\"no-result__text\">{{ '@pry.widget.tile.empty' | i18n }}</span>\n </div>\n <ul class=\"o-widget--tile__list\">\n <li\n class=\"o-widget--tile__list__item\"\n *ngFor=\"let item of items$ | async; let index = index\"\n (contextmenu)=\"contextMenu($event, item)\"\n >\n <div class=\"o-widget--tile__list__item__header\">\n <img\n [height]=\"25\"\n [width]=\"25\"\n [src]=\"item | translateItemToSymbol | async\"\n [alt]=\"item.oClass | translateId: { type: 'class', output: 'name' } | async\"\n />\n <h3 class=\"o-widget--tile__list__item__header__title\">\n {{\n item.oClass\n | translateId\n : {\n type: 'class',\n output: 'name'\n }\n | async\n }}\n </h3>\n <pry-checkbox\n [ngModel]=\"(selectedIds$ | async).includes(item.id)\"\n (click)=\"select($event, item)\"\n ></pry-checkbox>\n </div>\n <div class=\"o-properties\">\n <ul class=\"o-properties__list\">\n <ng-container *ngFor=\"let property of (propertiesToDisplay$ | async)![item.oClass]\">\n <li\n *ngFor=\"let value of getValue(item, property.prop); let i = index; let c = count\"\n class=\"o-properties__list__item\"\n >\n <p class=\"o-properties__list__item__label\" *ngIf=\"i === 0\">{{ property.label }}:</p>\n <p class=\"o-properties__list__item__value\" *ngIf=\"!isRaw(property.label, item.oClass)\">\n {{ value?.value }}\n </p>\n <p class=\"o-properties__list__item__raw\" *ngIf=\"isRaw(property.label, item.oClass)\">\n <pry-vizualize-raw [item]=\"item\" [propertyName]=\"property.label\"></pry-vizualize-raw>\n </p>\n <ng-container *ngIf=\"hasMetadata(value)\">\n <pry-metadata [item]=\"value\"></pry-metadata>\n </ng-container>\n </li>\n </ng-container>\n </ul>\n </div>\n <div (click)=\"toggleMap(index)\" class=\"o-widget--tile__list__item__footer\">\n <button\n type=\"button\"\n class=\"a-btn a-btn--icon-text -link-like\"\n aria-controls=\"tile-map-xxx\"\n aria-expanded=\"false\"\n >\n {{ '@pry.widget.tile.map' | i18n }}\n </button>\n </div>\n <div class=\"o-widget--tile__list__item__map\" *ngIf=\"maps[index]\" id=\"tile-map-xxx\">\n <pry-widget-map [staticManifest]=\"mapManifest\" [staticResultSet]=\"resultSetWithItem(item)\"></pry-widget-map>\n </div>\n </li>\n </ul>\n </div>\n</div>\n", dependencies: [{ kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i3.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: i4.SettingsComponent, selector: "pry-settings", inputs: ["widgetIndex", "isDisable", "headerPresent", "open$", "header"], outputs: ["triggerClick", "saveTriggered", "changeTitle"] }, { kind: "component", type: i4.MetadataComponent, selector: "pry-metadata", inputs: ["item"] }, { kind: "component", type: i4.PryWidgetHeaderComponent, selector: "pry-widget-header", inputs: ["manifest", "openData$", "additionalOptions", "headerOptions", "displayCount", "datasourceIds", "widgetIndex"], outputs: ["manifestModified"] }, { kind: "component", type: i4.VizualizeRawComponent, selector: "pry-vizualize-raw", inputs: ["item", "propertyName", "alt"] }, { kind: "component", type: i4.PrySelectComponent, selector: "pry-select", inputs: ["items", "clearable", "multiple", "closeOnSelect", "placeholder", "isForm", "required", "name", "readonly", "autocomplete", "alwaysShowAutosuggestedValues", "externalAutocompleteService", "bindValue", "bindLabel", "iconSize", "bindIcon", "template", "i18nPrefix", "bindClasses", "loading", "elementRef"], outputs: ["searched", "cleared", "clicked"] }, { kind: "component", type: i5.PryCheckboxComponent, selector: "pry-checkbox", inputs: ["circle"] }, { kind: "component", type: i6.WidgetMapComponent, selector: "pry-widget-map" }, { kind: "component", type: i7.PryWidgetTileCssComponent, selector: "pry-widget-tile-css" }, { kind: "pipe", type: i2.AsyncPipe, name: "async" }, { kind: "pipe", type: i4.TranslateIdPipe, name: "translateId" }, { kind: "pipe", type: i4.TranslateItemToSymbolPipe, name: "translateItemToSymbol" }, { kind: "pipe", type: i4.I18nPipe, name: "i18n" }] }); }
110
129
  }
111
130
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.1.3", ngImport: i0, type: WidgetTileComponent, decorators: [{
112
131
  type: Component,
113
- args: [{ selector: 'pry-widget-tile', template: "<pry-widget-tile-css></pry-widget-tile-css>\n<div class=\"o-widget o-widget--tile\">\n <pry-widget-header\n [datasourceIds]=\"(datasourceIds$ | async) ?? []\"\n *ngIf=\"displayHeader$ | async\"\n [widgetIndex]=\"widgetIndex\"\n [manifest]=\"manifest\"\n (manifestModified)=\"manifestModified.emit($event)\"\n [headerOptions]=\"(displayHeader$ | async) ?? {}\"\n #header\n >\n <pry-settings\n (saveTriggered)=\"emitManifest()\"\n (changeTitle)=\"changeWidgetTitle($event)\"\n [headerPresent]=\"displayHeader$ | async\"\n [widgetIndex]=\"widgetIndex\"\n [header]=\"header\"\n [open$]=\"open$\"\n class=\"o-settings\"\n >\n <div class=\"m-form-label-field\">\n <label class=\"a-label\" for=\"tile_classes\">{{ '@pry.widget.tile.classes' | i18n }}</label>\n <pry-select\n (ngModelChange)=\"changeClasses($event)\"\n [items]=\"classes$ | async\"\n [multiple]=\"true\"\n [closeOnSelect]=\"false\"\n [ngModel]=\"optionsCopy.classes\"\n bindLabel=\"name\"\n bindValue=\"id\"\n id=\"tile_classes\"\n ></pry-select>\n </div>\n </pry-settings>\n </pry-widget-header>\n <div class=\"o-widget--tile-wrapper\" [style.top]=\"(displayHeader$ | async) ? WIDGET_HEADER_HEIGHT + 'px' : 0\">\n <div class=\"no-result\" *ngIf=\"(items$ | async)?.length === 0\">\n <img\n class=\"no-result__search\"\n src=\"../../../assets/svgs/pry_recherche_pas_de_resultat.svg\"\n alt=\"{{ '@pry.widget.tile.empty' | i18n }}\"\n aria-hidden=\"true\"\n />\n <span class=\"no-result__text\">{{ '@pry.widget.tile.empty' | i18n }}</span>\n </div>\n <ul class=\"o-widget--tile__list\">\n <li\n class=\"o-widget--tile__list__item\"\n *ngFor=\"let item of items$ | async; let index = index\"\n (contextmenu)=\"contextMenu($event, item)\"\n >\n <div class=\"o-widget--tile__list__item__header\">\n <img\n [height]=\"25\"\n [width]=\"25\"\n [src]=\"item | translateItemToSymbol | async\"\n [alt]=\"item.oClass | translateId : { type: 'class', output: 'name' } | async\"\n />\n <h3 class=\"o-widget--tile__list__item__header__title\">\n {{\n item.oClass\n | translateId\n : {\n type: 'class',\n output: 'name'\n }\n | async\n }}\n </h3>\n <pry-checkbox\n [ngModel]=\"(selectedIds$ | async).includes(item.id)\"\n (click)=\"select($event, item)\"\n ></pry-checkbox>\n </div>\n <div class=\"o-properties\">\n <ul class=\"o-properties__list\">\n <ng-container *ngFor=\"let property of (propertiesToDisplay$ | async)![item.oClass]\">\n <li\n *ngFor=\"let value of getValue(item, property); let i = index; let c = count\"\n class=\"o-properties__list__item\"\n >\n <p class=\"o-properties__list__item__label\" *ngIf=\"i === 0\">{{ property }}:</p>\n <p class=\"o-properties__list__item__value\" *ngIf=\"!isRaw(property, item.oClass)\">{{ value?.value }}</p>\n <p class=\"o-properties__list__item__raw\" *ngIf=\"isRaw(property, item.oClass)\">\n <pry-vizualize-raw [item]=\"item\" [propertyName]=\"property\"></pry-vizualize-raw>\n </p>\n <ng-container *ngIf=\"hasMetadata(value)\">\n <pry-metadata [item]=\"value\"></pry-metadata>\n </ng-container>\n </li>\n </ng-container>\n </ul>\n </div>\n <div (click)=\"toggleMap(index)\" class=\"o-widget--tile__list__item__footer\">\n <button\n type=\"button\"\n class=\"a-btn a-btn--icon-text -link-like\"\n aria-controls=\"tile-map-xxx\"\n aria-expanded=\"false\"\n >\n {{ '@pry.widget.tile.map' | i18n }}\n </button>\n </div>\n <div class=\"o-widget--tile__list__item__map\" *ngIf=\"maps[index]\" id=\"tile-map-xxx\">\n <pry-widget-map [staticManifest]=\"mapManifest\" [staticResultSet]=\"resultSetWithItem(item)\"></pry-widget-map>\n </div>\n </li>\n </ul>\n </div>\n</div>\n" }]
132
+ args: [{ selector: 'pry-widget-tile', template: "<pry-widget-tile-css></pry-widget-tile-css>\n<div class=\"o-widget o-widget--tile\">\n <pry-widget-header\n [datasourceIds]=\"(datasourceIds$ | async) ?? []\"\n *ngIf=\"displayHeader$ | async\"\n [widgetIndex]=\"widgetIndex\"\n [manifest]=\"manifest\"\n (manifestModified)=\"manifestModified.emit($event)\"\n [headerOptions]=\"(displayHeader$ | async) ?? {}\"\n #header\n >\n <pry-settings\n (saveTriggered)=\"emitManifest()\"\n (changeTitle)=\"changeWidgetTitle($event)\"\n [headerPresent]=\"displayHeader$ | async\"\n [widgetIndex]=\"widgetIndex\"\n [header]=\"header\"\n [open$]=\"open$\"\n class=\"o-settings\"\n >\n <div class=\"m-form-label-field\">\n <label class=\"a-label\" for=\"tile_classes\">{{ '@pry.widget.tile.classes' | i18n }}</label>\n <pry-select\n (ngModelChange)=\"changeClasses($event)\"\n [items]=\"classes$ | async\"\n [multiple]=\"true\"\n [closeOnSelect]=\"false\"\n [ngModel]=\"optionsCopy.classes\"\n bindLabel=\"name\"\n bindValue=\"id\"\n id=\"tile_classes\"\n ></pry-select>\n </div>\n </pry-settings>\n </pry-widget-header>\n <div class=\"o-widget--tile-wrapper\" [style.top]=\"(displayHeader$ | async) ? WIDGET_HEADER_HEIGHT + 'px' : 0\">\n <div class=\"no-result\" *ngIf=\"(items$ | async)?.length === 0\">\n <img\n class=\"no-result__search\"\n src=\"../../../assets/svgs/pry_recherche_pas_de_resultat.svg\"\n alt=\"{{ '@pry.widget.tile.empty' | i18n }}\"\n aria-hidden=\"true\"\n />\n <span class=\"no-result__text\">{{ '@pry.widget.tile.empty' | i18n }}</span>\n </div>\n <ul class=\"o-widget--tile__list\">\n <li\n class=\"o-widget--tile__list__item\"\n *ngFor=\"let item of items$ | async; let index = index\"\n (contextmenu)=\"contextMenu($event, item)\"\n >\n <div class=\"o-widget--tile__list__item__header\">\n <img\n [height]=\"25\"\n [width]=\"25\"\n [src]=\"item | translateItemToSymbol | async\"\n [alt]=\"item.oClass | translateId: { type: 'class', output: 'name' } | async\"\n />\n <h3 class=\"o-widget--tile__list__item__header__title\">\n {{\n item.oClass\n | translateId\n : {\n type: 'class',\n output: 'name'\n }\n | async\n }}\n </h3>\n <pry-checkbox\n [ngModel]=\"(selectedIds$ | async).includes(item.id)\"\n (click)=\"select($event, item)\"\n ></pry-checkbox>\n </div>\n <div class=\"o-properties\">\n <ul class=\"o-properties__list\">\n <ng-container *ngFor=\"let property of (propertiesToDisplay$ | async)![item.oClass]\">\n <li\n *ngFor=\"let value of getValue(item, property.prop); let i = index; let c = count\"\n class=\"o-properties__list__item\"\n >\n <p class=\"o-properties__list__item__label\" *ngIf=\"i === 0\">{{ property.label }}:</p>\n <p class=\"o-properties__list__item__value\" *ngIf=\"!isRaw(property.label, item.oClass)\">\n {{ value?.value }}\n </p>\n <p class=\"o-properties__list__item__raw\" *ngIf=\"isRaw(property.label, item.oClass)\">\n <pry-vizualize-raw [item]=\"item\" [propertyName]=\"property.label\"></pry-vizualize-raw>\n </p>\n <ng-container *ngIf=\"hasMetadata(value)\">\n <pry-metadata [item]=\"value\"></pry-metadata>\n </ng-container>\n </li>\n </ng-container>\n </ul>\n </div>\n <div (click)=\"toggleMap(index)\" class=\"o-widget--tile__list__item__footer\">\n <button\n type=\"button\"\n class=\"a-btn a-btn--icon-text -link-like\"\n aria-controls=\"tile-map-xxx\"\n aria-expanded=\"false\"\n >\n {{ '@pry.widget.tile.map' | i18n }}\n </button>\n </div>\n <div class=\"o-widget--tile__list__item__map\" *ngIf=\"maps[index]\" id=\"tile-map-xxx\">\n <pry-widget-map [staticManifest]=\"mapManifest\" [staticResultSet]=\"resultSetWithItem(item)\"></pry-widget-map>\n </div>\n </li>\n </ul>\n </div>\n</div>\n" }]
114
133
  }], ctorParameters: () => [{ type: i1.Store }, { type: i0.ElementRef }] });
115
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"widget-tile.component.js","sourceRoot":"","sources":["../../../../../../../projects/provoly/dashboard/widgets/widget-tile/component/widget-tile.component.ts","../../../../../../../projects/provoly/dashboard/widgets/widget-tile/component/widget-tile.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef } from '@angular/core';\nimport { Store } from '@ngrx/store';\nimport {\n  AttributeSimpleValue,\n  Class,\n  ClassSelectors,\n  ConfigSelectors,\n  ContextMenuActions,\n  DashboardActions,\n  DashboardSelectors,\n  DataWidgetComponent,\n  Field,\n  FieldSelectors,\n  FieldType,\n  Item,\n  ItemUtils,\n  ResultSet,\n  TileWidgetOptions,\n  WIDGET_HEADER_HEIGHT\n} from '@provoly/dashboard';\nimport { combineLatest, Observable } from 'rxjs';\nimport { filter, map } from 'rxjs/operators';\n\n@Component({\n  selector: 'pry-widget-tile',\n  templateUrl: './widget-tile.component.html'\n})\nexport class WidgetTileComponent extends DataWidgetComponent {\n  options$: Observable<TileWidgetOptions>;\n  items$: Observable<Item[]>;\n  selectedIds$: Observable<any>;\n  maps: boolean[] = [];\n  mapManifest = {\n    type: 'map',\n    header: false,\n    layout: { x: 1, y: 1, width: 1, height: 1 },\n    options: { layers: [{ type: 'marker' }], fit: true }\n  };\n  classes$: Observable<Class[]>;\n  optionsCopy: TileWidgetOptions = {};\n  WIDGET_HEADER_HEIGHT = WIDGET_HEADER_HEIGHT;\n  propertiesToDisplay$: Observable<{ [p: string]: string[] } | undefined>;\n  fields: Field[] = [];\n  classes: Class[] = [];\n\n  constructor(store: Store<any>, el: ElementRef) {\n    super(store, el);\n    this.options$ = this.manifest$.pipe(map((manifest) => manifest.options as TileWidgetOptions));\n    this.subscriptions.add(\n      this.options$.pipe(filter((opt) => !!opt)).subscribe((options) => {\n        this.optionsCopy = { ...(JSON.parse(JSON.stringify(options)) as TileWidgetOptions) };\n      })\n    );\n    this.items$ = combineLatest([\n      this.resultSet$.pipe(map((rs) => Object.values(rs.items).reduce((p, c) => [...p, ...c], []))),\n      this.options$\n    ]).pipe(\n      map(([items, options]) =>\n        items.filter(\n          (item) => !options.classes || options.classes.length === 0 || options.classes.includes(item.oClass)\n        )\n      )\n    );\n    this.selectedIds$ = this.store.select(DashboardSelectors.selectedItemIds);\n    this.classes$ = this.store.select(ClassSelectors.classes);\n    this.propertiesToDisplay$ = combineLatest([\n      this.classes$,\n      this.store.select(ConfigSelectors.tileAttributesDefinitions),\n      this.store.select(ConfigSelectors.maxAttributesInTiles)\n    ]).pipe(\n      map(([classes, tileDefs, maxTilesAttributes]) => ({\n        ...classes\n          .map((clazz) => ({\n            [clazz.id]: [...clazz.attributes.map((attr) => attr.name).filter((_, i) => i < maxTilesAttributes)]\n          }))\n          .reduce((p, c) => ({ ...p, ...c }), {}),\n        ...tileDefs\n      }))\n    );\n    this.subscriptions.add(this.store.select(FieldSelectors.fields).subscribe((fields) => (this.fields = fields)));\n    this.subscriptions.add(this.classes$.subscribe((classes) => (this.classes = classes)));\n  }\n\n  contextMenu($event: MouseEvent, item: Item) {\n    if (this.manifest.contextMenu !== false) {\n      $event.stopPropagation();\n      $event.preventDefault();\n      this.store.dispatch(\n        ContextMenuActions.open({\n          itemId: item.id,\n          clientX: $event.clientX,\n          clientY: $event.clientY,\n          from: 'tile',\n          allowObjectCreation: false\n        })\n      );\n    }\n  }\n\n  select($event: MouseEvent, item: Item) {\n    $event.stopPropagation();\n    $event.preventDefault();\n    this.store.dispatch(DashboardActions.select({ id: item.id, cumulative: !$event.shiftKey }));\n  }\n\n  toggleMap(index: number) {\n    this.maps[index] = !this.maps[index];\n  }\n\n  resultSetWithItem(item: Item): ResultSet {\n    return { items: { [item.oClass]: [item] }, relations: [] };\n  }\n\n  emitManifest() {\n    this.manifestModified.emit({\n      widgetIndex: this.widgetIndex,\n      manifest: {\n        ...this.manifest,\n        options: this.optionsCopy\n      }\n    });\n  }\n\n  changeClasses($event: any) {\n    this.optionsCopy.classes = $event;\n  }\n\n  getValue(item: Item, property: string): AttributeSimpleValue[] {\n    const value = ItemUtils.getAttributeValue(item, property);\n    return Array.isArray(value) ? value : [value];\n  }\n\n  isRaw(property: string, classId: string): boolean {\n    const clazz = this.classes.find((clazz) => clazz.id === classId);\n    if (!!clazz) {\n      const prop = clazz.attributes.find((attr) => attr.name === property);\n      if (!!prop) {\n        const field = this.fields.find((field) => field.id === prop.field);\n        if (!!field) {\n          return field.type === FieldType.RAW;\n        }\n      }\n    }\n    return false;\n  }\n\n  hasMetadata(value: AttributeSimpleValue | Item) {\n    return Object.keys(value?.metadata ?? {}).length > 0;\n  }\n}\n","<pry-widget-tile-css></pry-widget-tile-css>\n<div class=\"o-widget o-widget--tile\">\n  <pry-widget-header\n    [datasourceIds]=\"(datasourceIds$ | async) ?? []\"\n    *ngIf=\"displayHeader$ | async\"\n    [widgetIndex]=\"widgetIndex\"\n    [manifest]=\"manifest\"\n    (manifestModified)=\"manifestModified.emit($event)\"\n    [headerOptions]=\"(displayHeader$ | async) ?? {}\"\n    #header\n  >\n    <pry-settings\n      (saveTriggered)=\"emitManifest()\"\n      (changeTitle)=\"changeWidgetTitle($event)\"\n      [headerPresent]=\"displayHeader$ | async\"\n      [widgetIndex]=\"widgetIndex\"\n      [header]=\"header\"\n      [open$]=\"open$\"\n      class=\"o-settings\"\n    >\n      <div class=\"m-form-label-field\">\n        <label class=\"a-label\" for=\"tile_classes\">{{ '@pry.widget.tile.classes' | i18n }}</label>\n        <pry-select\n          (ngModelChange)=\"changeClasses($event)\"\n          [items]=\"classes$ | async\"\n          [multiple]=\"true\"\n          [closeOnSelect]=\"false\"\n          [ngModel]=\"optionsCopy.classes\"\n          bindLabel=\"name\"\n          bindValue=\"id\"\n          id=\"tile_classes\"\n        ></pry-select>\n      </div>\n    </pry-settings>\n  </pry-widget-header>\n  <div class=\"o-widget--tile-wrapper\" [style.top]=\"(displayHeader$ | async) ? WIDGET_HEADER_HEIGHT + 'px' : 0\">\n    <div class=\"no-result\" *ngIf=\"(items$ | async)?.length === 0\">\n      <img\n        class=\"no-result__search\"\n        src=\"../../../assets/svgs/pry_recherche_pas_de_resultat.svg\"\n        alt=\"{{ '@pry.widget.tile.empty' | i18n }}\"\n        aria-hidden=\"true\"\n      />\n      <span class=\"no-result__text\">{{ '@pry.widget.tile.empty' | i18n }}</span>\n    </div>\n    <ul class=\"o-widget--tile__list\">\n      <li\n        class=\"o-widget--tile__list__item\"\n        *ngFor=\"let item of items$ | async; let index = index\"\n        (contextmenu)=\"contextMenu($event, item)\"\n      >\n        <div class=\"o-widget--tile__list__item__header\">\n          <img\n            [height]=\"25\"\n            [width]=\"25\"\n            [src]=\"item | translateItemToSymbol | async\"\n            [alt]=\"item.oClass | translateId : { type: 'class', output: 'name' } | async\"\n          />\n          <h3 class=\"o-widget--tile__list__item__header__title\">\n            {{\n              item.oClass\n                | translateId\n                  : {\n                      type: 'class',\n                      output: 'name'\n                    }\n                | async\n            }}\n          </h3>\n          <pry-checkbox\n            [ngModel]=\"(selectedIds$ | async).includes(item.id)\"\n            (click)=\"select($event, item)\"\n          ></pry-checkbox>\n        </div>\n        <div class=\"o-properties\">\n          <ul class=\"o-properties__list\">\n            <ng-container *ngFor=\"let property of (propertiesToDisplay$ | async)![item.oClass]\">\n              <li\n                *ngFor=\"let value of getValue(item, property); let i = index; let c = count\"\n                class=\"o-properties__list__item\"\n              >\n                <p class=\"o-properties__list__item__label\" *ngIf=\"i === 0\">{{ property }}:</p>\n                <p class=\"o-properties__list__item__value\" *ngIf=\"!isRaw(property, item.oClass)\">{{ value?.value }}</p>\n                <p class=\"o-properties__list__item__raw\" *ngIf=\"isRaw(property, item.oClass)\">\n                  <pry-vizualize-raw [item]=\"item\" [propertyName]=\"property\"></pry-vizualize-raw>\n                </p>\n                <ng-container *ngIf=\"hasMetadata(value)\">\n                  <pry-metadata [item]=\"value\"></pry-metadata>\n                </ng-container>\n              </li>\n            </ng-container>\n          </ul>\n        </div>\n        <div (click)=\"toggleMap(index)\" class=\"o-widget--tile__list__item__footer\">\n          <button\n            type=\"button\"\n            class=\"a-btn a-btn--icon-text -link-like\"\n            aria-controls=\"tile-map-xxx\"\n            aria-expanded=\"false\"\n          >\n            {{ '@pry.widget.tile.map' | i18n }}\n          </button>\n        </div>\n        <div class=\"o-widget--tile__list__item__map\" *ngIf=\"maps[index]\" id=\"tile-map-xxx\">\n          <pry-widget-map [staticManifest]=\"mapManifest\" [staticResultSet]=\"resultSetWithItem(item)\"></pry-widget-map>\n        </div>\n      </li>\n    </ul>\n  </div>\n</div>\n"]}
134
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"widget-tile.component.js","sourceRoot":"","sources":["../../../../../../../projects/provoly/dashboard/widgets/widget-tile/component/widget-tile.component.ts","../../../../../../../projects/provoly/dashboard/widgets/widget-tile/component/widget-tile.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef } from '@angular/core';\nimport { Store } from '@ngrx/store';\nimport {\n  AttributeSimpleValue,\n  Class,\n  ClassSelectors,\n  ConfigSelectors,\n  ContextMenuActions,\n  DashboardActions,\n  DashboardSelectors,\n  DataWidgetComponent,\n  Field,\n  FieldSelectors,\n  FieldType,\n  Item,\n  ItemUtils,\n  ResultSet,\n  TileWidgetOptions,\n  WIDGET_HEADER_HEIGHT\n} from '@provoly/dashboard';\nimport { combineLatest, Observable } from 'rxjs';\nimport { filter, map } from 'rxjs/operators';\n\n@Component({\n  selector: 'pry-widget-tile',\n  templateUrl: './widget-tile.component.html'\n})\nexport class WidgetTileComponent extends DataWidgetComponent {\n  options$: Observable<TileWidgetOptions>;\n  items$: Observable<Item[]>;\n  selectedIds$: Observable<any>;\n  maps: boolean[] = [];\n  mapManifest = {\n    type: 'map',\n    header: false,\n    layout: { x: 1, y: 1, width: 1, height: 1 },\n    options: { layers: [{ type: 'marker' }], fit: true }\n  };\n  classes$: Observable<Class[]>;\n  optionsCopy: TileWidgetOptions = {};\n  WIDGET_HEADER_HEIGHT = WIDGET_HEADER_HEIGHT;\n  propertiesToDisplay$: Observable<{ [p: string]: { label: string; prop: string }[] } | undefined>;\n  fields: Field[] = [];\n  classes: Class[] = [];\n\n  constructor(store: Store<any>, el: ElementRef) {\n    super(store, el);\n    this.options$ = this.manifest$.pipe(map((manifest) => manifest.options as TileWidgetOptions));\n    this.subscriptions.add(\n      this.options$.pipe(filter((opt) => !!opt)).subscribe((options) => {\n        this.optionsCopy = { ...(JSON.parse(JSON.stringify(options)) as TileWidgetOptions) };\n      })\n    );\n    this.items$ = combineLatest([\n      this.resultSet$.pipe(map((rs) => Object.values(rs.items).reduce((p, c) => [...p, ...c], []))),\n      this.options$\n    ]).pipe(\n      map(([items, options]) =>\n        items.filter(\n          (item) => !options.classes || options.classes.length === 0 || options.classes.includes(item.oClass)\n        )\n      )\n    );\n    this.selectedIds$ = this.store.select(DashboardSelectors.selectedItemIds);\n    this.classes$ = this.store.select(ClassSelectors.classes);\n    this.propertiesToDisplay$ = combineLatest([\n      this.classes$,\n      this.store.select(ConfigSelectors.tileAttributesDefinitions),\n      this.store.select(ConfigSelectors.maxAttributesInTiles)\n    ]).pipe(\n      map(([classes, tileDefs, maxTilesAttributes]) => ({\n        ...classes\n          .map((clazz) => ({\n            [clazz.id]: [\n              ...clazz.attributes\n                .map((attr) => ({\n                  label: attr.name,\n                  prop: attr.technicalName\n                }))\n                .filter((_, i) => i < maxTilesAttributes)\n            ]\n          }))\n          .reduce((p, c) => ({ ...p, ...c }), {}),\n        ...classes\n          .filter((clazz) => !!tileDefs[clazz.id])\n          .map((clazz) => ({\n            [clazz.id]: [\n              ...clazz.attributes\n                .filter((attr) => tileDefs[clazz.id].includes(attr.name))\n                .map((attr) => ({\n                  label: attr.name,\n                  prop: attr.technicalName\n                }))\n            ]\n          }))\n          .reduce((p, c) => ({ ...p, ...c }), {})\n      }))\n    );\n    this.subscriptions.add(this.store.select(FieldSelectors.fields).subscribe((fields) => (this.fields = fields)));\n    this.subscriptions.add(this.classes$.subscribe((classes) => (this.classes = classes)));\n  }\n\n  contextMenu($event: MouseEvent, item: Item) {\n    if (this.manifest.contextMenu !== false) {\n      $event.stopPropagation();\n      $event.preventDefault();\n      this.store.dispatch(\n        ContextMenuActions.open({\n          itemId: item.id,\n          clientX: $event.clientX,\n          clientY: $event.clientY,\n          from: 'tile',\n          allowObjectCreation: false\n        })\n      );\n    }\n  }\n\n  select($event: MouseEvent, item: Item) {\n    $event.stopPropagation();\n    $event.preventDefault();\n    this.store.dispatch(DashboardActions.select({ id: item.id, cumulative: !$event.shiftKey }));\n  }\n\n  toggleMap(index: number) {\n    this.maps[index] = !this.maps[index];\n  }\n\n  resultSetWithItem(item: Item): ResultSet {\n    return { items: { [item.oClass]: [item] }, relations: [] };\n  }\n\n  emitManifest() {\n    this.manifestModified.emit({\n      widgetIndex: this.widgetIndex,\n      manifest: {\n        ...this.manifest,\n        options: this.optionsCopy\n      }\n    });\n  }\n\n  changeClasses($event: any) {\n    this.optionsCopy.classes = $event;\n  }\n\n  getValue(item: Item, property: string): AttributeSimpleValue[] {\n    const value = ItemUtils.getAttributeValue(item, property);\n    return Array.isArray(value) ? value : [value];\n  }\n\n  isRaw(property: string, classId: string): boolean {\n    const clazz = this.classes.find((clazz) => clazz.id === classId);\n    if (!!clazz) {\n      const prop = clazz.attributes.find((attr) => attr.name === property);\n      if (!!prop) {\n        const field = this.fields.find((field) => field.id === prop.field);\n        if (!!field) {\n          return field.type === FieldType.RAW;\n        }\n      }\n    }\n    return false;\n  }\n\n  hasMetadata(value: AttributeSimpleValue | Item) {\n    return Object.keys(value?.metadata ?? {}).length > 0;\n  }\n}\n","<pry-widget-tile-css></pry-widget-tile-css>\n<div class=\"o-widget o-widget--tile\">\n  <pry-widget-header\n    [datasourceIds]=\"(datasourceIds$ | async) ?? []\"\n    *ngIf=\"displayHeader$ | async\"\n    [widgetIndex]=\"widgetIndex\"\n    [manifest]=\"manifest\"\n    (manifestModified)=\"manifestModified.emit($event)\"\n    [headerOptions]=\"(displayHeader$ | async) ?? {}\"\n    #header\n  >\n    <pry-settings\n      (saveTriggered)=\"emitManifest()\"\n      (changeTitle)=\"changeWidgetTitle($event)\"\n      [headerPresent]=\"displayHeader$ | async\"\n      [widgetIndex]=\"widgetIndex\"\n      [header]=\"header\"\n      [open$]=\"open$\"\n      class=\"o-settings\"\n    >\n      <div class=\"m-form-label-field\">\n        <label class=\"a-label\" for=\"tile_classes\">{{ '@pry.widget.tile.classes' | i18n }}</label>\n        <pry-select\n          (ngModelChange)=\"changeClasses($event)\"\n          [items]=\"classes$ | async\"\n          [multiple]=\"true\"\n          [closeOnSelect]=\"false\"\n          [ngModel]=\"optionsCopy.classes\"\n          bindLabel=\"name\"\n          bindValue=\"id\"\n          id=\"tile_classes\"\n        ></pry-select>\n      </div>\n    </pry-settings>\n  </pry-widget-header>\n  <div class=\"o-widget--tile-wrapper\" [style.top]=\"(displayHeader$ | async) ? WIDGET_HEADER_HEIGHT + 'px' : 0\">\n    <div class=\"no-result\" *ngIf=\"(items$ | async)?.length === 0\">\n      <img\n        class=\"no-result__search\"\n        src=\"../../../assets/svgs/pry_recherche_pas_de_resultat.svg\"\n        alt=\"{{ '@pry.widget.tile.empty' | i18n }}\"\n        aria-hidden=\"true\"\n      />\n      <span class=\"no-result__text\">{{ '@pry.widget.tile.empty' | i18n }}</span>\n    </div>\n    <ul class=\"o-widget--tile__list\">\n      <li\n        class=\"o-widget--tile__list__item\"\n        *ngFor=\"let item of items$ | async; let index = index\"\n        (contextmenu)=\"contextMenu($event, item)\"\n      >\n        <div class=\"o-widget--tile__list__item__header\">\n          <img\n            [height]=\"25\"\n            [width]=\"25\"\n            [src]=\"item | translateItemToSymbol | async\"\n            [alt]=\"item.oClass | translateId: { type: 'class', output: 'name' } | async\"\n          />\n          <h3 class=\"o-widget--tile__list__item__header__title\">\n            {{\n              item.oClass\n                | translateId\n                  : {\n                      type: 'class',\n                      output: 'name'\n                    }\n                | async\n            }}\n          </h3>\n          <pry-checkbox\n            [ngModel]=\"(selectedIds$ | async).includes(item.id)\"\n            (click)=\"select($event, item)\"\n          ></pry-checkbox>\n        </div>\n        <div class=\"o-properties\">\n          <ul class=\"o-properties__list\">\n            <ng-container *ngFor=\"let property of (propertiesToDisplay$ | async)![item.oClass]\">\n              <li\n                *ngFor=\"let value of getValue(item, property.prop); let i = index; let c = count\"\n                class=\"o-properties__list__item\"\n              >\n                <p class=\"o-properties__list__item__label\" *ngIf=\"i === 0\">{{ property.label }}:</p>\n                <p class=\"o-properties__list__item__value\" *ngIf=\"!isRaw(property.label, item.oClass)\">\n                  {{ value?.value }}\n                </p>\n                <p class=\"o-properties__list__item__raw\" *ngIf=\"isRaw(property.label, item.oClass)\">\n                  <pry-vizualize-raw [item]=\"item\" [propertyName]=\"property.label\"></pry-vizualize-raw>\n                </p>\n                <ng-container *ngIf=\"hasMetadata(value)\">\n                  <pry-metadata [item]=\"value\"></pry-metadata>\n                </ng-container>\n              </li>\n            </ng-container>\n          </ul>\n        </div>\n        <div (click)=\"toggleMap(index)\" class=\"o-widget--tile__list__item__footer\">\n          <button\n            type=\"button\"\n            class=\"a-btn a-btn--icon-text -link-like\"\n            aria-controls=\"tile-map-xxx\"\n            aria-expanded=\"false\"\n          >\n            {{ '@pry.widget.tile.map' | i18n }}\n          </button>\n        </div>\n        <div class=\"o-widget--tile__list__item__map\" *ngIf=\"maps[index]\" id=\"tile-map-xxx\">\n          <pry-widget-map [staticManifest]=\"mapManifest\" [staticResultSet]=\"resultSetWithItem(item)\"></pry-widget-map>\n        </div>\n      </li>\n    </ul>\n  </div>\n</div>\n"]}
@@ -2795,7 +2795,7 @@ class AdminSelectDatasetComponent extends SubscriptionnerDirective {
2795
2795
  }));
2796
2796
  }
2797
2797
  ngOnInit() {
2798
- this.subscriptions.add(this.selectedDataset$.subscribe((dataset) => {
2798
+ this.subscriptions.add(combineLatest([this.selectedDataset$, this.store.select(DataSourceSelectors.datasetVersions)]).subscribe(([dataset, versions]) => {
2799
2799
  if (dataset) {
2800
2800
  this.accessGroups = dataset?.groups ?? [];
2801
2801
  const editPath = this.router.createUrlTree(['.', 'dataset', 'edit', dataset.id], {
@@ -2824,7 +2824,8 @@ class AdminSelectDatasetComponent extends SubscriptionnerDirective {
2824
2824
  icon: 'delete',
2825
2825
  moduleAccess: 'admin',
2826
2826
  pageAccess: 'dataset',
2827
- actionAccess: 'write'
2827
+ actionAccess: 'write',
2828
+ disabled: versions.filter((v) => v.dataset === dataset.id).length > 0
2828
2829
  }
2829
2830
  ]
2830
2831
  }));