Subversion Repositories eFlore/Applications.cel

Rev

Rev 3845 | Rev 3848 | Go to most recent revision | Details | Compare with Previous | Last modification | View Log | RSS feed

Rev Author Line No. Line
3845 idir 1
import {NOMINATIM_OSM_URL,TbPlaces} from "./modules/Locality.js";
2
 
3
const tileLayers = {
4
    googleHybrid: [
5
        'https://{s}.google.com/vt/lyrs=s,h&x={x}&y={y}&z={z}',
6
        {
7
            attribution: '<a href="https://www.google.com" target="_blank">\xa9 Google Maps</a>',
8
            subdomains: ["mt0","mt1","mt2","mt3"],
9
            maxZoom: 20
10
        }
11
    ],
12
    osm: [
13
        'https://osm.tela-botanica.org/tuiles/osmfr/{z}/{x}/{y}.png',
14
        {
15
            attribution: 'Data © <a href="http://osm.org/copyright">OpenStreetMap</a>',
16
             maxZoom: 18
17
        }
18
    ]
19
};
20
const defaultPosition = {
21
    lat: 47.0504,
22
    lng: 2.2347
23
};
24
const defaultCityZoom = 12;
25
const defaultZoom = 4;
26
const geometryFilterDefault = 'point';
27
const defaultMapIdAttr = 'tb-geolocation';
28
const markerImgBaseUrl = URL_BASE +'js/tb-geoloc/img/';
29
const markerIcon = L.Icon.extend({
30
    options: {
31
        shadowUrl: markerImgBaseUrl +'marker-shadow.png',
32
        iconAnchor: new L.Point(12, 40),//correctly replaces the dot of the pointer
33
        iconSize: new L.Point(24,40),
34
        iconUrl: markerImgBaseUrl +'marker-icon.png',
35
    }
36
});
37
const defaultPolylineOptions = {
38
    shapeOptions: {
39
        weight: 5
40
    },
41
    showLength: true,
42
    repeatMode: false
43
};
44
 
45
const drawLocale = {
46
    edit: {
47
        handlers: {
48
            edit: {
49
                tooltip: {
50
                    subtext: 'Cliquer sur annuler pour supprimer les changements',
51
                    text: 'Déplacez les marqueurs pour modifier leur position'
52
                }
53
            },
54
            remove: {
55
                tooltip: {
56
                    text: 'cliquer sur une ligne pour supprimer'
57
                }
58
            }
59
        },
60
        toolbar: {
61
            actions: {
62
                cancel: {
63
                    text: 'Annuler',
64
                    title: 'Annuler'
65
                },
66
                clearAll: {
67
                    text: 'Effacer',
68
                    title: 'Tout effacer'
69
                },
70
                save: {
71
                    text: 'Sauvegarder',
72
                    title: 'Sauvegarder les changements'
73
                }
74
            },
75
            buttons: {
76
                edit: 'Editer',
77
                editDisabled: 'Rien à editer',
78
                remove: 'Supprimer',
79
                removeDisabled: 'Rien à supprimer'
80
            }
81
        }
82
    },
83
    draw : {
84
        toolbar: {
85
            actions: {
86
                text: 'Annuler',
87
                title: 'Annuler'
88
            },
89
            buttons: {
90
                polyline: 'Dessiner une ligne',
91
                marker: 'Pointer une position'
92
            },
93
            finish: {
94
                text: 'Terminer',
95
                title: 'Terminer'
96
            },
97
            undo: {
98
                text: 'Supprimer',
99
                title: 'Supprimer le dernier point'
100
            }
101
        },
102
        handlers: {
103
            polyline: {
104
                tooltip: {
105
                    start: 'Cliquer sur la carte pour placer le début de la ligne',
106
                    cont: 'Positionner le prochain point et cliquer',
107
                    end: 'Re-cliquer sur le dernier point pour finir la ligne'
108
                }
109
            },
110
            marker: {
111
                tooltip: {
112
                    start: 'Cliquer sur la carte pour placer le marqueur'
113
                }
114
            },
115
            rectangle: {tooltip: {}},
116
            simpleshape: {tooltip: {}},
117
            circle: {tooltip: {}},
118
            circlemarker: {tooltip: {}}
119
        }
120
    }
121
};
122
 
123
/***************************************************/
124
 
125
export function Geoloc() {
126
    this.map = {};
127
    this.coordinates = defaultPosition;
128
    this.geojson = {};
129
    this.editableLayers = {};
130
    this.defaultDrawControlOptions = {};
131
    this.drawControl = {};
132
    this.defaultEditOptions = false;
133
    this.layer = {};
134
};
135
 
136
Geoloc.prototype.init = function(suffix = '') {
137
    this.mapIdAttr = defaultMapIdAttr + suffix;
138
    this.geolocLabel = document.getElementById('geoloc-label' + suffix);
139
    this.initForm();
140
    this.initEvts();
141
};
142
 
143
Geoloc.prototype.initForm = function() {
144
    this.mapEl = document.getElementById(this.mapIdAttr);
145
    this.zoom = this.mapEl.dataset.zoom || defaultZoom;
146
    this.geometryFilter = this.mapEl.dataset.typeLocalisation || geometryFilterDefault;
147
 
148
    const formSuffix = this.mapEl.dataset.formSuffix;
149
 
150
    this.latitudeEl = document.getElementById('latitude' + formSuffix);
151
    this.longitudeEl = document.getElementById('longitude' + formSuffix);
152
};
153
 
154
Geoloc.prototype.initEvts = function() {
155
    this.initMap();
156
    this.handleCoordinates();
157
    this.onCoordinates();
158
    if('point' === this.geometryFilter) {
159
        this.initSearchLocality();
160
    }
161
};
162
 
163
Geoloc.prototype.initMap = function() {
164
    this.map = this.createLocationMap();
165
 
166
    // interactions with map
167
    this.map.on(L.Draw.Event.CREATED, evt => {
168
        this.layer = evt.layer;
169
 
170
        // created marker or polyline with drawControl
171
        // no more need this drawcontrol: remove
172
        // (polyline: another one will be added with only edit toolbar)
173
        this.map.removeControl(this.drawControl);
174
 
175
        if ('marker' === evt.layerType) {
176
            this.onMarkerCreated();
177
        } else if ('polyline' === evt.layerType) {
178
            this.handlePolylineEvents();
179
        }
180
    });
181
 
182
    this.map.on(L.Draw.Event.EDITED, evt => {
183
        const layers = evt.layers;
184
 
185
        this.map.removeLayer(this.layer);
186
 
187
        layers.eachLayer(layer => {
188
            this.layer = layer;
189
            this.handlePolylineEvents(false);
190
        });
191
    });
192
 
193
    this.map.on(L.Draw.Event.DELETED, evt => {
194
        this.reSetDrawControl();
195
    });
196
};
197
 
198
 
199
Geoloc.prototype.reSetDrawControl = function() {
200
    this.map.removeLayer(this.layer);
201
    this.map.removeControl(this.drawControl);
202
    this.setDrawControl(this.map);
203
};
204
 
205
Geoloc.prototype.createLocationMap = function() {
206
    const lthis = this,
207
        map = L.map(this.mapIdAttr, {zoomControl: true, gestureHandling: true}).setView([this.coordinates.lat, this.coordinates.lng], this.zoom),
208
        tileLayer = this.mapEl.dataset.layer || 'osm';
209
 
210
    L.tileLayer(...tileLayers[tileLayer]).addTo(map);
211
 
212
    this.editableLayers = new L.FeatureGroup();
213
    map.addLayer(this.editableLayers);
214
 
215
    this.setDrawControl(map);
216
 
217
    return map;
218
};
219
 
220
Geoloc.prototype.setDrawControl = function(map) {
221
    this.defaultDrawControlOptions = this.generateDrawControlOptions();
222
    this.drawControl = this.generateDrawControl(...Object.values(this.defaultDrawControlOptions));
223
    map.addControl(this.drawControl);
224
};
225
 
226
Geoloc.prototype.generateDrawControlOptions = function() {
227
    const options = {
228
        editOptions: false,
229
        markerOptions: false,
230
        polylineOptions: false
231
    };
232
 
233
    this.defaultEditOptions = {
234
        featureGroup: this.editableLayers,// REQUIRED!!
235
        remove: true
236
    }
237
 
238
    switch (this.geometryFilter) {
239
        case 'point':
240
            options.markerOptions = {
241
                icon: new markerIcon(),
242
                repeatMode: false
243
            };
244
            break;
245
        case 'rue':
246
            options.polylineOptions = defaultPolylineOptions;
247
            break;
248
        default:
249
            break;
250
    }
251
 
252
    return options;
253
};
254
 
255
Geoloc.prototype.generateDrawControl = function(
256
    editOptions,
257
    markerOptions = false,
258
    polylineOptions = false
259
) {
260
    L.drawLocal = drawLocale;
261
 
262
    return new L.Control.Draw({
263
        position: 'topleft',
264
        draw: {
265
            polyline: polylineOptions,
266
            polygon: false,
267
            circle: false,
268
            rectangle: false,
269
            marker: markerOptions,
270
            circlemarker: false
271
        },
272
        edit: editOptions
273
    });
274
};
275
 
276
Geoloc.prototype.onMarkerCreated = function() {
277
    const coordinates = this.layer.getLatLng();
278
 
279
    this.handleNewLocation(coordinates);
280
};
281
 
282
Geoloc.prototype.handleMarkerEvents = function() {
3847 idir 283
    if(this.map) {
284
        const lthis = this;
285
        // move marker on click
286
        $(this.map).off('click').on('click', function(evt) {
287
            lthis.handleNewLocation(evt.latlng);
288
        });
289
        // move marker on drag
290
        $(this.map.marker).off('dragend').on('dragend', function() {
291
            lthis.handleNewLocation(lthis.map.marker.getLatLng());
292
        });
293
    }
3845 idir 294
};
295
 
296
Geoloc.prototype.handlePolylineEvents = function(requiresNewEditDrawControl = true) {
297
    const latLngs = this.layer.getLatLngs(),
298
        polyline = this.formatPolyline(latLngs),
299
        coordinates = this.layer.getBounds().getCenter();
300
 
301
 
302
    if (requiresNewEditDrawControl) {
303
        this.drawControl = this.generateDrawControl(this.defaultEditOptions);
304
        this.map.addControl(this.drawControl);
305
    }
306
 
307
    // make polyline removable
308
    this.editableLayers.addLayer(L.polyline(latLngs));
309
 
310
    this.map.addLayer(this.layer);
311
    this.handleNewLocation(coordinates, polyline);
312
};
313
 
314
/**
315
 * triggers location event
316
 * @param coordinates.
317
 * @param coordinates.lat latitude.
318
 * @param coordinates.lng longitude.
319
 * @param {array||null} polyline array of coordinates object
320
 */
321
Geoloc.prototype.handleNewLocation = function (coordinates, polyline = []) {
322
    coordinates = this.formatCoordinates(coordinates);
323
 
324
    if(!!coordinates && !!coordinates.lat && coordinates.lng) {
325
        this.setMapPosition(coordinates);
326
        this.getLocationInfo(coordinates, polyline);
327
    }
328
};
329
 
330
Geoloc.prototype.formatCoordinates = function (coordinates) {
331
    coordinates.lat = Number.parseFloat(coordinates.lat);
332
    coordinates.lng = Number.parseFloat(coordinates.lng);
333
 
334
    if(Number.isNaN(coordinates.lat) || Number.isNaN(coordinates.lng)) {
335
        return null;
336
    }
337
 
338
    return coordinates;
339
};
340
 
341
Geoloc.prototype.formatPolyline = function (latLngs) {
342
    const polyline = [];
343
 
344
    latLngs.forEach(coordinates => polyline.push(Object.values(coordinates)));
345
 
346
    return polyline;
347
};
348
 
349
Geoloc.prototype.setMapPosition = function (coordinates) {
350
    const latLng = new L.LatLng(coordinates.lat, coordinates.lng);
351
 
352
 
353
    this.coordinates = coordinates;
354
    if('point' === this.geometryFilter) {
355
        this.createDraggableMarker(latLng);
356
        this.handleMarkerEvents();
357
    }
358
    // updates map
359
    this.map.setView(latLng);
360
};
361
 
362
Geoloc.prototype.createDraggableMarker = function(latLng) {
363
    if (undefined === latLng) {
364
        latLng = new L.LatLng(this.coordinates.lat, this.coordinates.lng);
365
    }
366
 
367
    if (undefined === this.map.marker) {
368
        // after many attempts, did not manage
369
        // to make marker from draw control draggable
370
        // solution: replace it
371
        if(this.layer) {
372
            this.map.removeLayer(this.layer);
373
        }
374
        this.map.marker = new L.Marker(this.coordinates, {
375
            draggable: true,
376
            icon: new markerIcon()
377
        });
378
        this.layer = this.map.marker;
379
        this.map.addLayer(this.map.marker);
380
        if(this.drawControl) {
381
            this.map.removeControl(this.drawControl);
382
        }
383
    }
384
 
385
    this.map.marker.setLatLng(latLng, {draggable: 'true'});
386
};
387
 
388
Geoloc.prototype.getLocationInfo = function(coordinates, polyline) {
389
    const lthis = this,
390
        url = NOMINATIM_OSM_URL+'reverse',
391
        params = {
392
            'format': 'json',
393
            'lat': coordinates.lat,
394
            'lon': coordinates.lng
395
        };
396
 
397
    this.geolocLabel.classList.add('loading');
398
 
399
    $.ajax({
400
        method: "GET",
401
        url: url,
402
        data: params,
403
        success: function(locationData) {
404
            lthis.loadGeolocation(coordinates,locationData,polyline);
405
        },
406
        error: (err) => {
407
            console.warn(err);
408
            lthis.geolocLabel.classList.remove('loading');
409
        }
410
    });
411
};
412
 
413
Geoloc.prototype.loadGeolocation = function(coordinates,locationData,polyline) {
414
    const lthis = this,
415
        query = {
416
            'lat': coordinates.lat,
417
            'lon': coordinates.lng
418
        };
419
 
420
    $.ajax({
421
        method: "GET",
422
        url: URL_GEOLOC_SERVICE,
423
        data: query,
424
        success: function (geoLocationData) {
425
            lthis.triggerLocationEvent(
426
                lthis.formatLocationEventObject(coordinates,geoLocationData,locationData,polyline)
427
            );
428
            lthis.geolocLabel.classList.remove('loading');
429
        },
430
        error:  (err) => {
431
            console.warn(err);
432
            lthis.geolocLabel.classList.remove('loading');
433
        }
434
    });
435
};
436
 
437
Geoloc.prototype.triggerLocationEvent = function (locationDataObject) {
438
    const location = new CustomEvent('location', {detail: locationDataObject});
439
 
440
    this.mapEl.dispatchEvent(location);
441
};
442
 
443
Geoloc.prototype.formatLocationEventObject = function(coordinates,geoLocationData,locationData,polyline) {
444
    const detail = {
445
        centroid: {
446
            type: 'Point',
447
            coordinates: Object.values(coordinates)
448
        },
449
        elevation: geoLocationData.altitude,
450
        inseeData: {
451
            code: geoLocationData.code_zone,
452
            nom: geoLocationData.nom
453
        },
454
        osmCountry: locationData.address.country,
455
        osmCountryCode: geoLocationData.code_pays,
456
        osmCounty: locationData.address.county,
457
        osmPostcode:locationData.address.postcode,
458
        locality: this.getLocalityFromData(locationData),
459
        locality: geoLocationData.nom,
460
        osmRoad: locationData.address.road,
461
        osmState: locationData.address.state,
462
        osmId: locationData.osm_id,
463
        osmPlaceId: locationData.place_id
464
    };
465
 
466
    if (0 < polyline.length) {
467
        detail.geometry = {
468
            type: 'LineString',
469
            coordinates: polyline
470
        };
471
    } else {
472
        detail.geometry = detail.centroid;
473
    }
474
 
475
    return detail;
476
}
477
 
478
Geoloc.prototype.handleCoordinates = function() {
479
    if (!!this.latitudeEl.value && !!this.longitudeEl.value) {
480
        this.handleNewLocation({
481
            'lat': this.latitudeEl.value,
482
            'lng': this.longitudeEl.value
483
        });
484
    }
485
};
486
 
487
Geoloc.prototype.onCoordinates = function() {
488
    [this.latitudeEl,this.longitudeEl].forEach(coordinate =>
489
        coordinate.addEventListener('blur', function() {
490
            this.handleCoordinates();
491
        }.bind(this))
492
    );
493
};
494
 
495
Geoloc.prototype.initSearchLocality = function() {
496
    const placesZone = document.getElementById('tb-places-zone');
497
 
498
    if(placesZone) {
499
        placesZone.classList.remove('hidden');
500
        this.tbPlaces = new TbPlaces(this.tbPlacesCallback.bind(this));
501
        this.tbPlaces.init();
502
    }
503
};
504
 
505
Geoloc.prototype.tbPlacesCallback = function(localityData) {
506
    const locality = this.getLocalityFromData(localityData);
507
 
508
    if(!!locality) {
509
        const coordinates = {
510
            'lat' : localityData.lat,
511
            'lng' : localityData.lon
512
        };
513
 
514
        this.map.removeControl(this.drawControl);
515
        this.handleNewLocation(coordinates);
516
    }
517
};
518
 
519
Geoloc.prototype.getLocalityFromData = function(localityData) {
520
    const addressData = localityData.address,
521
        locationNameType = ['village', 'city', 'locality', 'municipality', 'county'].find(locationNameType => addressData[locationNameType] !== undefined);
522
 
523
    if (!locationNameType) {
524
        return;
525
    }
526
 
527
    return addressData[locationNameType];
528
};
529
 
530
Geoloc.prototype.closeMap = function () {
531
    // reset map
532
    this.map = L.DomUtil.get(this.mapIdAttr);
533
    if (this.map != null) {
534
        this.map._leaflet_id = null;
535
    }
536
};