@theseam/ui-common 0.3.1 → 0.3.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.
- package/breadcrumbs/_breadcrumbs-theme.scss +3 -3
- package/breadcrumbs/breadcrumbs/breadcrumbs.component.scss +10 -10
- package/bundles/theseam-ui-common-form-field.umd.js +1 -1
- package/bundles/theseam-ui-common-form-field.umd.js.map +1 -1
- package/bundles/theseam-ui-common-google-maps.umd.js +2202 -0
- package/bundles/theseam-ui-common-google-maps.umd.js.map +1 -0
- package/bundles/theseam-ui-common-menu.umd.js +1 -0
- package/bundles/theseam-ui-common-menu.umd.js.map +1 -1
- package/bundles/theseam-ui-common-utils.umd.js +610 -136
- package/bundles/theseam-ui-common-utils.umd.js.map +1 -1
- package/esm2015/form-field/input.directive.js +2 -2
- package/esm2015/google-maps/google-maps/google-maps.component.js +261 -0
- package/esm2015/google-maps/google-maps-contextmenu.js +113 -0
- package/esm2015/google-maps/google-maps-controls.service.js +70 -0
- package/esm2015/google-maps/google-maps-feature-helpers.js +177 -0
- package/esm2015/google-maps/google-maps-places-autocomplete/google-maps-places-autocomplete.component.js +195 -0
- package/esm2015/google-maps/google-maps-places-autocomplete/google-maps-places-autocomplete.directive.js +163 -0
- package/esm2015/google-maps/google-maps-recenter-button-control/google-maps-recenter-button-control.component.js +57 -0
- package/esm2015/google-maps/google-maps-upload-button-control/google-maps-upload-button-control.component.js +119 -0
- package/esm2015/google-maps/google-maps.module.js +45 -0
- package/esm2015/google-maps/google-maps.service.js +344 -0
- package/esm2015/google-maps/map-control.component.js +65 -0
- package/esm2015/google-maps/map-controls-service.js +4 -0
- package/esm2015/google-maps/map-file-drop/map-file-drop.component.js +135 -0
- package/esm2015/google-maps/map-value-manager.service.js +46 -0
- package/esm2015/google-maps/public-api.js +14 -0
- package/esm2015/google-maps/theseam-ui-common-google-maps.js +6 -0
- package/esm2015/menu/menu-toggle.directive.js +2 -1
- package/esm2015/utils/geo-json/coerce-feature-collection.js +44 -0
- package/esm2015/utils/geo-json/geo-json-to-area.js +11 -0
- package/esm2015/utils/geo-json/is-feature-collection.validator.js +21 -0
- package/esm2015/utils/geo-json/is-only-geometry-types.js +23 -0
- package/esm2015/utils/geo-json/is-only-geometry-types.validator.js +32 -0
- package/esm2015/utils/geo-json/merge-polygons.js +35 -0
- package/esm2015/utils/geo-json/no-inner-rings.validator.js +63 -0
- package/esm2015/utils/geo-json/no-kinks.validator.js +39 -0
- package/esm2015/utils/geo-json/read-geo-file.js +99 -0
- package/esm2015/utils/geo-json/split-multi-polygons.js +29 -0
- package/esm2015/utils/is-null-or-undefined.js +1 -1
- package/esm2015/utils/public-api.js +11 -1
- package/fesm2015/theseam-ui-common-form-field.js +1 -1
- package/fesm2015/theseam-ui-common-form-field.js.map +1 -1
- package/fesm2015/theseam-ui-common-google-maps.js +1729 -0
- package/fesm2015/theseam-ui-common-google-maps.js.map +1 -0
- package/fesm2015/theseam-ui-common-menu.js +1 -0
- package/fesm2015/theseam-ui-common-menu.js.map +1 -1
- package/fesm2015/theseam-ui-common-utils.js +477 -94
- package/fesm2015/theseam-ui-common-utils.js.map +1 -1
- package/form-field/theseam-ui-common-form-field.metadata.json +1 -1
- package/framework/top-bar/_top-bar-theme.scss +5 -5
- package/google-maps/google-maps/google-maps.component.d.ts +89 -0
- package/google-maps/google-maps-contextmenu.d.ts +15 -0
- package/google-maps/google-maps-controls.service.d.ts +23 -0
- package/google-maps/google-maps-feature-helpers.d.ts +37 -0
- package/google-maps/google-maps-places-autocomplete/google-maps-places-autocomplete.component.d.ts +104 -0
- package/google-maps/google-maps-places-autocomplete/google-maps-places-autocomplete.directive.d.ts +80 -0
- package/google-maps/google-maps-recenter-button-control/google-maps-recenter-button-control.component.d.ts +21 -0
- package/google-maps/google-maps-upload-button-control/google-maps-upload-button-control.component.d.ts +34 -0
- package/google-maps/google-maps.module.d.ts +2 -0
- package/google-maps/google-maps.service.d.ts +53 -0
- package/google-maps/map-control.component.d.ts +20 -0
- package/google-maps/map-controls-service.d.ts +13 -0
- package/google-maps/map-file-drop/map-file-drop.component.d.ts +34 -0
- package/google-maps/map-value-manager.service.d.ts +18 -0
- package/google-maps/package.json +11 -0
- package/google-maps/public-api.d.ts +13 -0
- package/google-maps/theseam-ui-common-google-maps.d.ts +5 -0
- package/google-maps/theseam-ui-common-google-maps.metadata.json +1 -0
- package/package.json +17 -10
- package/utils/geo-json/coerce-feature-collection.d.ts +2 -0
- package/utils/geo-json/geo-json-to-area.d.ts +6 -0
- package/utils/geo-json/is-feature-collection.validator.d.ts +3 -0
- package/utils/geo-json/is-only-geometry-types.d.ts +5 -0
- package/utils/geo-json/is-only-geometry-types.validator.d.ts +4 -0
- package/utils/geo-json/merge-polygons.d.ts +9 -0
- package/utils/geo-json/no-inner-rings.validator.d.ts +10 -0
- package/utils/geo-json/no-kinks.validator.d.ts +3 -0
- package/utils/geo-json/read-geo-file.d.ts +7 -0
- package/utils/geo-json/split-multi-polygons.d.ts +8 -0
- package/utils/is-null-or-undefined.d.ts +1 -1
- package/utils/public-api.d.ts +10 -0
- package/utils/theseam-ui-common-utils.metadata.json +1 -1
|
@@ -0,0 +1,344 @@
|
|
|
1
|
+
import { __awaiter } from "tslib";
|
|
2
|
+
import { Injectable, NgZone, ViewContainerRef } from '@angular/core';
|
|
3
|
+
import { BehaviorSubject, from, Subject } from 'rxjs';
|
|
4
|
+
import { switchMap, takeUntil, tap } from 'rxjs/operators';
|
|
5
|
+
import { isNullOrUndefined, notNullOrUndefined } from '@theseam/ui-common/utils';
|
|
6
|
+
import { GoogleMapsContextMenu } from './google-maps-contextmenu';
|
|
7
|
+
import { addInnerFeatureCutoutToExteriorFeature, createDataFeatureFromPolygon, createFeatureChangeObservable, getBoundsWithAllFeatures, getFeatureCenter, getFeaturesCount, getPossibleExteriorFeature, isFeatureSelected, removeAllFeatures, setFeatureSelected, stripAppFeaturePropertiesFromJson, } from './google-maps-feature-helpers';
|
|
8
|
+
import { MapValueManagerService, MapValueSource } from './map-value-manager.service';
|
|
9
|
+
const DEFAULT_POLYGON_OPTIONS = {
|
|
10
|
+
clickable: true,
|
|
11
|
+
draggable: true,
|
|
12
|
+
editable: true,
|
|
13
|
+
// fillColor: 'rgba(60,150,60,1)',
|
|
14
|
+
// strokeColor: 'rgba(155,255,0,1)'
|
|
15
|
+
};
|
|
16
|
+
const DEFAULT_DRAWING_MANAGER_OPTIONS = () => ({
|
|
17
|
+
drawingControl: true,
|
|
18
|
+
drawingControlOptions: {
|
|
19
|
+
drawingModes: [
|
|
20
|
+
google.maps.drawing.OverlayType.POLYGON
|
|
21
|
+
]
|
|
22
|
+
},
|
|
23
|
+
polygonOptions: DEFAULT_POLYGON_OPTIONS,
|
|
24
|
+
drawingMode: null
|
|
25
|
+
});
|
|
26
|
+
const ɵ0 = DEFAULT_DRAWING_MANAGER_OPTIONS;
|
|
27
|
+
const FEATURE_STYLE_OPTIONS_DEFAULT = {
|
|
28
|
+
clickable: true,
|
|
29
|
+
visible: true,
|
|
30
|
+
// zIndex?: number;
|
|
31
|
+
// cursor?: string;
|
|
32
|
+
draggable: false,
|
|
33
|
+
editable: false,
|
|
34
|
+
fillColor: 'teal',
|
|
35
|
+
fillOpacity: 0.3,
|
|
36
|
+
strokeColor: 'blue',
|
|
37
|
+
strokeOpacity: 1,
|
|
38
|
+
strokeWeight: 2,
|
|
39
|
+
};
|
|
40
|
+
const FEATURE_STYLE_OPTIONS_SELECTED = Object.assign(Object.assign({}, FEATURE_STYLE_OPTIONS_DEFAULT), { draggable: true, editable: true, fillColor: 'green', fillOpacity: 0.7, strokeColor: 'limegreen', strokeOpacity: 1, strokeWeight: 2 });
|
|
41
|
+
export class GoogleMapsService {
|
|
42
|
+
constructor(_mapValueManager, _ngZone, _vcr) {
|
|
43
|
+
this._mapValueManager = _mapValueManager;
|
|
44
|
+
this._ngZone = _ngZone;
|
|
45
|
+
this._vcr = _vcr;
|
|
46
|
+
this._ngUnsubscribe = new Subject();
|
|
47
|
+
this._mapReadySubject = new BehaviorSubject(false);
|
|
48
|
+
this._featureContextMenu = null;
|
|
49
|
+
this._activeContextMenu = null;
|
|
50
|
+
this._allowDrawingHoleInPolygon = false;
|
|
51
|
+
this.mapReady$ = this._mapReadySubject.asObservable();
|
|
52
|
+
}
|
|
53
|
+
get mapReady() { return this._mapReadySubject.value; }
|
|
54
|
+
ngOnDestroy() {
|
|
55
|
+
this._ngUnsubscribe.next();
|
|
56
|
+
this._ngUnsubscribe.complete();
|
|
57
|
+
}
|
|
58
|
+
setMap(map) {
|
|
59
|
+
this.googleMap = map;
|
|
60
|
+
this._mapReadySubject.next(true);
|
|
61
|
+
this._initDrawingManager();
|
|
62
|
+
this._initFeatureStyling();
|
|
63
|
+
this._initFeatureChangeListeners();
|
|
64
|
+
}
|
|
65
|
+
setBaseLatLng(lat, lng) {
|
|
66
|
+
this._baseLatLng = { lat, lng };
|
|
67
|
+
}
|
|
68
|
+
// TODO: Refactor out of the service meant to just wrap the google maps api.
|
|
69
|
+
setFeatureContextMenu(menu) {
|
|
70
|
+
this._featureContextMenu = menu;
|
|
71
|
+
}
|
|
72
|
+
getDiv() {
|
|
73
|
+
this._assertInitialized();
|
|
74
|
+
return this.googleMap.getDiv();
|
|
75
|
+
}
|
|
76
|
+
fitBounds(bounds, padding) {
|
|
77
|
+
this._assertInitialized();
|
|
78
|
+
this.googleMap.fitBounds(bounds, padding);
|
|
79
|
+
}
|
|
80
|
+
/**
|
|
81
|
+
* Iterates the map's features and removes any that are selected.
|
|
82
|
+
*/
|
|
83
|
+
deleteSelection() {
|
|
84
|
+
this._assertInitialized();
|
|
85
|
+
const mapData = this.googleMap.data;
|
|
86
|
+
mapData.forEach(f => {
|
|
87
|
+
if (isFeatureSelected(f)) {
|
|
88
|
+
mapData.remove(f);
|
|
89
|
+
}
|
|
90
|
+
});
|
|
91
|
+
}
|
|
92
|
+
/**
|
|
93
|
+
* Stops the current drawing.
|
|
94
|
+
*/
|
|
95
|
+
stopDrawing() {
|
|
96
|
+
this._ngZone.runOutsideAngular(() => {
|
|
97
|
+
if (isNullOrUndefined(this._drawingManager) || this._drawingManager.getDrawingMode() === null) {
|
|
98
|
+
return;
|
|
99
|
+
}
|
|
100
|
+
// Listening for the completion event of the overlay currently being drawn
|
|
101
|
+
// and removing it. I haven't found a way to cancel the current drawing,
|
|
102
|
+
// without this hack that assumes our listeners will not run into a race
|
|
103
|
+
// condition.
|
|
104
|
+
const listener = google.maps.event.addListener(this._drawingManager, 'overlaycomplete', (event) => {
|
|
105
|
+
event.overlay.setMap(null);
|
|
106
|
+
listener.remove();
|
|
107
|
+
});
|
|
108
|
+
// To fake canceling the current drawing, without disabling the drawing
|
|
109
|
+
// mode, the drawin mode is being unset then immediately set back. When
|
|
110
|
+
// the mode is unset the 'overlaycomplete' event will fire, which will
|
|
111
|
+
// give a reference to the current overlay to remove, then it is set back
|
|
112
|
+
// to the mode the user was using. To the user is should just seem like
|
|
113
|
+
// the current drawing was canceled and they can start drawing again.
|
|
114
|
+
const mode = this._drawingManager.getDrawingMode();
|
|
115
|
+
this._drawingManager.setDrawingMode(null);
|
|
116
|
+
this._drawingManager.setDrawingMode(mode);
|
|
117
|
+
// 'overlaycomplete' should fire immediately, unless an overlay hadn't
|
|
118
|
+
// started drawing. This timeout will make sure the listener gets removed.
|
|
119
|
+
setTimeout(() => {
|
|
120
|
+
listener.remove();
|
|
121
|
+
});
|
|
122
|
+
});
|
|
123
|
+
}
|
|
124
|
+
_initDrawingManager() {
|
|
125
|
+
if (notNullOrUndefined(this._drawingManager)) {
|
|
126
|
+
throw Error(`DrawingManager is already initialized.`);
|
|
127
|
+
}
|
|
128
|
+
this._assertInitialized();
|
|
129
|
+
const options = DEFAULT_DRAWING_MANAGER_OPTIONS();
|
|
130
|
+
const drawingManager = new google.maps.drawing.DrawingManager(options);
|
|
131
|
+
drawingManager.setMap(this.googleMap);
|
|
132
|
+
this._drawingManager = drawingManager;
|
|
133
|
+
this._drawingManager.addListener('drawingmode_changed', event => {
|
|
134
|
+
var _a;
|
|
135
|
+
if (((_a = this._drawingManager) === null || _a === void 0 ? void 0 : _a.getDrawingMode()) !== null) {
|
|
136
|
+
this._assertInitialized();
|
|
137
|
+
this.googleMap.data.forEach(f => {
|
|
138
|
+
if (isFeatureSelected(f)) {
|
|
139
|
+
setFeatureSelected(f, false);
|
|
140
|
+
}
|
|
141
|
+
});
|
|
142
|
+
}
|
|
143
|
+
});
|
|
144
|
+
}
|
|
145
|
+
addControl(element, position) {
|
|
146
|
+
this._assertInitialized();
|
|
147
|
+
this.googleMap.controls[position].push(element);
|
|
148
|
+
}
|
|
149
|
+
setData(data) {
|
|
150
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
151
|
+
this._assertInitialized();
|
|
152
|
+
removeAllFeatures(this.googleMap.data);
|
|
153
|
+
this.googleMap.data.addGeoJson(data);
|
|
154
|
+
this.googleMap.fitBounds(getBoundsWithAllFeatures(this.googleMap.data));
|
|
155
|
+
});
|
|
156
|
+
}
|
|
157
|
+
// TODO: Refactor out of the service meant to just wrap the google maps api.
|
|
158
|
+
reCenterOnFeatures() {
|
|
159
|
+
this._assertInitialized();
|
|
160
|
+
if (getFeaturesCount(this.googleMap.data) === 0) {
|
|
161
|
+
if (!this._baseLatLng) {
|
|
162
|
+
return;
|
|
163
|
+
}
|
|
164
|
+
this.googleMap.panTo(this._baseLatLng);
|
|
165
|
+
return;
|
|
166
|
+
}
|
|
167
|
+
this.googleMap.fitBounds(getBoundsWithAllFeatures(this.googleMap.data));
|
|
168
|
+
// TODO: Fix to pan to center. Currently fitBounds results in the expected
|
|
169
|
+
// result, but pantToBounds animates.
|
|
170
|
+
// this.googleMap.panToBounds(getBoundsWithAllFeatures(this.googleMap.data))
|
|
171
|
+
}
|
|
172
|
+
allowDrawingHoleInPolygon(allow) {
|
|
173
|
+
this._allowDrawingHoleInPolygon = allow;
|
|
174
|
+
}
|
|
175
|
+
setFileInputHandler(handler) {
|
|
176
|
+
this._fileInputHandler;
|
|
177
|
+
}
|
|
178
|
+
getFileInputHandler() {
|
|
179
|
+
return this._fileInputHandler;
|
|
180
|
+
}
|
|
181
|
+
_initFeatureStyling() {
|
|
182
|
+
this._assertInitialized();
|
|
183
|
+
// Disable any selection when clicking the map.
|
|
184
|
+
//
|
|
185
|
+
// TODO: There may be a better way to do this that would be more accurate or
|
|
186
|
+
// additional events that should be listened to, such as the disabling
|
|
187
|
+
// selection when the map looses focus.
|
|
188
|
+
this.googleMap.addListener('click', (even) => {
|
|
189
|
+
this._assertInitialized();
|
|
190
|
+
this.googleMap.data.forEach(f => setFeatureSelected(f, false));
|
|
191
|
+
});
|
|
192
|
+
// Determine what the style of the features are.
|
|
193
|
+
this.googleMap.data.setStyle((feature) => {
|
|
194
|
+
if (isFeatureSelected(feature)) {
|
|
195
|
+
return FEATURE_STYLE_OPTIONS_SELECTED;
|
|
196
|
+
}
|
|
197
|
+
return FEATURE_STYLE_OPTIONS_DEFAULT;
|
|
198
|
+
});
|
|
199
|
+
// Select a feature when clicked.
|
|
200
|
+
this.googleMap.data.addListener('click', (event) => {
|
|
201
|
+
this._assertInitialized();
|
|
202
|
+
setFeatureSelected(event.feature, true);
|
|
203
|
+
this.googleMap.data.forEach(f => {
|
|
204
|
+
if (f !== event.feature && isFeatureSelected(f)) {
|
|
205
|
+
setFeatureSelected(f, false);
|
|
206
|
+
}
|
|
207
|
+
});
|
|
208
|
+
});
|
|
209
|
+
// Set a style on hovered features that can be selected.
|
|
210
|
+
this.googleMap.data.addListener('mouseover', (event) => {
|
|
211
|
+
this._assertInitialized();
|
|
212
|
+
this.googleMap.data.revertStyle();
|
|
213
|
+
if (!this.isDrawing() && !isFeatureSelected(event.feature)) {
|
|
214
|
+
this.googleMap.data.overrideStyle(event.feature, { strokeWeight: 4 });
|
|
215
|
+
}
|
|
216
|
+
});
|
|
217
|
+
// Remove any hover styles when mouse moves away.
|
|
218
|
+
this.googleMap.data.addListener('mouseout', (event) => {
|
|
219
|
+
this._assertInitialized();
|
|
220
|
+
this.googleMap.data.revertStyle();
|
|
221
|
+
});
|
|
222
|
+
}
|
|
223
|
+
_initFeatureChangeListeners() {
|
|
224
|
+
this._assertInitialized();
|
|
225
|
+
createFeatureChangeObservable(this.googleMap.data, this._ngZone).pipe(switchMap(() => from(this.getGeoJson()).pipe(tap(geoJson => this._mapValueManager.setValue(geoJson, MapValueSource.FeatureChange)))), takeUntil(this._ngUnsubscribe)).subscribe();
|
|
226
|
+
this.googleMap.data.addListener('contextmenu', (event) => {
|
|
227
|
+
if (!isFeatureSelected(event.feature)) {
|
|
228
|
+
return;
|
|
229
|
+
}
|
|
230
|
+
this._openContextMenuForFeature(event.feature, event.latLng);
|
|
231
|
+
});
|
|
232
|
+
if (notNullOrUndefined(this._drawingManager)) {
|
|
233
|
+
google.maps.event.addListener(this._drawingManager, 'polygoncomplete', (polygon) => {
|
|
234
|
+
var _a;
|
|
235
|
+
// The DrawingManager doesn't seem to have a way to access the overlays,
|
|
236
|
+
// so if the map is not set then it shouldn'y be considered a successful
|
|
237
|
+
// completion. I am canceling the active drawing by disabling drawing
|
|
238
|
+
// mode and setting the map null in the 'overlaycomplete' event, which
|
|
239
|
+
// fires before the 'polygoncomplete' event.
|
|
240
|
+
if (isNullOrUndefined(polygon.getMap())) {
|
|
241
|
+
return;
|
|
242
|
+
}
|
|
243
|
+
this._assertInitialized();
|
|
244
|
+
// Create a map feature of the drawn polygon.
|
|
245
|
+
const feature = createDataFeatureFromPolygon(polygon);
|
|
246
|
+
// Remove the drawn polygon.
|
|
247
|
+
polygon.setMap(null);
|
|
248
|
+
// Stop drawing.
|
|
249
|
+
(_a = this._drawingManager) === null || _a === void 0 ? void 0 : _a.setDrawingMode(null);
|
|
250
|
+
// Check if the feature should be used as a cutout to an existing
|
|
251
|
+
// feature or added as it's own feature.
|
|
252
|
+
const exteriorPolygonFeature = this._allowDrawingHoleInPolygon
|
|
253
|
+
? getPossibleExteriorFeature(this.googleMap.data, feature)
|
|
254
|
+
: undefined;
|
|
255
|
+
if (exteriorPolygonFeature) {
|
|
256
|
+
addInnerFeatureCutoutToExteriorFeature(exteriorPolygonFeature, feature);
|
|
257
|
+
setFeatureSelected(exteriorPolygonFeature, true);
|
|
258
|
+
}
|
|
259
|
+
else {
|
|
260
|
+
this.googleMap.data.add(feature);
|
|
261
|
+
setFeatureSelected(feature, true);
|
|
262
|
+
}
|
|
263
|
+
});
|
|
264
|
+
}
|
|
265
|
+
}
|
|
266
|
+
isDrawing() {
|
|
267
|
+
if (isNullOrUndefined(this._drawingManager)) {
|
|
268
|
+
return true;
|
|
269
|
+
}
|
|
270
|
+
return this._drawingManager.getDrawingMode() !== null;
|
|
271
|
+
}
|
|
272
|
+
// TODO: Refactor out of the service meant to just wrap the google maps api.
|
|
273
|
+
hasSelectedFeature() {
|
|
274
|
+
this._assertInitialized();
|
|
275
|
+
let isSelected = false;
|
|
276
|
+
this.googleMap.data.forEach(f => {
|
|
277
|
+
if (isFeatureSelected(f)) {
|
|
278
|
+
isSelected = true;
|
|
279
|
+
}
|
|
280
|
+
});
|
|
281
|
+
return isSelected;
|
|
282
|
+
}
|
|
283
|
+
// TODO: Refactor out of the service meant to just wrap the google maps api.
|
|
284
|
+
getSelectedFeature() {
|
|
285
|
+
this._assertInitialized();
|
|
286
|
+
let feature = null;
|
|
287
|
+
this.googleMap.data.forEach(f => {
|
|
288
|
+
if (isFeatureSelected(f)) {
|
|
289
|
+
feature = f;
|
|
290
|
+
}
|
|
291
|
+
});
|
|
292
|
+
return feature;
|
|
293
|
+
}
|
|
294
|
+
// TODO: Refactor out of the service meant to just wrap the google maps api.
|
|
295
|
+
openContextMenu() {
|
|
296
|
+
const feature = this.getSelectedFeature();
|
|
297
|
+
if (feature) {
|
|
298
|
+
this._openContextMenuForFeature(feature);
|
|
299
|
+
}
|
|
300
|
+
}
|
|
301
|
+
// TODO: Refactor out of the service meant to just wrap the google maps api.
|
|
302
|
+
_openContextMenuForFeature(feature, position) {
|
|
303
|
+
if (this._activeContextMenu) {
|
|
304
|
+
this._activeContextMenu.close();
|
|
305
|
+
this._activeContextMenu = null;
|
|
306
|
+
}
|
|
307
|
+
this._assertInitialized();
|
|
308
|
+
let _position = position;
|
|
309
|
+
if (!_position) {
|
|
310
|
+
_position = getFeatureCenter(feature);
|
|
311
|
+
}
|
|
312
|
+
if (this._featureContextMenu) {
|
|
313
|
+
this._activeContextMenu = new GoogleMapsContextMenu(this.googleMap, this._featureContextMenu, _position, this._vcr, this._ngZone, this.googleMap.data, feature);
|
|
314
|
+
}
|
|
315
|
+
}
|
|
316
|
+
getGeoJson(removeAppProperties = true) {
|
|
317
|
+
return new Promise((resolve, reject) => {
|
|
318
|
+
this._assertInitialized();
|
|
319
|
+
this.googleMap.data.toGeoJson(f => {
|
|
320
|
+
if (removeAppProperties) {
|
|
321
|
+
stripAppFeaturePropertiesFromJson(f);
|
|
322
|
+
}
|
|
323
|
+
resolve(f);
|
|
324
|
+
});
|
|
325
|
+
});
|
|
326
|
+
}
|
|
327
|
+
/** Asserts that the map has been initialized. */
|
|
328
|
+
_assertInitialized() {
|
|
329
|
+
if (!this.googleMap && (typeof ngDevMode === 'undefined' || ngDevMode)) {
|
|
330
|
+
throw Error('Cannot access Google Map information before the API has been initialized. ' +
|
|
331
|
+
'Please wait for the API to load before trying to interact with it.');
|
|
332
|
+
}
|
|
333
|
+
}
|
|
334
|
+
}
|
|
335
|
+
GoogleMapsService.decorators = [
|
|
336
|
+
{ type: Injectable }
|
|
337
|
+
];
|
|
338
|
+
GoogleMapsService.ctorParameters = () => [
|
|
339
|
+
{ type: MapValueManagerService },
|
|
340
|
+
{ type: NgZone },
|
|
341
|
+
{ type: ViewContainerRef }
|
|
342
|
+
];
|
|
343
|
+
export { ɵ0 };
|
|
344
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"google-maps.service.js","sourceRoot":"","sources":["../../../../projects/ui-common/google-maps/google-maps.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, NgZone, OnDestroy, ViewContainerRef } from '@angular/core'\nimport { BehaviorSubject, from, Observable, Subject } from 'rxjs'\nimport { switchMap, takeUntil, tap } from 'rxjs/operators'\n\nimport { MenuComponent } from '@theseam/ui-common/menu'\nimport { isNullOrUndefined, notNullOrUndefined } from '@theseam/ui-common/utils'\n\nimport { GoogleMapsContextMenu } from './google-maps-contextmenu'\nimport {\n  addInnerFeatureCutoutToExteriorFeature,\n  createDataFeatureFromPolygon,\n  createFeatureChangeObservable,\n  getBoundsWithAllFeatures,\n  getFeatureCenter,\n  getFeaturesCount,\n  getPossibleExteriorFeature,\n  isFeatureSelected,\n  removeAllFeatures,\n  setFeatureSelected,\n  stripAppFeaturePropertiesFromJson,\n} from './google-maps-feature-helpers'\nimport { MapValueManagerService, MapValueSource } from './map-value-manager.service'\n\ndeclare const ngDevMode: boolean | undefined\n\nconst DEFAULT_POLYGON_OPTIONS: google.maps.PolygonOptions = {\n  clickable: true,\n  draggable: true,\n  editable: true,\n  // fillColor: 'rgba(60,150,60,1)',\n  // strokeColor: 'rgba(155,255,0,1)'\n}\n\nconst DEFAULT_DRAWING_MANAGER_OPTIONS: (() => google.maps.drawing.DrawingManagerOptions) = () => ({\n  drawingControl: true,\n  drawingControlOptions: {\n    drawingModes: [\n      google.maps.drawing.OverlayType.POLYGON\n    ]\n  },\n  polygonOptions: DEFAULT_POLYGON_OPTIONS,\n  drawingMode: null\n})\n\nconst FEATURE_STYLE_OPTIONS_DEFAULT: google.maps.Data.StyleOptions = {\n  clickable: true,\n  visible: true,\n  // zIndex?: number;\n\n  // cursor?: string;\n  draggable: false,\n  editable: false,\n  fillColor: 'teal',\n  fillOpacity: 0.3,\n  strokeColor: 'blue',\n  strokeOpacity: 1,\n  strokeWeight: 2,\n}\n\nconst FEATURE_STYLE_OPTIONS_SELECTED: google.maps.Data.StyleOptions = {\n  ...FEATURE_STYLE_OPTIONS_DEFAULT,\n  draggable: true,\n  editable: true,\n  fillColor: 'green',\n  fillOpacity: 0.7,\n  strokeColor: 'limegreen',\n  strokeOpacity: 1,\n  strokeWeight: 2,\n}\n\ntype WithRequired<T, K extends keyof T> = T & { [P in K]-?: T[P] }\n\n@Injectable()\nexport class GoogleMapsService implements OnDestroy {\n  private readonly _ngUnsubscribe = new Subject<void>()\n\n  private readonly _mapReadySubject = new BehaviorSubject<boolean>(false)\n\n  private _drawingManager?: google.maps.drawing.DrawingManager\n  private _featureContextMenu: MenuComponent | null = null\n  private _activeContextMenu: GoogleMapsContextMenu | null = null\n  private _baseLatLng?: google.maps.LatLngLiteral\n\n  private _allowDrawingHoleInPolygon: boolean = false\n\n  // TODO: Move to a better place than the map wrapper service.\n  private _fileInputHandler: ((file: File) => void) | undefined | null\n\n  public googleMap?: google.maps.Map\n\n  public readonly mapReady$: Observable<boolean>\n\n  public get mapReady(): boolean { return this._mapReadySubject.value }\n\n  constructor(\n    private readonly _mapValueManager: MapValueManagerService,\n    private readonly _ngZone: NgZone,\n    private readonly _vcr: ViewContainerRef,\n  ) {\n    this.mapReady$ = this._mapReadySubject.asObservable()\n  }\n\n  ngOnDestroy(): void {\n    this._ngUnsubscribe.next()\n    this._ngUnsubscribe.complete()\n  }\n\n  public setMap(map: google.maps.Map): void {\n    this.googleMap = map\n    this._mapReadySubject.next(true)\n    this._initDrawingManager()\n    this._initFeatureStyling()\n    this._initFeatureChangeListeners()\n  }\n\n  public setBaseLatLng(lat: number, lng: number): void {\n    this._baseLatLng = { lat, lng }\n  }\n\n  // TODO: Refactor out of the service meant to just wrap the google maps api.\n  public setFeatureContextMenu(menu: MenuComponent | null): void {\n    this._featureContextMenu = menu\n  }\n\n  public getDiv(): HTMLDivElement {\n    this._assertInitialized()\n    return this.googleMap.getDiv() as HTMLDivElement\n  }\n\n  public fitBounds(bounds: google.maps.LatLngBounds | google.maps.LatLngBoundsLiteral, padding?: number | google.maps.Padding): void {\n    this._assertInitialized()\n    this.googleMap.fitBounds(bounds, padding)\n  }\n\n  /**\n   * Iterates the map's features and removes any that are selected.\n   */\n  public deleteSelection(): void {\n    this._assertInitialized()\n    const mapData = this.googleMap.data\n    mapData.forEach(f => {\n      if (isFeatureSelected(f)) {\n        mapData.remove(f)\n      }\n    })\n  }\n\n  /**\n   * Stops the current drawing.\n   */\n  public stopDrawing(): void {\n    this._ngZone.runOutsideAngular(() => {\n      if (isNullOrUndefined(this._drawingManager) || this._drawingManager.getDrawingMode() === null) {\n        return\n      }\n\n      // Listening for the completion event of the overlay currently being drawn\n      // and removing it. I haven't found a way to cancel the current drawing,\n      // without this hack that assumes our listeners will not run into a race\n      // condition.\n      const listener = google.maps.event.addListener(\n        this._drawingManager,\n        'overlaycomplete',\n        (event: google.maps.drawing.OverlayCompleteEvent) => {\n          event.overlay.setMap(null)\n          listener.remove()\n        }\n      )\n\n      // To fake canceling the current drawing, without disabling the drawing\n      // mode, the drawin mode is being unset then immediately set back. When\n      // the mode is unset the 'overlaycomplete' event will fire, which will\n      // give a reference to the current overlay to remove, then it is set back\n      // to the mode the user was using. To the user is should just seem like\n      // the current drawing was canceled and they can start drawing again.\n      const mode = this._drawingManager.getDrawingMode()\n      this._drawingManager.setDrawingMode(null)\n      this._drawingManager.setDrawingMode(mode)\n\n      // 'overlaycomplete' should fire immediately, unless an overlay hadn't\n      // started drawing. This timeout will make sure the listener gets removed.\n      setTimeout(() => {\n        listener.remove()\n      })\n\n    })\n  }\n\n  private _initDrawingManager(): void {\n    if (notNullOrUndefined(this._drawingManager)) {\n      throw Error(`DrawingManager is already initialized.`)\n    }\n\n    this._assertInitialized()\n\n    const options = DEFAULT_DRAWING_MANAGER_OPTIONS()\n\n    const drawingManager = new google.maps.drawing.DrawingManager(options)\n    drawingManager.setMap(this.googleMap)\n\n    this._drawingManager = drawingManager\n\n    this._drawingManager.addListener('drawingmode_changed', event => {\n      if (this._drawingManager?.getDrawingMode() !== null) {\n        this._assertInitialized()\n        this.googleMap.data.forEach(f => {\n          if (isFeatureSelected(f)) {\n            setFeatureSelected(f, false)\n          }\n        })\n      }\n    })\n  }\n\n  public addControl(element: HTMLElement, position: google.maps.ControlPosition): void {\n    this._assertInitialized()\n    this.googleMap.controls[position].push(element)\n  }\n\n  public async setData(data: any): Promise<void> {\n    this._assertInitialized()\n    removeAllFeatures(this.googleMap.data)\n    this.googleMap.data.addGeoJson(data)\n    this.googleMap.fitBounds(getBoundsWithAllFeatures(this.googleMap.data))\n  }\n\n  // TODO: Refactor out of the service meant to just wrap the google maps api.\n  public reCenterOnFeatures(): void {\n    this._assertInitialized()\n    if (getFeaturesCount(this.googleMap.data) === 0) {\n      if (!this._baseLatLng) {\n        return\n      }\n\n      this.googleMap.panTo(this._baseLatLng)\n      return\n    }\n    this.googleMap.fitBounds(getBoundsWithAllFeatures(this.googleMap.data))\n\n    // TODO: Fix to pan to center. Currently fitBounds results in the expected\n    // result, but pantToBounds animates.\n    // this.googleMap.panToBounds(getBoundsWithAllFeatures(this.googleMap.data))\n  }\n\n  public allowDrawingHoleInPolygon(allow: boolean): void {\n    this._allowDrawingHoleInPolygon = allow\n  }\n\n  public setFileInputHandler(handler: ((file: File) => void) | undefined | null): void {\n    this._fileInputHandler\n  }\n\n  public getFileInputHandler(): ((file: File) => void) | undefined | null {\n    return this._fileInputHandler\n  }\n\n  private _initFeatureStyling(): void {\n    this._assertInitialized()\n\n    // Disable any selection when clicking the map.\n    //\n    // TODO: There may be a better way to do this that would be more accurate or\n    // additional events that should be listened to, such as the disabling\n    // selection when the map looses focus.\n    this.googleMap.addListener('click', (even: google.maps.MapMouseEvent | google.maps.IconMouseEvent) => {\n      this._assertInitialized()\n      this.googleMap.data.forEach(f => setFeatureSelected(f, false))\n    })\n\n    // Determine what the style of the features are.\n    this.googleMap.data.setStyle((feature) => {\n      if (isFeatureSelected(feature)) {\n        return FEATURE_STYLE_OPTIONS_SELECTED\n      }\n\n      return FEATURE_STYLE_OPTIONS_DEFAULT\n    })\n\n    // Select a feature when clicked.\n    this.googleMap.data.addListener('click', (event: google.maps.Data.MouseEvent) => {\n      this._assertInitialized()\n\n      setFeatureSelected(event.feature, true)\n      this.googleMap.data.forEach(f => {\n        if (f !== event.feature && isFeatureSelected(f)) {\n          setFeatureSelected(f, false)\n        }\n      })\n    })\n\n    // Set a style on hovered features that can be selected.\n    this.googleMap.data.addListener('mouseover', (event: google.maps.Data.MouseEvent) => {\n      this._assertInitialized()\n      this.googleMap.data.revertStyle()\n      if (!this.isDrawing() && !isFeatureSelected(event.feature)) {\n        this.googleMap.data.overrideStyle(event.feature, { strokeWeight: 4 })\n      }\n    })\n\n    // Remove any hover styles when mouse moves away.\n    this.googleMap.data.addListener('mouseout', (event: google.maps.Data.MouseEvent) => {\n      this._assertInitialized()\n      this.googleMap.data.revertStyle()\n    })\n  }\n\n  private _initFeatureChangeListeners(): void {\n    this._assertInitialized()\n\n    createFeatureChangeObservable(this.googleMap.data, this._ngZone).pipe(\n      switchMap(() => from(this.getGeoJson()).pipe(\n        tap(geoJson => this._mapValueManager.setValue(geoJson, MapValueSource.FeatureChange)),\n      )),\n      takeUntil(this._ngUnsubscribe),\n    ).subscribe()\n\n    this.googleMap.data.addListener('contextmenu', (event: google.maps.Data.MouseEvent) => {\n      if (!isFeatureSelected(event.feature)) {\n        return\n      }\n\n      this._openContextMenuForFeature(event.feature, event.latLng)\n    })\n\n    if (notNullOrUndefined(this._drawingManager)) {\n      google.maps.event.addListener(this._drawingManager, 'polygoncomplete', (polygon: google.maps.Polygon) => {\n        // The DrawingManager doesn't seem to have a way to access the overlays,\n        // so if the map is not set then it shouldn'y be considered a successful\n        // completion. I am canceling the active drawing by disabling drawing\n        // mode and setting the map null in the 'overlaycomplete' event, which\n        // fires before the 'polygoncomplete' event.\n        if (isNullOrUndefined(polygon.getMap())) {\n          return\n        }\n\n        this._assertInitialized()\n\n        // Create a map feature of the drawn polygon.\n        const feature = createDataFeatureFromPolygon(polygon)\n        // Remove the drawn polygon.\n        polygon.setMap(null)\n\n        // Stop drawing.\n        this._drawingManager?.setDrawingMode(null)\n\n        // Check if the feature should be used as a cutout to an existing\n        // feature or added as it's own feature.\n        const exteriorPolygonFeature = this._allowDrawingHoleInPolygon\n          ? getPossibleExteriorFeature(this.googleMap.data, feature)\n          : undefined\n        if (exteriorPolygonFeature) {\n          addInnerFeatureCutoutToExteriorFeature(exteriorPolygonFeature, feature)\n          setFeatureSelected(exteriorPolygonFeature, true)\n        } else {\n          this.googleMap.data.add(feature)\n          setFeatureSelected(feature, true)\n        }\n      })\n    }\n  }\n\n  public isDrawing(): boolean {\n    if (isNullOrUndefined(this._drawingManager)) {\n      return true\n    }\n\n    return this._drawingManager.getDrawingMode() !== null\n  }\n\n  // TODO: Refactor out of the service meant to just wrap the google maps api.\n  public hasSelectedFeature(): boolean {\n    this._assertInitialized()\n    let isSelected = false\n    this.googleMap.data.forEach(f => {\n      if (isFeatureSelected(f)) {\n        isSelected = true\n      }\n    })\n    return isSelected\n  }\n\n  // TODO: Refactor out of the service meant to just wrap the google maps api.\n  public getSelectedFeature(): google.maps.Data.Feature | null {\n    this._assertInitialized()\n    let feature: google.maps.Data.Feature | null = null\n    this.googleMap.data.forEach(f => {\n      if (isFeatureSelected(f)) {\n        feature = f\n      }\n    })\n    return feature\n  }\n\n  // TODO: Refactor out of the service meant to just wrap the google maps api.\n  public openContextMenu(): void {\n    const feature = this.getSelectedFeature()\n    if (feature) {\n      this._openContextMenuForFeature(feature)\n    }\n  }\n\n  // TODO: Refactor out of the service meant to just wrap the google maps api.\n  private _openContextMenuForFeature(feature: google.maps.Data.Feature, position?: google.maps.LatLng) {\n    if (this._activeContextMenu) {\n      this._activeContextMenu.close()\n      this._activeContextMenu = null\n    }\n\n    this._assertInitialized()\n\n    let _position = position\n    if (!_position) {\n      _position = getFeatureCenter(feature)\n    }\n    if (this._featureContextMenu) {\n      this._activeContextMenu = new GoogleMapsContextMenu(\n        this.googleMap,\n        this._featureContextMenu,\n        _position,\n        this._vcr,\n        this._ngZone,\n        this.googleMap.data,\n        feature,\n      )\n    }\n  }\n\n  public getGeoJson(removeAppProperties: boolean = true): Promise<object> {\n    return new Promise((resolve, reject) => {\n      this._assertInitialized()\n      this.googleMap.data.toGeoJson(f => {\n        if (removeAppProperties) {\n          stripAppFeaturePropertiesFromJson(f)\n        }\n        resolve(f)\n      })\n    })\n  }\n\n  /** Asserts that the map has been initialized. */\n  private _assertInitialized(): asserts this is WithRequired<GoogleMapsService, 'googleMap'> {\n    if (!this.googleMap && (typeof ngDevMode === 'undefined' || ngDevMode)) {\n      throw Error(\n        'Cannot access Google Map information before the API has been initialized. ' +\n          'Please wait for the API to load before trying to interact with it.',\n      )\n    }\n  }\n}\n"]}
|
|
@@ -0,0 +1,65 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, Component, Inject, Input, } from '@angular/core';
|
|
2
|
+
import { Subject } from 'rxjs';
|
|
3
|
+
import { takeUntil, tap } from 'rxjs/operators';
|
|
4
|
+
import { GoogleMapsControlsService } from './google-maps-controls.service';
|
|
5
|
+
import { GoogleMapsService } from './google-maps.service';
|
|
6
|
+
import { MAP_CONTROLS_SERVICE } from './map-controls-service';
|
|
7
|
+
export class TheSeamMapsControlComponent {
|
|
8
|
+
constructor(_googleMaps, _googleMapsControls) {
|
|
9
|
+
this._googleMaps = _googleMaps;
|
|
10
|
+
this._googleMapsControls = _googleMapsControls;
|
|
11
|
+
this._ngUnsubscribe = new Subject();
|
|
12
|
+
}
|
|
13
|
+
set def(value) {
|
|
14
|
+
if (value !== this._def) {
|
|
15
|
+
this._remove();
|
|
16
|
+
this._def = value;
|
|
17
|
+
}
|
|
18
|
+
}
|
|
19
|
+
/** @ignore */
|
|
20
|
+
ngOnInit() {
|
|
21
|
+
this._googleMaps.mapReady$.pipe(tap(ready => {
|
|
22
|
+
if (ready) {
|
|
23
|
+
this._add();
|
|
24
|
+
}
|
|
25
|
+
else {
|
|
26
|
+
this._remove();
|
|
27
|
+
}
|
|
28
|
+
}), takeUntil(this._ngUnsubscribe)).subscribe();
|
|
29
|
+
}
|
|
30
|
+
/** @ignore */
|
|
31
|
+
ngOnDestroy() {
|
|
32
|
+
this._remove();
|
|
33
|
+
this._ngUnsubscribe.next();
|
|
34
|
+
this._ngUnsubscribe.complete();
|
|
35
|
+
}
|
|
36
|
+
_add() {
|
|
37
|
+
if (this._controlRef) {
|
|
38
|
+
return;
|
|
39
|
+
}
|
|
40
|
+
if (this._def === null || this._def === undefined) {
|
|
41
|
+
return;
|
|
42
|
+
}
|
|
43
|
+
this._controlRef = this._googleMapsControls.add(this._def);
|
|
44
|
+
}
|
|
45
|
+
_remove() {
|
|
46
|
+
var _a;
|
|
47
|
+
(_a = this._controlRef) === null || _a === void 0 ? void 0 : _a.destroy();
|
|
48
|
+
this._controlRef = undefined;
|
|
49
|
+
}
|
|
50
|
+
}
|
|
51
|
+
TheSeamMapsControlComponent.decorators = [
|
|
52
|
+
{ type: Component, args: [{
|
|
53
|
+
selector: 'seam-map-control',
|
|
54
|
+
template: '',
|
|
55
|
+
changeDetection: ChangeDetectionStrategy.OnPush
|
|
56
|
+
},] }
|
|
57
|
+
];
|
|
58
|
+
TheSeamMapsControlComponent.ctorParameters = () => [
|
|
59
|
+
{ type: GoogleMapsService },
|
|
60
|
+
{ type: GoogleMapsControlsService, decorators: [{ type: Inject, args: [MAP_CONTROLS_SERVICE,] }] }
|
|
61
|
+
];
|
|
62
|
+
TheSeamMapsControlComponent.propDecorators = {
|
|
63
|
+
def: [{ type: Input }]
|
|
64
|
+
};
|
|
65
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWFwLWNvbnRyb2wuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vcHJvamVjdHMvdWktY29tbW9uL2dvb2dsZS1tYXBzL21hcC1jb250cm9sLmNvbXBvbmVudC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFDQSxPQUFPLEVBQ0wsdUJBQXVCLEVBQ3ZCLFNBQVMsRUFDVCxNQUFNLEVBQ04sS0FBSyxHQUVOLE1BQU0sZUFBZSxDQUFBO0FBQ3RCLE9BQU8sRUFBRSxPQUFPLEVBQUUsTUFBTSxNQUFNLENBQUE7QUFDOUIsT0FBTyxFQUFFLFNBQVMsRUFBRSxHQUFHLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQTtBQUUvQyxPQUFPLEVBQUUseUJBQXlCLEVBQWlCLE1BQU0sZ0NBQWdDLENBQUE7QUFDekYsT0FBTyxFQUFFLGlCQUFpQixFQUFFLE1BQU0sdUJBQXVCLENBQUE7QUFDekQsT0FBTyxFQUFjLG9CQUFvQixFQUFFLE1BQU0sd0JBQXdCLENBQUE7QUFPekUsTUFBTSxPQUFPLDJCQUEyQjtJQWN0QyxZQUNtQixXQUE4QixFQUNBLG1CQUE4QztRQUQ1RSxnQkFBVyxHQUFYLFdBQVcsQ0FBbUI7UUFDQSx3QkFBbUIsR0FBbkIsbUJBQW1CLENBQTJCO1FBZjlFLG1CQUFjLEdBQUcsSUFBSSxPQUFPLEVBQVEsQ0FBQTtJQWdCakQsQ0FBQztJQVpMLElBQ0ksR0FBRyxDQUFDLEtBQW9DO1FBQzFDLElBQUksS0FBSyxLQUFLLElBQUksQ0FBQyxJQUFJLEVBQUU7WUFDdkIsSUFBSSxDQUFDLE9BQU8sRUFBRSxDQUFBO1lBQ2QsSUFBSSxDQUFDLElBQUksR0FBRyxLQUFLLENBQUE7U0FDbEI7SUFDSCxDQUFDO0lBUUQsY0FBYztJQUNkLFFBQVE7UUFDTixJQUFJLENBQUMsV0FBVyxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQzdCLEdBQUcsQ0FBQyxLQUFLLENBQUMsRUFBRTtZQUNWLElBQUksS0FBSyxFQUFFO2dCQUNULElBQUksQ0FBQyxJQUFJLEVBQUUsQ0FBQTthQUNaO2lCQUFNO2dCQUNMLElBQUksQ0FBQyxPQUFPLEVBQUUsQ0FBQTthQUNmO1FBQ0gsQ0FBQyxDQUFDLEVBQ0YsU0FBUyxDQUFDLElBQUksQ0FBQyxjQUFjLENBQUMsQ0FDL0IsQ0FBQyxTQUFTLEVBQUUsQ0FBQTtJQUNmLENBQUM7SUFFRCxjQUFjO0lBQ2QsV0FBVztRQUNULElBQUksQ0FBQyxPQUFPLEVBQUUsQ0FBQTtRQUVkLElBQUksQ0FBQyxjQUFjLENBQUMsSUFBSSxFQUFFLENBQUE7UUFDMUIsSUFBSSxDQUFDLGNBQWMsQ0FBQyxRQUFRLEVBQUUsQ0FBQTtJQUNoQyxDQUFDO0lBRU8sSUFBSTtRQUNWLElBQUksSUFBSSxDQUFDLFdBQVcsRUFBRTtZQUNwQixPQUFNO1NBQ1A7UUFFRCxJQUFJLElBQUksQ0FBQyxJQUFJLEtBQUssSUFBSSxJQUFJLElBQUksQ0FBQyxJQUFJLEtBQUssU0FBUyxFQUFFO1lBQ2pELE9BQU07U0FDUDtRQUVELElBQUksQ0FBQyxXQUFXLEdBQUcsSUFBSSxDQUFDLG1CQUFtQixDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLENBQUE7SUFDNUQsQ0FBQztJQUVPLE9BQU87O1FBQ2IsTUFBQSxJQUFJLENBQUMsV0FBVywwQ0FBRSxPQUFPLEVBQUUsQ0FBQTtRQUMzQixJQUFJLENBQUMsV0FBVyxHQUFHLFNBQVMsQ0FBQTtJQUM5QixDQUFDOzs7WUE3REYsU0FBUyxTQUFDO2dCQUNULFFBQVEsRUFBRSxrQkFBa0I7Z0JBQzVCLFFBQVEsRUFBRSxFQUFFO2dCQUNaLGVBQWUsRUFBRSx1QkFBdUIsQ0FBQyxNQUFNO2FBQ2hEOzs7WUFQUSxpQkFBaUI7WUFEakIseUJBQXlCLHVCQXlCN0IsTUFBTSxTQUFDLG9CQUFvQjs7O2tCQVg3QixLQUFLIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgT25Jbml0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSdcbmltcG9ydCB7XG4gIENoYW5nZURldGVjdGlvblN0cmF0ZWd5LFxuICBDb21wb25lbnQsXG4gIEluamVjdCxcbiAgSW5wdXQsXG4gIE9uRGVzdHJveSxcbn0gZnJvbSAnQGFuZ3VsYXIvY29yZSdcbmltcG9ydCB7IFN1YmplY3QgfSBmcm9tICdyeGpzJ1xuaW1wb3J0IHsgdGFrZVVudGlsLCB0YXAgfSBmcm9tICdyeGpzL29wZXJhdG9ycydcblxuaW1wb3J0IHsgR29vZ2xlTWFwc0NvbnRyb2xzU2VydmljZSwgTWFwQ29udHJvbFJlZiB9IGZyb20gJy4vZ29vZ2xlLW1hcHMtY29udHJvbHMuc2VydmljZSdcbmltcG9ydCB7IEdvb2dsZU1hcHNTZXJ2aWNlIH0gZnJvbSAnLi9nb29nbGUtbWFwcy5zZXJ2aWNlJ1xuaW1wb3J0IHsgTWFwQ29udHJvbCwgTUFQX0NPTlRST0xTX1NFUlZJQ0UgfSBmcm9tICcuL21hcC1jb250cm9scy1zZXJ2aWNlJ1xuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdzZWFtLW1hcC1jb250cm9sJyxcbiAgdGVtcGxhdGU6ICcnLFxuICBjaGFuZ2VEZXRlY3Rpb246IENoYW5nZURldGVjdGlvblN0cmF0ZWd5Lk9uUHVzaFxufSlcbmV4cG9ydCBjbGFzcyBUaGVTZWFtTWFwc0NvbnRyb2xDb21wb25lbnQgaW1wbGVtZW50cyBPbkluaXQsIE9uRGVzdHJveSB7XG4gIHByaXZhdGUgcmVhZG9ubHkgX25nVW5zdWJzY3JpYmUgPSBuZXcgU3ViamVjdDx2b2lkPigpXG5cbiAgcHJpdmF0ZSBfY29udHJvbFJlZj86IE1hcENvbnRyb2xSZWZcblxuICBASW5wdXQoKVxuICBzZXQgZGVmKHZhbHVlOiBNYXBDb250cm9sIHwgbnVsbCB8IHVuZGVmaW5lZCkge1xuICAgIGlmICh2YWx1ZSAhPT0gdGhpcy5fZGVmKSB7XG4gICAgICB0aGlzLl9yZW1vdmUoKVxuICAgICAgdGhpcy5fZGVmID0gdmFsdWVcbiAgICB9XG4gIH1cbiAgX2RlZjogTWFwQ29udHJvbCB8IG51bGwgfCB1bmRlZmluZWRcblxuICBjb25zdHJ1Y3RvcihcbiAgICBwcml2YXRlIHJlYWRvbmx5IF9nb29nbGVNYXBzOiBHb29nbGVNYXBzU2VydmljZSxcbiAgICBASW5qZWN0KE1BUF9DT05UUk9MU19TRVJWSUNFKSBwcml2YXRlIHJlYWRvbmx5IF9nb29nbGVNYXBzQ29udHJvbHM6IEdvb2dsZU1hcHNDb250cm9sc1NlcnZpY2UsXG4gICkgeyB9XG5cbiAgLyoqIEBpZ25vcmUgKi9cbiAgbmdPbkluaXQoKTogdm9pZCB7XG4gICAgdGhpcy5fZ29vZ2xlTWFwcy5tYXBSZWFkeSQucGlwZShcbiAgICAgIHRhcChyZWFkeSA9PiB7XG4gICAgICAgIGlmIChyZWFkeSkge1xuICAgICAgICAgIHRoaXMuX2FkZCgpXG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgdGhpcy5fcmVtb3ZlKClcbiAgICAgICAgfVxuICAgICAgfSksXG4gICAgICB0YWtlVW50aWwodGhpcy5fbmdVbnN1YnNjcmliZSlcbiAgICApLnN1YnNjcmliZSgpXG4gIH1cblxuICAvKiogQGlnbm9yZSAqL1xuICBuZ09uRGVzdHJveSgpOiB2b2lkIHtcbiAgICB0aGlzLl9yZW1vdmUoKVxuXG4gICAgdGhpcy5fbmdVbnN1YnNjcmliZS5uZXh0KClcbiAgICB0aGlzLl9uZ1Vuc3Vic2NyaWJlLmNvbXBsZXRlKClcbiAgfVxuXG4gIHByaXZhdGUgX2FkZCgpOiB2b2lkIHtcbiAgICBpZiAodGhpcy5fY29udHJvbFJlZikge1xuICAgICAgcmV0dXJuXG4gICAgfVxuXG4gICAgaWYgKHRoaXMuX2RlZiA9PT0gbnVsbCB8fCB0aGlzLl9kZWYgPT09IHVuZGVmaW5lZCkge1xuICAgICAgcmV0dXJuXG4gICAgfVxuXG4gICAgdGhpcy5fY29udHJvbFJlZiA9IHRoaXMuX2dvb2dsZU1hcHNDb250cm9scy5hZGQodGhpcy5fZGVmKVxuICB9XG5cbiAgcHJpdmF0ZSBfcmVtb3ZlKCk6IHZvaWQge1xuICAgIHRoaXMuX2NvbnRyb2xSZWY/LmRlc3Ryb3koKVxuICAgIHRoaXMuX2NvbnRyb2xSZWYgPSB1bmRlZmluZWRcbiAgfVxufVxuIl19
|
|
@@ -0,0 +1,4 @@
|
|
|
1
|
+
import { InjectionToken } from '@angular/core';
|
|
2
|
+
export const MAP_CONTROLS_SERVICE = new InjectionToken('MAP_CONTROLS_SERVICE');
|
|
3
|
+
export const MAP_CONTROL_DATA = new InjectionToken('MAP_CONTROL_DATA');
|
|
4
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWFwLWNvbnRyb2xzLXNlcnZpY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy91aS1jb21tb24vZ29vZ2xlLW1hcHMvbWFwLWNvbnRyb2xzLXNlcnZpY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLGVBQWUsQ0FBQTtBQWM5QyxNQUFNLENBQUMsTUFBTSxvQkFBb0IsR0FBRyxJQUFJLGNBQWMsQ0FBcUIsc0JBQXNCLENBQUMsQ0FBQTtBQUVsRyxNQUFNLENBQUMsTUFBTSxnQkFBZ0IsR0FBRyxJQUFJLGNBQWMsQ0FBTSxrQkFBa0IsQ0FBQyxDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSW5qZWN0aW9uVG9rZW4gfSBmcm9tICdAYW5ndWxhci9jb3JlJ1xuXG5pbXBvcnQgeyBDb21wb25lbnRUeXBlIH0gZnJvbSAnQHRoZXNlYW0vdWktY29tbW9uL21vZGVscydcblxuZXhwb3J0IGludGVyZmFjZSBNYXBDb250cm9sPFREYXRhID0gYW55PiB7XG4gIGNvbXBvbmVudDogQ29tcG9uZW50VHlwZTxhbnk+XG4gIGRhdGE/OiBURGF0YVxuICBwb3NpdGlvbj86IGdvb2dsZS5tYXBzLkNvbnRyb2xQb3NpdGlvblxufVxuXG5leHBvcnQgaW50ZXJmYWNlIE1hcENvbnRyb2xzU2VydmljZSB7XG4gIGFkZChjb250cm9sOiBNYXBDb250cm9sKTogdm9pZFxufVxuXG5leHBvcnQgY29uc3QgTUFQX0NPTlRST0xTX1NFUlZJQ0UgPSBuZXcgSW5qZWN0aW9uVG9rZW48TWFwQ29udHJvbHNTZXJ2aWNlPignTUFQX0NPTlRST0xTX1NFUlZJQ0UnKVxuXG5leHBvcnQgY29uc3QgTUFQX0NPTlRST0xfREFUQSA9IG5ldyBJbmplY3Rpb25Ub2tlbjxhbnk+KCdNQVBfQ09OVFJPTF9EQVRBJylcbiJdfQ==
|
|
@@ -0,0 +1,135 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, Component, ElementRef, NgZone, Renderer2, } from '@angular/core';
|
|
2
|
+
import { Subject } from 'rxjs';
|
|
3
|
+
import { takeUntil, tap } from 'rxjs/operators';
|
|
4
|
+
import { readGeoFile } from '@theseam/ui-common/utils';
|
|
5
|
+
import { GoogleMapsService } from '../google-maps.service';
|
|
6
|
+
import { MapValueManagerService, MapValueSource } from '../map-value-manager.service';
|
|
7
|
+
/**
|
|
8
|
+
*
|
|
9
|
+
*/
|
|
10
|
+
export class TheSeamMapFileDropComponent {
|
|
11
|
+
constructor(_elementRef, _ngZone, _googleMaps, _mapValueManager, _renderer) {
|
|
12
|
+
this._elementRef = _elementRef;
|
|
13
|
+
this._ngZone = _ngZone;
|
|
14
|
+
this._googleMaps = _googleMaps;
|
|
15
|
+
this._mapValueManager = _mapValueManager;
|
|
16
|
+
this._renderer = _renderer;
|
|
17
|
+
this._ngUnsubscribe = new Subject();
|
|
18
|
+
this._listeners = [];
|
|
19
|
+
this._globalDragInProgress = false;
|
|
20
|
+
this._handleDragOverEvent = (event) => {
|
|
21
|
+
if (!this._dropAllowed()) {
|
|
22
|
+
return;
|
|
23
|
+
}
|
|
24
|
+
if (!this._isSupportedDataTransferTypes(event.dataTransfer)) {
|
|
25
|
+
return;
|
|
26
|
+
}
|
|
27
|
+
event.preventDefault();
|
|
28
|
+
event.dataTransfer.dropEffect = 'copy';
|
|
29
|
+
};
|
|
30
|
+
this._handleDropEvent = (event) => {
|
|
31
|
+
if (!this._dropAllowed()) {
|
|
32
|
+
return;
|
|
33
|
+
}
|
|
34
|
+
event.preventDefault();
|
|
35
|
+
event.stopPropagation();
|
|
36
|
+
this._renderer.setStyle(this._elementRef.nativeElement, 'display', 'none');
|
|
37
|
+
// TODO: Show error if multiple files?
|
|
38
|
+
if (!this._isSupportedDataTransfer(event.dataTransfer)) {
|
|
39
|
+
return;
|
|
40
|
+
}
|
|
41
|
+
const item = event.dataTransfer.items[0];
|
|
42
|
+
const file = item.getAsFile();
|
|
43
|
+
readGeoFile(file).then(json => {
|
|
44
|
+
this._mapValueManager.setValue(json, MapValueSource.Input);
|
|
45
|
+
});
|
|
46
|
+
};
|
|
47
|
+
this._handleDragEnterEvent = (event) => {
|
|
48
|
+
if (!this._dropAllowed()) {
|
|
49
|
+
return;
|
|
50
|
+
}
|
|
51
|
+
if (!this._isSupportedDataTransferTypes(event.dataTransfer)) {
|
|
52
|
+
return;
|
|
53
|
+
}
|
|
54
|
+
this._renderer.setStyle(this._elementRef.nativeElement, 'display', 'block');
|
|
55
|
+
};
|
|
56
|
+
this._handleDragLeaveEvent = (event) => {
|
|
57
|
+
if (!this._dropAllowed()) {
|
|
58
|
+
return;
|
|
59
|
+
}
|
|
60
|
+
this._renderer.setStyle(this._elementRef.nativeElement, 'display', 'none');
|
|
61
|
+
};
|
|
62
|
+
}
|
|
63
|
+
/** @ignore */
|
|
64
|
+
ngOnInit() {
|
|
65
|
+
this._googleMaps.mapReady$.pipe(tap(ready => {
|
|
66
|
+
if (ready) {
|
|
67
|
+
this._enableFileDrop();
|
|
68
|
+
}
|
|
69
|
+
else {
|
|
70
|
+
this._disableFileDrop();
|
|
71
|
+
}
|
|
72
|
+
}), takeUntil(this._ngUnsubscribe)).subscribe();
|
|
73
|
+
}
|
|
74
|
+
/** @ignore */
|
|
75
|
+
ngOnDestroy() {
|
|
76
|
+
this._disableFileDrop();
|
|
77
|
+
this._ngUnsubscribe.next();
|
|
78
|
+
this._ngUnsubscribe.complete();
|
|
79
|
+
}
|
|
80
|
+
_enableFileDrop() {
|
|
81
|
+
const divElement = this._googleMaps.getDiv();
|
|
82
|
+
if (this._mapDiv === divElement) {
|
|
83
|
+
return;
|
|
84
|
+
}
|
|
85
|
+
if (this._mapDiv !== undefined) {
|
|
86
|
+
// Disable file drop on previous map div.
|
|
87
|
+
this._disableFileDrop();
|
|
88
|
+
}
|
|
89
|
+
this._mapDiv = divElement;
|
|
90
|
+
this._ngZone.runOutsideAngular(() => {
|
|
91
|
+
this._listeners.push(this._renderer.listen('document', 'dragstart', (event) => {
|
|
92
|
+
this._globalDragInProgress = true;
|
|
93
|
+
}));
|
|
94
|
+
this._listeners.push(this._renderer.listen('document', 'dragend', (event) => {
|
|
95
|
+
this._globalDragInProgress = false;
|
|
96
|
+
}));
|
|
97
|
+
this._listeners.push(this._renderer.listen(divElement, 'dragover', this._handleDragOverEvent));
|
|
98
|
+
this._listeners.push(this._renderer.listen(this._elementRef.nativeElement, 'dragover', this._handleDragOverEvent));
|
|
99
|
+
this._listeners.push(this._renderer.listen(this._elementRef.nativeElement, 'drop', this._handleDropEvent));
|
|
100
|
+
this._listeners.push(this._renderer.listen(divElement, 'dragenter', this._handleDragEnterEvent));
|
|
101
|
+
this._listeners.push(this._renderer.listen(this._elementRef.nativeElement, 'dragleave', this._handleDragLeaveEvent));
|
|
102
|
+
});
|
|
103
|
+
}
|
|
104
|
+
_disableFileDrop() {
|
|
105
|
+
if (this._listeners.length > 0) {
|
|
106
|
+
this._listeners.forEach(l => l());
|
|
107
|
+
this._listeners = [];
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
_dropAllowed() {
|
|
111
|
+
return !this._globalDragInProgress;
|
|
112
|
+
}
|
|
113
|
+
_isSupportedDataTransfer(dataTransfer) {
|
|
114
|
+
return dataTransfer.files.length === 1 && dataTransfer.types[0] === 'Files';
|
|
115
|
+
}
|
|
116
|
+
_isSupportedDataTransferTypes(dataTransfer) {
|
|
117
|
+
return dataTransfer.types[0] === 'Files';
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
TheSeamMapFileDropComponent.decorators = [
|
|
121
|
+
{ type: Component, args: [{
|
|
122
|
+
selector: 'seam-map-file-drop',
|
|
123
|
+
template: "",
|
|
124
|
+
changeDetection: ChangeDetectionStrategy.OnPush,
|
|
125
|
+
styles: [":host{display:none;position:absolute;top:0;left:0;width:100%;height:100%;z-index:9999!important;background-color:#1e1e1eb3}\n"]
|
|
126
|
+
},] }
|
|
127
|
+
];
|
|
128
|
+
TheSeamMapFileDropComponent.ctorParameters = () => [
|
|
129
|
+
{ type: ElementRef },
|
|
130
|
+
{ type: NgZone },
|
|
131
|
+
{ type: GoogleMapsService },
|
|
132
|
+
{ type: MapValueManagerService },
|
|
133
|
+
{ type: Renderer2 }
|
|
134
|
+
];
|
|
135
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"map-file-drop.component.js","sourceRoot":"","sources":["../../../../../projects/ui-common/google-maps/map-file-drop/map-file-drop.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  Component,\n  ElementRef,\n  NgZone,\n  OnDestroy,\n  OnInit,\n  Renderer2,\n} from '@angular/core'\nimport { Subject } from 'rxjs'\nimport { takeUntil, tap } from 'rxjs/operators'\n\nimport { readGeoFile } from '@theseam/ui-common/utils'\n\nimport { GoogleMapsService } from '../google-maps.service'\nimport { MapValueManagerService, MapValueSource } from '../map-value-manager.service'\n\n/**\n *\n */\n@Component({\n  selector: 'seam-map-file-drop',\n  templateUrl: './map-file-drop.component.html',\n  styleUrls: ['./map-file-drop.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class TheSeamMapFileDropComponent implements OnInit, OnDestroy {\n\n  private readonly _ngUnsubscribe = new Subject<void>()\n\n  /**\n   * Map element drag events added to.\n   */\n  private _mapDiv: HTMLDivElement | undefined\n\n  private _listeners: (() => void)[]  = []\n\n  private _globalDragInProgress: boolean = false\n\n  constructor(\n    private readonly _elementRef: ElementRef,\n    private readonly _ngZone: NgZone,\n    private readonly _googleMaps: GoogleMapsService,\n    private readonly _mapValueManager: MapValueManagerService,\n    private readonly _renderer: Renderer2\n  ) { }\n\n  /** @ignore */\n  ngOnInit() {\n    this._googleMaps.mapReady$.pipe(\n      tap(ready => {\n        if (ready) {\n          this._enableFileDrop()\n        } else {\n          this._disableFileDrop()\n        }\n      }),\n      takeUntil(this._ngUnsubscribe)\n    ).subscribe()\n  }\n\n  /** @ignore */\n  ngOnDestroy() {\n    this._disableFileDrop()\n\n    this._ngUnsubscribe.next()\n    this._ngUnsubscribe.complete()\n  }\n\n  private _enableFileDrop(): void {\n    const divElement = this._googleMaps.getDiv()\n    if (this._mapDiv === divElement) {\n      return\n    }\n\n    if (this._mapDiv !== undefined) {\n      // Disable file drop on previous map div.\n      this._disableFileDrop()\n    }\n\n    this._mapDiv = divElement\n    this._ngZone.runOutsideAngular(() => {\n      this._listeners.push(this._renderer.listen('document', 'dragstart', (event: Event) => {\n        this._globalDragInProgress = true\n      }))\n\n      this._listeners.push(this._renderer.listen('document', 'dragend', (event: Event) => {\n        this._globalDragInProgress = false\n      }))\n\n      this._listeners.push(this._renderer.listen(divElement, 'dragover', this._handleDragOverEvent))\n      this._listeners.push(this._renderer.listen(this._elementRef.nativeElement, 'dragover', this._handleDragOverEvent))\n      this._listeners.push(this._renderer.listen(this._elementRef.nativeElement, 'drop', this._handleDropEvent))\n      this._listeners.push(this._renderer.listen(divElement, 'dragenter', this._handleDragEnterEvent))\n      this._listeners.push(this._renderer.listen(this._elementRef.nativeElement, 'dragleave', this._handleDragLeaveEvent))\n    })\n  }\n\n  private _disableFileDrop(): void {\n    if (this._listeners.length > 0) {\n      this._listeners.forEach(l => l())\n      this._listeners = []\n    }\n  }\n\n  private readonly _handleDragOverEvent = (event: any) => {\n    if (!this._dropAllowed()) {\n      return\n    }\n\n    if (!this._isSupportedDataTransferTypes(event.dataTransfer)) {\n      return\n    }\n\n    event.preventDefault()\n    event.dataTransfer.dropEffect = 'copy'\n  }\n\n  private readonly _handleDropEvent = (event: any) => {\n    if (!this._dropAllowed()) {\n      return\n    }\n\n    event.preventDefault()\n    event.stopPropagation()\n    this._renderer.setStyle(this._elementRef.nativeElement, 'display', 'none')\n\n    // TODO: Show error if multiple files?\n    if (!this._isSupportedDataTransfer(event.dataTransfer)) {\n      return\n    }\n\n    const item = event.dataTransfer.items[0]\n    const file = item.getAsFile()\n    readGeoFile(file).then(json => {\n      this._mapValueManager.setValue(json, MapValueSource.Input)\n    })\n  }\n\n  private readonly _handleDragEnterEvent = (event: any) => {\n    if (!this._dropAllowed()) {\n      return\n    }\n\n    if (!this._isSupportedDataTransferTypes(event.dataTransfer)) {\n      return\n    }\n\n    this._renderer.setStyle(this._elementRef.nativeElement, 'display', 'block')\n  }\n\n  private readonly _handleDragLeaveEvent = (event: any) => {\n    if (!this._dropAllowed()) {\n      return\n    }\n\n    this._renderer.setStyle(this._elementRef.nativeElement, 'display', 'none')\n  }\n\n  private _dropAllowed(): boolean {\n    return !this._globalDragInProgress\n  }\n\n  private _isSupportedDataTransfer(dataTransfer: DataTransfer): boolean {\n    return dataTransfer.files.length === 1 && dataTransfer.types[0] === 'Files'\n  }\n\n  private _isSupportedDataTransferTypes(dataTransfer: DataTransfer): boolean {\n    return dataTransfer.types[0] === 'Files'\n  }\n\n}\n"]}
|