simstadt_openlayers.js 14 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;
14
15
16
17
18
19
	var gmlId;
	
	publicScope.init = function(){
		//NOTE: Only called from JavaFX. At startup, or when Repo has been changed.
		gmlId = 0;
		kml_source.clear();
Eric Duminil's avatar
Eric Duminil committed
20
		document.getElementById("select_repository").style.visibility = "visible";
21
	}
22
23
	
	if (fromJavaFX){
Eric Duminil's avatar
Eric Duminil committed
24
		document.documentElement.className = 'wait';
25
	}
26
27
28

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

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

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

	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
46
47
		})
	});
48

49
	publicScope.addCityGmlHull = function(kmlString) {
50
		options = {featureProjection: ol.proj.get('EPSG:3857')};
51
		feature = kmlFormat.readFeature(kmlString, options);
52
		feature.setId(gmlId++);
53
54
		kml_source.addFeature(feature);
		dataPanel.append('.');
55
		srsName = feature.get("srsName");
56
		if (proj4.defs(srsName) === undefined){
Eric Duminil's avatar
Eric Duminil committed
57
			console.warn(srsName + " isn't defined by Proj4js!")
58
		}
59
60
	};
	
61
62
	var map = new ol.Map({
		target : 'map',
63
		layers : [ osm_layer, kml_layer, intersections_layer ],
64
65
66
67
68
		interactions : ol.interaction.defaults({
			keyboard : true
		})
	});

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

72
	kml_source.addEventListener("addfeature", function() {
73
74
75
76
77
		map.getView().fitExtent(kml_source.getExtent(), (map.getSize()));
	});

	function updateGMLPolygons() {
		kml_source.forEachFeature(function(feature) {
78
			feature["geoJSON"] = geoJsonFormat.writeFeatureObject(feature);
79
			feature["area"] = feature.getGeometry().getArea();
Eric Duminil's avatar
Eric Duminil committed
80
81
			feature["project"] = feature.get("project");
			feature["name"] = feature.get("name");
82
			feature["source"] = "CityGML";
Eric Duminil's avatar
Eric Duminil committed
83
			feature["status"] = "original";
84
		});
Eric Duminil's avatar
Eric Duminil committed
85
		
Eric Duminil's avatar
Eric Duminil committed
86
87
88
		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
89
		features_by_project = utils.groupBy(features, "project");
Eric Duminil's avatar
Eric Duminil committed
90
91
		// Sort CityGMLs inside each project
		Object.values(features_by_project).forEach(features => features.sort((a, b) => a.name.localeCompare(b.name)));
92
93
94
95
96
97
	}

	// 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
98
	var drawnLayer = new ol.FeatureOverlay({
99
		style : new ol.style.Style({
eric.duminil's avatar
eric.duminil committed
100
			fill : new ol.style.Fill({
101
102
103
104
105
106
107
108
109
110
111
				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
112
113
114
			})
		})
	});
Eric Duminil's avatar
rename    
Eric Duminil committed
115
	drawnLayer.setMap(map);
116

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

	var modify = new ol.interaction.Modify({
Eric Duminil's avatar
rename    
Eric Duminil committed
125
		features : drawnLayer.getFeatures(),
126
127
128
129
130
131
132
133
134
135
		// 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
136
		features : drawnLayer.getFeatures(),
137
138
139
140
141
		type : 'Polygon'
	});
	map.addInteraction(draw);

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

143
144
145
146
147
	draw.on('drawstart', function(evt) {
		sketch = evt.feature;
		updateGMLPolygons();
	});
	var sourceProj = map.getView().getProjection();
148
149
150
151
152
153
	
	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);
154
		var id = feature.getId();
155
		intersections.addFeature(intersection);
156
157
158
		var link = '<li onmouseover="regionChooser.highlightPolygon(' + id + ')" onmouseout="regionChooser.resetHighlight(' + id +')">';
		link += '<input type="checkbox" id="citygml_' + feature.getId() + '" class="select_citygml" onclick="regionChooser.isDownloadPossible();">'
		 	+ '<label for="citygml_' + feature.getId() + '">' + feature['name'] + '</label>';
159
160
161
162
163
164
165
166

		link += " (" + citygml_percentage + "%";
		if (sketch_percentage == 100) {
			link += ", all inside";
		}
		dataPanel.append(link + ")\n");
	}
	
167
168
	publicScope.highlightPolygon = function(i) {
		var feature = kml_source.getFeatureById(i);
Eric Duminil's avatar
Eric Duminil committed
169
		feature.setStyle(styles.highlighted);
170
171
172
173
	}
	
	publicScope.resetHighlight = function(i) {
		var feature = kml_source.getFeatureById(i);
Eric Duminil's avatar
Eric Duminil committed
174
175
176
177
178
		refreshStyle(feature);
	}
	
	refreshStyle = function(feature){
		feature.setStyle(styles[feature.status]);
179
180
	}
	
181
	publicScope.isDownloadPossible = function(){
Eric Duminil's avatar
Eric Duminil committed
182
		kml_source.getFeatures().forEach(f => {f.status = "original"; refreshStyle(f)});
Eric Duminil's avatar
Eric Duminil committed
183
184
		
		//TODO: Dry
185
		var checkedBoxes = Array.from(document.querySelectorAll("input.select_citygml")).filter(c => c.checked);
Eric Duminil's avatar
Eric Duminil committed
186
187
		var checkbox_ids = checkedBoxes.map(c => c.id);
		var features = getCheckedPolygons(checkbox_ids);
Eric Duminil's avatar
Eric Duminil committed
188
		features.forEach(f => {f.status = "selected"; refreshStyle(f)});
Eric Duminil's avatar
Eric Duminil committed
189
		
190
		document.getElementById("download_region_button").disabled = (checkedBoxes.length == 0);
Eric Duminil's avatar
Eric Duminil committed
191
192
	}
	
193
	function findIntersection(feature, polygon) {
194
		try {
195
			return turf.intersect(polygon, feature["geoJSON"]);
196
197
198
199
		} catch (err) {
			console.log(feature.get('name') + " - " + err);
		}
	}
200
201

	function findIntersections() {
202
203
		var polygon = geoJsonFormat.writeFeatureObject(sketch);
		var polygonArea = sketch.getGeometry().getArea();
204
205
		var intersection_found = false;
		intersections.clear();
206
		
207
		Object.keys(features_by_project).forEach(function(project) {
208
209
210
211
			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;
212
				dataPanel.append("<h2 class='info'>" + project);
213
214
				features_and_intersections.forEach(l => showLinkToDownload(l[0], l[1], polygonArea));
			}
215
		});
216
		
217
		if (intersection_found) {
218
			document.getElementById("download_region").style.visibility = 'visible';
219
220
		}
		else {
221
			document.getElementById("download_region").style.visibility = 'hidden';
222
			dataPanel.append("No intersection found with any CityGML file.<br/>\n");
eric.duminil's avatar
eric.duminil committed
223
224
		}
	}
225

226
227
228
229
	publicScope.display = function(text){
		dataPanel.append(text + "<br/>\n");
	}
	
Eric Duminil's avatar
Eric Duminil committed
230
231
	getCheckedPolygons = function(checkbox_ids){
		return checkbox_ids.map(checkbox_id => {
232
233
234
			var i = Number(checkbox_id.replace("citygml_", ""));
			return kml_source.getFeatureById(i);
		})
Eric Duminil's avatar
Eric Duminil committed
235
236
237
238
	}
	
	publicScope.downloadRegionFromCityGMLs = function(checkbox_ids) {
		var features = getCheckedPolygons(checkbox_ids);
239
		
Eric Duminil's avatar
Eric Duminil committed
240
241
242
243
		var project = features[0].get("project");
		var srsName = features[0].get("srsName");
		
		if (!features.every( f => f.get("project") === project)){
244
			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
245
246
247
248
			return;
		}
		
		if (!features.every( f => f.get("srsName") === srsName)){
249
			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
250
251
		}
		
Eric Duminil's avatar
Eric Duminil committed
252
253
		document.documentElement.className = 'wait';
		
254
		var citygmlNames = features.map(f => f.get("name"));
255
256
257
258
259
		// Waiting 100ms in order to let the cursor change
		setTimeout(function() {
			var start = new Date().getTime();
			if (proj4.defs(srsName)){
				console.log("Selected region is written in " + srsName + " coordinate system.");
260
				try {
Eric Duminil's avatar
Eric Duminil committed
261
					var count = fxapp.downloadRegionFromCityGMLs(sketchAsWKT(srsName), project, citygmlNames.join(";"), srsName);
Eric Duminil's avatar
Eric Duminil committed
262
263
264
265
266
					if (count == -1){
						console.log("No output file has been selected.");
					} else {
						dataPanel.prepend("<h2 class='ok'>Done! (" + count + " buildings found) </h2><br/>\n");
					}
267
				} catch (e) {
Eric Duminil's avatar
Eric Duminil committed
268
					console.warn("ERROR : " + e);
269
					dataPanel.prepend("<h2 class='error'>Some problem occured!</h2><br/>\n");
270
				}
271
272
				var end = new Date().getTime();
				var time = end - start;
Eric Duminil's avatar
Eric Duminil committed
273
				console.log('Download Execution time: ' + (time / 1000).toFixed(3) + 's');
274
				setTimeout(function() {
275
					document.getElementById("download_region_button").disabled = false;
Eric Duminil's avatar
Eric Duminil committed
276
					document.documentElement.className = ''; // Stop waiting
277
278
279
280
281
282
283
				}, 100);
			} else {
				var msg = "ERROR : Unknown coordinate system : \"" + srsName + "\". Cannot extract any region";
				console.log(msg);
				dataPanel.append(msg + "<br/>\n");
			}
		}, 100);
eric.duminil's avatar
eric.duminil committed
284
285
	}

286
287
	function displayInfo() {
		dataPanel.empty();
288
		var geom = sketch.getGeometry().clone().transform(sourceProj, 'EPSG:4326');
289
290
		var coordinates = geom.getLinearRing(0).getCoordinates();
		var area = Math.abs(wgs84Sphere.geodesicArea(coordinates));
291
		//NOTE: Could show m², ha or km² depending on magnitude
292
		dataPanel.append("<h3 class='clean'>Area : " + (area / 10000).toFixed(1) + " ha\n");
293
294
		dataPanel.append('<div style="visibility:hidden" id="download_region">' + 
			'<button type="button" onclick="regionChooser.downloadFromSelectedCityGMLs()" id="download_region_button" disabled>Download Region</button><br/>\n' +
Eric Duminil's avatar
rename    
Eric Duminil committed
295
296
			'<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
297
			'</div>\n');
298
		findIntersections();
Eric Duminil's avatar
Eric Duminil committed
299
		dataPanel.append('<button type="button" onclick="regionChooser.copyCoordinatesToClipboard()" id="get_wgs84">Copy coordinates</button><br/>\n')
eric.duminil's avatar
eric.duminil committed
300
	}
301
302
303
304
305

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

320
321
	function resetDrawing(){
		console.log("Reset drawing");
322
323
324
		try {
			draw.finishDrawing();
		} finally {
Eric Duminil's avatar
Eric Duminil committed
325
			displayHelp();
Eric Duminil's avatar
Eric Duminil committed
326
			document.documentElement.className = ''; // Stop waiting
327
			draw.setActive(true);
Eric Duminil's avatar
rename    
Eric Duminil committed
328
			drawnLayer.getFeatures().clear();
329
330
331
			intersections.clear();
			focusOnMap();
		}
332
	}
333
	
334
335
336
337
338
339
340
	function sketchAsWKT(srsName) {
		var wktFormat = new ol.format.WKT();
		return wktFormat.writeFeature(sketch, {
			dataProjection : ol.proj.get(srsName),
			featureProjection : ol.proj.get('EPSG:3857')
		});
	}
341
	
342
	function focusOnMap() {
Eric Duminil's avatar
Eric Duminil committed
343
		document.getElementById("map").focus();
344
345
	}
	
346
347
348
349
	var fxapp = undefined;
	
	publicScope.setFxApp = function(app){
		fxapp = app;
350
351
352
		console.log = function(message){
			fxapp.log(message);
		}
353
		
Eric Duminil's avatar
Eric Duminil committed
354
		console.warn = function(message){
355
356
			fxapp.warning(message);
		}
357
	}
Eric Duminil's avatar
Eric Duminil committed
358
359
360
	
	function displayHelp(){
		dataPanel.empty();
361
		dataPanel.append("<h2 class='info'>Welcome to Region Chooser!<br><br>\n");
Eric Duminil's avatar
Eric Duminil committed
362
363
364
365
		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");
366
367
368
		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
369
370
371
		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
Eric Duminil committed
372
	}
373

374
375
	// Executed by JavaFX when whole page is loaded.
	publicScope.ready = function() {
376
		updateGMLPolygons();
Eric Duminil's avatar
Eric Duminil committed
377
		displayHelp();
Eric Duminil's avatar
Eric Duminil committed
378
		document.documentElement.className = ''; // Stop waiting
Eric Duminil's avatar
Eric Duminil committed
379
		console.log("Ready!");
380
	}
381
	
382
	publicScope.downloadFromSelectedCityGMLs = function() {
Eric Duminil's avatar
Eric Duminil committed
383
		document.getElementById("download_region_button").disabled = true;
384
		var checkedBoxes = Array.from(document.querySelectorAll("input.select_citygml")).filter(c => c.checked);
Eric Duminil's avatar
Eric Duminil committed
385
		// CheckBoxes isn't empty, because otherwise the button cannot be clicked.
Eric Duminil's avatar
Eric Duminil committed
386
387
388
389
		
		var checkbox_ids = checkedBoxes.map(c => c.id);
		var features = getCheckedPolygons(checkbox_ids);
		
Eric Duminil's avatar
Eric Duminil committed
390
		features.forEach(f => f.setStyle(utils.polygon_style("#ffff00", 0.8)));
Eric Duminil's avatar
Eric Duminil committed
391

Eric Duminil's avatar
Eric Duminil committed
392
		publicScope.downloadRegionFromCityGMLs(checkedBoxes.map(c => c.id));
393
	}
394
	
Eric Duminil's avatar
rename    
Eric Duminil committed
395
	publicScope.selectAllOrNone = function(allOrNone) {
Eric Duminil's avatar
Eric Duminil committed
396
		document.querySelectorAll("input.select_citygml").forEach(c => c.checked = allOrNone);
397
		publicScope.isDownloadPossible();
Eric Duminil's avatar
Eric Duminil committed
398
399
	}
	
400
	publicScope.selectRepository = function() {
401
		fxapp.selectRepository();
402
	}
Eric Duminil's avatar
Eric Duminil committed
403
	
404
405
406
407
408
	publicScope.copyCoordinatesToClipboard = function(){
		var geom = sketch.getGeometry().clone().transform(sourceProj, 'EPSG:4326');
		var wgs84Coords = geom.getLinearRing(0).getCoordinates();
		var wktPolygon = "POLYGON((";
		wktPolygon += wgs84Coords.map(lonLat => lonLat.join(" ")).join(", ");
Eric Duminil's avatar
Eric Duminil committed
409
		utils.copyToClipboard(wktPolygon + "))", dataPanel);
410
411
	}
	
Eric Duminil's avatar
Eric Duminil committed
412
413
414
	publicScope.showRepositoryName = function(path) {
		document.getElementById("repo_path").textContent = path;
	}
415

416
	focusOnMap();
Eric Duminil's avatar
Eric Duminil committed
417
418
	//var regionChooser = publicScope; //NOTE: In order to open closure. For debugging
	return publicScope;
419
})();