simstadt_openlayers.js 15.3 KB
Newer Older
Eric Duminil's avatar
Eric Duminil committed
1
2
3
4
5
6
7
const styles = {};

styles.original = utils.polygon_style('#447744', 0.2);
styles.highlighted = utils.polygon_style("#ff44a2", 0.7);
styles.selected = utils.polygon_style("#ffff00", 0.8);

const regionChooser = (function(){
Eric Duminil's avatar
Eric Duminil committed
8
	//TODO: Somehow split in classes. This file is getting too big and mixed
9
	var publicScope = {};
Eric Duminil's avatar
const    
Eric Duminil committed
10
11
12
	const fromJavaFX = navigator.userAgent.indexOf('JavaFX') !== -1;
	const dataPanel = $('#dataPanel');
	const wgs84Sphere = new ol.Sphere(6378137);
Eric Duminil's avatar
Eric Duminil committed
13
	var features_by_project;
Eric Duminil's avatar
Eric Duminil committed
14

15
16
17
	publicScope.init = function(){
		//NOTE: Only called from JavaFX. At startup, or when Repo has been changed.
		kml_source.clear();
Eric Duminil's avatar
Eric Duminil committed
18
		document.getElementById("select_repository").style.visibility = "visible";
19
	}
Eric Duminil's avatar
Eric Duminil committed
20

21
	if (fromJavaFX){
Eric Duminil's avatar
Eric Duminil committed
22
		document.documentElement.className = 'wait';
23
	}
24
25
26

	var osm_layer = new ol.layer.Tile({
		source: new ol.source.OSM()
eric.duminil's avatar
eric.duminil committed
27
28
	});

Eric Duminil's avatar
Eric Duminil committed
29
	var kml_source = utils.read_kml(fromJavaFX ? undefined : 'data/citygml_hulls.kml');
eric.duminil's avatar
eric.duminil committed
30

31
32
	var kml_layer = new ol.layer.Vector({
		source : kml_source,
Eric Duminil's avatar
Eric Duminil committed
33
		style : styles.original
34
35
36
37
38
39
40
41
42
43
	});

	var intersections = new ol.source.Vector();

	var intersections_layer = new ol.layer.Vector({
		source : intersections,
		style : new ol.style.Style({
			fill : new ol.style.Fill({
				color : 'rgba(255, 155, 51, 0.2)'
			})
eric.duminil's avatar
eric.duminil committed
44
45
		})
	});
46

47
	publicScope.addCityGmlHull = function(kmlString) {
48
		options = {featureProjection: ol.proj.get('EPSG:3857')};
49
		feature = kmlFormat.readFeature(kmlString, options);
50
51
		kml_source.addFeature(feature);
		dataPanel.append('.');
52
		srsName = feature.get("srsName");
53
		if (proj4.defs(srsName) === undefined){
Eric Duminil's avatar
Eric Duminil committed
54
			console.warn(srsName + " isn't defined by Proj4js!")
55
		}
56
	};
Eric Duminil's avatar
Eric Duminil committed
57

58
59
	var map = new ol.Map({
		target : 'map',
60
		layers : [ osm_layer, kml_layer, intersections_layer ],
61
62
63
64
65
		interactions : ol.interaction.defaults({
			keyboard : true
		})
	});

Eric Duminil's avatar
const    
Eric Duminil committed
66
67
	const geoJsonFormat = new ol.format.GeoJSON();
	const kmlFormat = new ol.format.KML({extractStyles: false});
68

69
	kml_source.addEventListener("addfeature", function() {
Eric Duminil's avatar
Eric Duminil committed
70
		map.getView().fitExtent(kml_source.getExtent(), map.getSize());
71
72
73
74
	});

	function updateGMLPolygons() {
		kml_source.forEachFeature(function(feature) {
75
			feature["geoJSON"] = geoJsonFormat.writeFeatureObject(feature);
76
			feature["area"] = feature.getGeometry().getArea();
Eric Duminil's avatar
Eric Duminil committed
77
78
			feature["project"] = feature.get("project");
			feature["name"] = feature.get("name");
79
			feature["source"] = "CityGML";
Eric Duminil's avatar
Eric Duminil committed
80
			feature["status"] = "original";
81
		});
Eric Duminil's avatar
Eric Duminil committed
82

Eric Duminil's avatar
Eric Duminil committed
83
84
85
		var features = Array.from(kml_source.getFeatures());
		// Sort projects
		features.sort((a, b) => a.project.localeCompare(b.project));
Eric Duminil's avatar
Eric Duminil committed
86
		features_by_project = utils.groupBy(features, "project");
Eric Duminil's avatar
Eric Duminil committed
87
88
		// Sort CityGMLs inside each project
		Object.values(features_by_project).forEach(features => features.sort((a, b) => a.name.localeCompare(b.name)));
89
90
91
92
93
94
	}

	// The features are not added to a regular vector layer/source,
	// but to a feature overlay which holds a collection of features.
	// This collection is passed to the modify and also the draw
	// interaction, so that both can add or modify features.
Eric Duminil's avatar
rename    
Eric Duminil committed
95
	var drawnLayer = new ol.FeatureOverlay({
96
		style : new ol.style.Style({
eric.duminil's avatar
eric.duminil committed
97
			fill : new ol.style.Fill({
98
99
100
101
102
103
104
105
106
107
108
				color : 'rgba(255, 155, 51, 0.5)'
			}),
			stroke : new ol.style.Stroke({
				color : '#ffcc33',
				width : 4
			}),
			image : new ol.style.Circle({
				radius : 5,
				fill : new ol.style.Fill({
					color : '#ffcc33'
				})
eric.duminil's avatar
eric.duminil committed
109
110
111
			})
		})
	});
Eric Duminil's avatar
rename    
Eric Duminil committed
112
	drawnLayer.setMap(map);
113

Eric Duminil's avatar
rename    
Eric Duminil committed
114
	drawnLayer.getFeatures().on('add', function(event) {
115
116
117
118
119
120
121
		var feature = event.element;
		feature.on("change", function() {
			displayInfo();
		});
	});

	var modify = new ol.interaction.Modify({
Eric Duminil's avatar
rename    
Eric Duminil committed
122
		features : drawnLayer.getFeatures(),
123
124
125
126
127
128
129
130
131
132
		// the SHIFT key must be pressed to delete vertices, so
		// that new vertices can be drawn at the same position
		// of existing vertices
		deleteCondition : function(event) {
			return ol.events.condition.shiftKeyOnly(event) && ol.events.condition.singleClick(event);
		}
	});
	map.addInteraction(modify);

	var draw = new ol.interaction.Draw({
Eric Duminil's avatar
rename    
Eric Duminil committed
133
		features : drawnLayer.getFeatures(),
134
135
136
137
138
		type : 'Polygon'
	});
	map.addInteraction(draw);

	var sketch;
eric.duminil's avatar
eric.duminil committed
139

140
141
142
143
144
	draw.on('drawstart', function(evt) {
		sketch = evt.feature;
		updateGMLPolygons();
	});
	var sourceProj = map.getView().getProjection();
Eric Duminil's avatar
Eric Duminil committed
145

146
147
148
149
150
151
	function showLinkToDownload(feature, jsonIntersection, polygonArea){
		var intersection = geoJsonFormat.readFeature(jsonIntersection);
		var intersectionArea = intersection.getGeometry().getArea();
		var citygml_percentage = Math.round(intersectionArea / feature["area"] * 100);
		var sketch_percentage = Math.round(intersectionArea / polygonArea * 100);
		intersections.addFeature(intersection);
Eric Duminil's avatar
Eric Duminil committed
152

Eric Duminil's avatar
Eric Duminil committed
153
154
		li = document.createElement('li');
		li.feature = feature;
Eric Duminil's avatar
Eric Duminil committed
155

Eric Duminil's avatar
Eric Duminil committed
156
157
		li.onmouseover = function(){ regionChooser.highlightPolygon(this.feature) };
		li.onmouseout = function(){ regionChooser.resetHighlight(this.feature) };
Eric Duminil's avatar
Eric Duminil committed
158

Eric Duminil's avatar
Eric Duminil committed
159
160
		let label = li.appendChild(document.createElement('label'));
		var text = feature.name;
Eric Duminil's avatar
Eric Duminil committed
161

Eric Duminil's avatar
Eric Duminil committed
162
		let checkbox = document.createElement('input');
Eric Duminil's avatar
Eric Duminil committed
163
		checkbox.type = 'checkbox'
Eric Duminil's avatar
Broken.    
Eric Duminil committed
164
		checkbox.className = "select_citygml";
Eric Duminil's avatar
Eric Duminil committed
165
166
		checkbox.feature = feature;
		checkbox.setAttribute('onclick', "regionChooser.isDownloadPossible()");
Eric Duminil's avatar
Eric Duminil committed
167

Eric Duminil's avatar
Eric Duminil committed
168
		text += " (" + citygml_percentage + "%";
169
		if (sketch_percentage == 100) {
Eric Duminil's avatar
Eric Duminil committed
170
			text += ", all inside";
171
		}
Eric Duminil's avatar
Eric Duminil committed
172

Eric Duminil's avatar
Eric Duminil committed
173
		label.textContent = text + ")\n";
Eric Duminil's avatar
Eric Duminil committed
174
		label.prepend(checkbox);
Eric Duminil's avatar
Eric Duminil committed
175

176
		// append to DOM element, not to jQuery object
Eric Duminil's avatar
Eric Duminil committed
177
		dataPanel[0].appendChild(li);
178
	}
Eric Duminil's avatar
Eric Duminil committed
179

Eric Duminil's avatar
Eric Duminil committed
180
	publicScope.highlightPolygon = function(feature) {
Eric Duminil's avatar
Eric Duminil committed
181
		feature.setStyle(styles.highlighted);
182
	}
Eric Duminil's avatar
Eric Duminil committed
183

Eric Duminil's avatar
Eric Duminil committed
184
	publicScope.resetHighlight = function(feature) {
Eric Duminil's avatar
Eric Duminil committed
185
186
		refreshStyle(feature);
	}
Eric Duminil's avatar
Eric Duminil committed
187

Eric Duminil's avatar
Eric Duminil committed
188
189
190
191
	refreshStyle = function(feature, status){
		if (status){
			feature.status = status;
		}
Eric Duminil's avatar
Eric Duminil committed
192
		feature.setStyle(styles[feature.status]);
193
	}
Eric Duminil's avatar
Eric Duminil committed
194

195
	publicScope.isDownloadPossible = function(){
Eric Duminil's avatar
Eric Duminil committed
196
		kml_source.getFeatures().forEach(f => refreshStyle(f, "original"));
Eric Duminil's avatar
Eric Duminil committed
197

Eric Duminil's avatar
Eric Duminil committed
198
		selectedFeatures = getSelectedGMLs();
Eric Duminil's avatar
Eric Duminil committed
199

Eric Duminil's avatar
Broken.    
Eric Duminil committed
200
		selectedFeatures.forEach(f => refreshStyle(f, "selected"));
Eric Duminil's avatar
Eric Duminil committed
201

Eric Duminil's avatar
Broken.    
Eric Duminil committed
202
		document.getElementById("download_region_button").disabled = (selectedFeatures.length == 0);
Eric Duminil's avatar
Eric Duminil committed
203
	}
Eric Duminil's avatar
Eric Duminil committed
204

Eric Duminil's avatar
Eric Duminil committed
205
206
207
	function getSelectedGMLs(){
		return Array.from(document.querySelectorAll("input.select_citygml")).filter(c => c.checked).map(c => c.feature);
	}
Eric Duminil's avatar
Eric Duminil committed
208

209
	function findIntersection(feature, polygon) {
210
		try {
211
			return turf.intersect(polygon, feature["geoJSON"]);
212
213
214
215
		} catch (err) {
			console.log(feature.get('name') + " - " + err);
		}
	}
216
217

	function findIntersections() {
218
219
		var polygon = geoJsonFormat.writeFeatureObject(sketch);
		var polygonArea = sketch.getGeometry().getArea();
220
221
		var intersection_found = false;
		intersections.clear();
Eric Duminil's avatar
Eric Duminil committed
222

223
		Object.keys(features_by_project).forEach(function(project) {
224
225
226
227
			features = features_by_project[project];
			features_and_intersections = features.map(f=> [f, findIntersection(f,polygon)]).filter(l => l[1] !== undefined);
			if (features_and_intersections.length > 0){
				intersection_found = true;
228
				dataPanel.append("<h2 class='info'>" + project);
229
230
				features_and_intersections.forEach(l => showLinkToDownload(l[0], l[1], polygonArea));
			}
231
		});
Eric Duminil's avatar
Eric Duminil committed
232

233
		if (intersection_found) {
234
			document.getElementById("download_region").style.visibility = 'visible';
235
236
		}
		else {
237
			document.getElementById("download_region").style.visibility = 'hidden';
238
			dataPanel.append("No intersection found with any CityGML file.<br/>\n");
eric.duminil's avatar
eric.duminil committed
239
240
		}
	}
241

242
243
244
	publicScope.display = function(text){
		dataPanel.append(text + "<br/>\n");
	}
Eric Duminil's avatar
Eric Duminil committed
245

246
247
248
	publicScope.downloadStart = function(){
		document.getElementById("download_region_button").disabled = true;
		document.documentElement.className = 'wait';
Eric Duminil's avatar
Eric Duminil committed
249
		dataPanel.prepend("<h2 id='download_start' class='ok'>Extracting region...</h2><br/>\n");
250
	}
Eric Duminil's avatar
Eric Duminil committed
251

252
	publicScope.downloadFinished = function(count){
253
254
		document.documentElement.className = ''; // Stop waiting
		document.getElementById("download_start").remove();
255
256
257
258
259
		if (count > 0){
			dataPanel.prepend("<h2 class='ok'>Done! (" + count + " buildings found) </h2><br/>\n");
		} else {
			dataPanel.prepend("<h2 class='error'>No building has been found in this region</h2><br/>\n");
		}
260
261
262
263
264
		var button = document.getElementById("download_region_button");
		if (button){ // Region might have been modified since download start
			button.disabled = false;
		}
	}
Eric Duminil's avatar
Eric Duminil committed
265

266
267
	publicScope.downloadFromSelectedCityGMLs = function() {
		var features = getSelectedGMLs();
Eric Duminil's avatar
Eric Duminil committed
268
269
		var project = features[0].get("project");
		var srsName = features[0].get("srsName");
Eric Duminil's avatar
Eric Duminil committed
270

Eric Duminil's avatar
Eric Duminil committed
271
		if (!features.every( f => f.get("project") === project)){
272
			dataPanel.prepend("<h2 class='error'>Sorry, the CityGML files should all belong to the same project.</h2><br/>\n");
Eric Duminil's avatar
Eric Duminil committed
273
274
			return;
		}
Eric Duminil's avatar
Eric Duminil committed
275

Eric Duminil's avatar
Eric Duminil committed
276
		if (!features.every( f => f.get("srsName") === srsName)){
277
			dataPanel.prepend("<h2 class='error'>Sorry, the CityGML files should all be written with the same coordinate system.</h2><br/>\n");
Eric Duminil's avatar
Eric Duminil committed
278
		}
Eric Duminil's avatar
Eric Duminil committed
279

280
		var citygmlNames = features.map(f => f.get("name"));
281
282
283
284
285
286
287
288
		if (proj4.defs(srsName)){
			console.log("Selected region is written in " + srsName + " coordinate system.");
			fxapp.downloadRegionFromCityGMLs(sketchAsWKT(srsName), project, citygmlNames.join(";"), srsName);
		} else {
			var msg = "ERROR : Unknown coordinate system : \"" + srsName + "\". Cannot extract any region";
			console.log(msg);
			dataPanel.append(msg + "<br/>\n");
		}
eric.duminil's avatar
eric.duminil committed
289
290
	}

291
292
	function displayInfo() {
		dataPanel.empty();
293
		var geom = sketch.getGeometry().clone().transform(sourceProj, 'EPSG:4326');
294
295
		var coordinates = geom.getLinearRing(0).getCoordinates();
		var area = Math.abs(wgs84Sphere.geodesicArea(coordinates));
296
		//NOTE: Could show m², ha or km² depending on magnitude
297
		dataPanel.append("<h3 class='clean'>Area : " + (area / 10000).toFixed(1) + " ha\n");
Eric Duminil's avatar
Eric Duminil committed
298
		dataPanel.append('<div style="visibility:hidden" id="download_region">' +
299
			'<button type="button" onclick="regionChooser.downloadFromSelectedCityGMLs()" id="download_region_button" disabled>Download Region</button><br/>\n' +
Eric Duminil's avatar
rename    
Eric Duminil committed
300
301
			'<a href="#" onclick="regionChooser.selectAllOrNone(true);">(Select All)</a>\n' +
			'<a href="#" onclick="regionChooser.selectAllOrNone(false);">(Select None)</a>\n'+
Eric Duminil's avatar
Eric Duminil committed
302
			'</div>\n');
303
		findIntersections();
Eric Duminil's avatar
Eric Duminil committed
304
		dataPanel.append('<button type="button" onclick="regionChooser.copyCoordinatesToClipboard()" id="get_wgs84" style="position:fixed; bottom:0;">Copy coordinates</button><br/>\n')
eric.duminil's avatar
eric.duminil committed
305
	}
306
307
308
309
310

	draw.on('drawend', function() {
		displayInfo();
		draw.setActive(false);
	});
Eric Duminil's avatar
Eric Duminil committed
311

312
313
	// With OpenLayers 3.9, draw_interaction.removeLastPoint(); might be better.
	document.addEventListener('keydown', function(e) {
Eric Duminil's avatar
Eric Duminil committed
314
315
		//NOTE: e.key isn't defined in JavaFX Browser
		if (e.which == 27 || e.which == 46){ // ESCAPE or DELETE.
316
317
			resetDrawing();
		}
Eric Duminil's avatar
Eric Duminil committed
318
319
320
321
322
		if (e.which == 116 && fromJavaFX){ // F5 for refresh
			dataPanel.prepend("<h2 class='ok'>Refreshing repository...</h2><br/>\n");
			document.documentElement.className = 'wait';
			fxapp.refreshHulls();
		}
323
	});
324

325
326
	function resetDrawing(){
		console.log("Reset drawing");
327
328
329
		try {
			draw.finishDrawing();
		} finally {
Eric Duminil's avatar
Eric Duminil committed
330
			displayHelp();
Eric Duminil's avatar
Eric Duminil committed
331
			document.documentElement.className = ''; // Stop waiting
332
			kml_source.getFeatures().forEach(f => refreshStyle(f, "original"));
333
			draw.setActive(true);
Eric Duminil's avatar
rename    
Eric Duminil committed
334
			drawnLayer.getFeatures().clear();
335
336
337
			intersections.clear();
			focusOnMap();
		}
338
	}
Eric Duminil's avatar
Eric Duminil committed
339

340
341
342
343
344
345
346
	function sketchAsWKT(srsName) {
		var wktFormat = new ol.format.WKT();
		return wktFormat.writeFeature(sketch, {
			dataProjection : ol.proj.get(srsName),
			featureProjection : ol.proj.get('EPSG:3857')
		});
	}
Eric Duminil's avatar
Eric Duminil committed
347

348
	function focusOnMap() {
Eric Duminil's avatar
Eric Duminil committed
349
		document.getElementById("map").focus();
350
	}
Eric Duminil's avatar
Eric Duminil committed
351

352
	var fxapp = undefined;
Eric Duminil's avatar
Eric Duminil committed
353

354
355
	publicScope.setFxApp = function(app){
		fxapp = app;
356
357
358
		console.log = function(message){
			fxapp.log(message);
		}
Eric Duminil's avatar
Eric Duminil committed
359

Eric Duminil's avatar
Eric Duminil committed
360
		console.warn = function(message){
361
362
			fxapp.warning(message);
		}
363
	}
Eric Duminil's avatar
Eric Duminil committed
364

Eric Duminil's avatar
Eric Duminil committed
365
366
	function displayHelp(){
		dataPanel.empty();
367
		dataPanel.append("<h2 class='info'>Welcome to Region Chooser!<br><br>\n");
Eric Duminil's avatar
Eric Duminil committed
368
369
370
371
		dataPanel.append("You can draw a polygon on the map by clicking.<br>\n");
		dataPanel.append("You can add a new point to an existing edge by clicking and dragging.<br>\n");
		dataPanel.append("You can remove a point with SHIFT + clicking.<br>\n");
		dataPanel.append("You can cancel drawing with ESC or DEL.<br><br>\n");
372
373
374
		dataPanel.append("After drawing a polygon which intersects with at least one GML file,<br>\n");
		dataPanel.append("you can download the corresponding part by checking the<br>\n");
		dataPanel.append("desired filenames and clicking on 'Download' button.<br>\n");
Eric Duminil's avatar
Eric Duminil committed
375
376
377
		dataPanel.append("<br>\n");
		dataPanel.append("More info is available in the ");
		dataPanel.append("<a href='http://simstadt.hft-stuttgart.de/related-softwares/region-chooser/'>SimStadt documentation</a><br>\n");
Eric Duminil's avatar
TODOs    
Eric Duminil committed
378
		//TODO: Show tooltip above.
Eric Duminil's avatar
Eric Duminil committed
379
		dataPanel.append("<form id='importWKT'>\n" +
Eric Duminil's avatar
Eric Duminil committed
380
		"<input id='wktPolygon' type='text' placeholder='WKT Polygon' " +
381
		"required pattern=' *POLYGON *\\( *\\([\\-0-9\., \)\()]+\\) *\\) *' " +
Eric Duminil's avatar
Eric Duminil committed
382
383
		"title='Please input a valid WKT Polygon. Example : POLYGON((9.961675 49.807053, 9.951375 49.798521, 9.969486 49.797746, 9.961675 49.807053)) '/>\n" +
		"<input type='submit' value='Import Polygon'/>\n" +
Eric Duminil's avatar
Eric Duminil committed
384
385
386
387
388
		"</form>\n");
		document.getElementById('importWKT').addEventListener('submit', importWKT);
	}

	importWKT = function(e){
Eric Duminil's avatar
Eric Duminil committed
389
390
391
		e.preventDefault(); // to avoid refresh


Eric Duminil's avatar
TODOs    
Eric Duminil committed
392
393
		//TODO: DRY, possibly with other events
		//TODO: Allow WKT as parameter for GUI?
Eric Duminil's avatar
Eric Duminil committed
394
		var wktPolygon = document.getElementById("wktPolygon").value;
Eric Duminil's avatar
Eric Duminil committed
395

Eric Duminil's avatar
Eric Duminil committed
396
397
		console.log("Try to import WKT geometry : " + wktPolygon);

Eric Duminil's avatar
Eric Duminil committed
398
399
		var wktFormat = new ol.format.WKT();

Eric Duminil's avatar
Eric Duminil committed
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
		try{
			var feature = wktFormat.readFeature(wktPolygon, {
				dataProjection : ol.proj.get('EPSG:4326'),
				featureProjection : ol.proj.get('EPSG:3857')
			});
		} catch (e){
			console.error("Couldn't import geometry!");
			dataPanel.prepend("<h2 class='error'>Couldn't import geometry!</h2><br/>\n");
			return false;
		}

		// Assuming the linear ring is closed
		var coordinatesCount = feature.getGeometry().getLinearRing(0).getCoordinates().length - 1;

		if (coordinatesCount < 2){
415
			console.error("Too few points!");
Eric Duminil's avatar
Eric Duminil committed
416
417
418
			dataPanel.prepend("<h2 class='error'>There should be at least 2 points in WKT polygon</h2><br/>\n");
			return false;
		}
Eric Duminil's avatar
Eric Duminil committed
419
420
421
422
423
424
425
426
427
428
429
430

		sketch = feature;
		updateGMLPolygons();

		drawnLayer.getFeatures().clear();
		intersections.clear();
		drawnLayer.addFeature(feature);
		map.getView().fitExtent(feature.getGeometry().getExtent(), map.getSize());

		displayInfo();
		draw.setActive(false);

431
432
		console.log("Import was succesful!");

Eric Duminil's avatar
Eric Duminil committed
433
		dataPanel.prepend("<h2 class='ok'>WKT Polygon succesfully imported!</h2><br/>");
Eric Duminil's avatar
Eric Duminil committed
434

Eric Duminil's avatar
Eric Duminil committed
435
	}
436

437
438
	// Executed by JavaFX when whole page is loaded.
	publicScope.ready = function() {
439
		updateGMLPolygons();
Eric Duminil's avatar
Eric Duminil committed
440
		displayHelp();
Eric Duminil's avatar
Eric Duminil committed
441
		document.documentElement.className = ''; // Stop waiting
Eric Duminil's avatar
Eric Duminil committed
442
		console.log("Ready!");
443
	}
Eric Duminil's avatar
Eric Duminil committed
444

Eric Duminil's avatar
rename    
Eric Duminil committed
445
	publicScope.selectAllOrNone = function(allOrNone) {
Eric Duminil's avatar
Eric Duminil committed
446
		document.querySelectorAll("input.select_citygml").forEach(c => c.checked = allOrNone);
447
		publicScope.isDownloadPossible();
Eric Duminil's avatar
Eric Duminil committed
448
	}
Eric Duminil's avatar
Eric Duminil committed
449

450
	publicScope.selectRepository = function() {
451
		fxapp.selectRepository();
452
	}
Eric Duminil's avatar
Eric Duminil committed
453

454
455
456
457
	publicScope.copyCoordinatesToClipboard = function(){
		var geom = sketch.getGeometry().clone().transform(sourceProj, 'EPSG:4326');
		var wgs84Coords = geom.getLinearRing(0).getCoordinates();
		var wktPolygon = "POLYGON((";
458
459
		var precision = 6; // ~ 10 cm precision
		wktPolygon += wgs84Coords.map(([lon, lat]) => lon.toFixed(precision) + " " + lat.toFixed(precision)).join(", ");
Eric Duminil's avatar
Eric Duminil committed
460
		utils.copyToClipboard(wktPolygon + "))", dataPanel);
461
	}
Eric Duminil's avatar
Eric Duminil committed
462

Eric Duminil's avatar
Eric Duminil committed
463
464
465
	publicScope.showRepositoryName = function(path) {
		document.getElementById("repo_path").textContent = path;
	}
466

467
	focusOnMap();
Eric Duminil's avatar
Eric Duminil committed
468
469
	//var regionChooser = publicScope; //NOTE: In order to open closure. For debugging
	return publicScope;
470
})();