forked from MapComplete/MapComplete
		
	SplitAction logic, not yet pushing changes to osm, pieter will take over
This commit is contained in:
		
							parent
							
								
									159e4d3350
								
							
						
					
					
						commit
						f77c1efdf5
					
				
					 6 changed files with 262 additions and 29 deletions
				
			
		|  | @ -280,7 +280,7 @@ export class GeoOperations { | ||||||
|      * @param point Point defined as [lon, lat] |      * @param point Point defined as [lon, lat] | ||||||
|      */ |      */ | ||||||
|     public static nearestPoint(way, point: [number, number]){ |     public static nearestPoint(way, point: [number, number]){ | ||||||
|         return turf.nearestPointOnLine(way, point); |         return turf.nearestPointOnLine(way, point, {units: "kilometers"}); | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -1,4 +1,4 @@ | ||||||
| import {OsmNode, OsmObject} from "./OsmObject"; | import {OsmNode, OsmObject, OsmWay} from "./OsmObject"; | ||||||
| import State from "../../State"; | import State from "../../State"; | ||||||
| import {Utils} from "../../Utils"; | import {Utils} from "../../Utils"; | ||||||
| import {UIEventSource} from "../UIEventSource"; | import {UIEventSource} from "../UIEventSource"; | ||||||
|  | @ -86,6 +86,14 @@ export class Changes implements FeatureSource{ | ||||||
|         this.uploadAll([], this.pending.data); |         this.uploadAll([], this.pending.data); | ||||||
|         this.pending.setData([]); |         this.pending.setData([]); | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Returns a new ID and updates the value for the next ID | ||||||
|  |      */ | ||||||
|  |     public getNewID(){ | ||||||
|  |         return Changes._nextId--; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     /** |     /** | ||||||
|      * Create a new node element at the given lat/long. |      * Create a new node element at the given lat/long. | ||||||
|      * An internal OsmObject is created to upload later on, a geojson represention is returned. |      * An internal OsmObject is created to upload later on, a geojson represention is returned. | ||||||
|  | @ -93,8 +101,7 @@ export class Changes implements FeatureSource{ | ||||||
|      */ |      */ | ||||||
|     public createElement(basicTags: Tag[], lat: number, lon: number) { |     public createElement(basicTags: Tag[], lat: number, lon: number) { | ||||||
|         console.log("Creating a new element with ", basicTags) |         console.log("Creating a new element with ", basicTags) | ||||||
|         const osmNode = new OsmNode(Changes._nextId); |         const osmNode = new OsmNode(this.getNewID()); | ||||||
|         Changes._nextId--; |  | ||||||
| 
 | 
 | ||||||
|         const id = "node/" + osmNode.id; |         const id = "node/" + osmNode.id; | ||||||
|         osmNode.lat = lat; |         osmNode.lat = lat; | ||||||
|  | @ -114,6 +121,56 @@ export class Changes implements FeatureSource{ | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  |         const changes = this.createTagChangeList(basicTags, properties, id); | ||||||
|  |         | ||||||
|  |         console.log("New feature added and pinged") | ||||||
|  |         this.features.data.push({feature:geojson, freshness: new Date()}); | ||||||
|  |         this.features.ping(); | ||||||
|  |          | ||||||
|  |         State.state.allElements.addOrGetElement(geojson).ping(); | ||||||
|  | 
 | ||||||
|  |         this.uploadAll([osmNode], changes); | ||||||
|  |         return geojson; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Creates a new road with given tags that consist of the points corresponding to given nodeIDs | ||||||
|  |      * @param basicTags The tags to add to the road | ||||||
|  |      * @param nodeIDs IDs of nodes of which the road consists. Those nodes must already exist in osm or already be added to the changeset. | ||||||
|  |      * @param coordinates The coordinates correspoinding to the nodeID at the same index. Each coordinate is a [lon, lat] point | ||||||
|  |      * @return geojson A geojson representation of the created road | ||||||
|  |      */ | ||||||
|  |     public createRoad(basicTags: Tag[], nodeIDs, coordinates) { | ||||||
|  |         const osmWay = new OsmWay(this.getNewID()); | ||||||
|  | 
 | ||||||
|  |         const id = "way/" + osmWay.id; | ||||||
|  |         osmWay.nodes = nodeIDs; | ||||||
|  |         const properties = {id: id}; | ||||||
|  | 
 | ||||||
|  |         const geojson = { | ||||||
|  |             "type": "Feature", | ||||||
|  |             "properties": properties, | ||||||
|  |             "id": id, | ||||||
|  |             "geometry": { | ||||||
|  |                 "type": "LineString", | ||||||
|  |                 "coordinates": coordinates | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         const changes = this.createTagChangeList(basicTags, properties, id); | ||||||
|  | 
 | ||||||
|  |         console.log("New feature added and pinged") | ||||||
|  |         this.features.data.push({feature:geojson, freshness: new Date()}); | ||||||
|  |         this.features.ping(); | ||||||
|  | 
 | ||||||
|  |         State.state.allElements.addOrGetElement(geojson).ping(); | ||||||
|  | 
 | ||||||
|  |         this.uploadAll([osmWay], changes); | ||||||
|  |         return geojson; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     private createTagChangeList(basicTags: Tag[], properties: { id: string }, id: string) { | ||||||
|         // The basictags are COPIED, the id is included in the properties
 |         // The basictags are COPIED, the id is included in the properties
 | ||||||
|         // The tags are not yet written into the OsmObject, but this is applied onto a
 |         // The tags are not yet written into the OsmObject, but this is applied onto a
 | ||||||
|         const changes = []; |         const changes = []; | ||||||
|  | @ -124,15 +181,7 @@ export class Changes implements FeatureSource{ | ||||||
|             } |             } | ||||||
|             changes.push({elementId: id, key: kv.key, value: kv.value}) |             changes.push({elementId: id, key: kv.key, value: kv.value}) | ||||||
|         } |         } | ||||||
|         |         return changes; | ||||||
|         console.log("New feature added and pinged") |  | ||||||
|         this.features.data.push({feature:geojson, freshness: new Date()}); |  | ||||||
|         this.features.ping(); |  | ||||||
|          |  | ||||||
|         State.state.allElements.addOrGetElement(geojson).ping(); |  | ||||||
| 
 |  | ||||||
|         this.uploadAll([osmNode], changes); |  | ||||||
|         return geojson; |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private uploadChangesWithLatestVersions( |     private uploadChangesWithLatestVersions( | ||||||
|  | @ -244,4 +293,13 @@ export class Changes implements FeatureSource{ | ||||||
|         }) |         }) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Changes the nodes of road with given id to the given nodes | ||||||
|  |      * @param roadID The ID of the road to update | ||||||
|  |      * @param newNodes The node id's the road consists of (should already be added to the changeset or in osm) | ||||||
|  |      */ | ||||||
|  |     public updateRoadCoordinates(roadID: string, newNodes: number[]) { | ||||||
|  |         // TODO
 | ||||||
|  |     } | ||||||
| } | } | ||||||
|  | @ -444,7 +444,7 @@ export class OsmWay extends OsmObject { | ||||||
|         this.nodes = element.nodes; |         this.nodes = element.nodes; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     asGeoJson() { |     public asGeoJson() { | ||||||
|         return { |         return { | ||||||
|             "type": "Feature", |             "type": "Feature", | ||||||
|             "properties": this.tags, |             "properties": this.tags, | ||||||
|  |  | ||||||
							
								
								
									
										162
									
								
								Logic/Osm/SplitAction.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										162
									
								
								Logic/Osm/SplitAction.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,162 @@ | ||||||
|  | import {UIEventSource} from "../UIEventSource"; | ||||||
|  | import {OsmNode, OsmObject, OsmWay} from "./OsmObject"; | ||||||
|  | import State from "../../State"; | ||||||
|  | import {distance} from "@turf/turf"; | ||||||
|  | import {GeoOperations} from "../GeoOperations"; | ||||||
|  | import {Changes} from "./Changes"; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Splits a road in different segments, each splitted at one of the given points (or a point on the road close to it) | ||||||
|  |  * @param roadID The id of the road you want to split | ||||||
|  |  * @param points The points on the road where you want the split to occur (geojson point list) | ||||||
|  |  */ | ||||||
|  | export async function splitRoad(roadID, points) { | ||||||
|  |     if (points.length != 1) { | ||||||
|  |         // TODO: more than one point
 | ||||||
|  |         console.log(points) | ||||||
|  |         window.alert("Warning, currently only tested on one point, you selected " + points.length + " points") | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     let road = State.state.allElements.ContainingFeatures.get(roadID); | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Compares two points based on the starting point of the road, can be used in sort function | ||||||
|  |      * @param point1 [lon, lat] point | ||||||
|  |      * @param point2 [lon, lat] point | ||||||
|  |      */ | ||||||
|  |     function comparePointDistance(point1, point2) { | ||||||
|  |         let distFromStart1 = GeoOperations.nearestPoint(road, point1).properties.location; | ||||||
|  |         let distFromStart2 = GeoOperations.nearestPoint(road, point2).properties.location; | ||||||
|  |         return distFromStart1 - distFromStart2; // Sort requires a number to return instead of a bool
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Eliminates split points close (<4m) to existing points on the road, so you can split on these points instead | ||||||
|  |      * @param road The road geojson object | ||||||
|  |      * @param points The points on the road where you want the split to occur (geojson point list) | ||||||
|  |      * @return realSplitPoints List containing all new locations where you should split | ||||||
|  |      */ | ||||||
|  |     function getSplitPoints(road, points) { | ||||||
|  |         // Copy the list
 | ||||||
|  |         let roadPoints = [...road.geometry.coordinates]; | ||||||
|  | 
 | ||||||
|  |         // Get the coordinates of all geojson points
 | ||||||
|  |         let splitPointsCoordinates = points.map((point) => point.geometry.coordinates); | ||||||
|  | 
 | ||||||
|  |         roadPoints.push(...splitPointsCoordinates); | ||||||
|  | 
 | ||||||
|  |         // Sort all points on the road based on the distance from the start
 | ||||||
|  |         roadPoints.sort(comparePointDistance) | ||||||
|  | 
 | ||||||
|  |         // Remove points close to existing points on road
 | ||||||
|  |         let realSplitPoints = [...splitPointsCoordinates]; | ||||||
|  |         for (let index = roadPoints.length - 1; index > 0; index--) { | ||||||
|  |             // Iterate backwards to prevent problems when removing elements
 | ||||||
|  |             let dist = distance(roadPoints[index - 1], roadPoints[index], {units: "kilometers"}); | ||||||
|  |             // Remove all cutpoints closer than 4m to their previous point
 | ||||||
|  |             if ((dist < 0.004) && (splitPointsCoordinates.includes(roadPoints[index]))) { | ||||||
|  |                 console.log("Removed a splitpoint, using a closer point to the road instead") | ||||||
|  |                 realSplitPoints.splice(index, 1) | ||||||
|  |                 realSplitPoints.push(roadPoints[index - 1]) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return realSplitPoints; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     let realSplitPoints = getSplitPoints(road, points); | ||||||
|  | 
 | ||||||
|  |     // Create a sorted list containing all points
 | ||||||
|  |     let allPoints = [...road.geometry.coordinates]; | ||||||
|  |     allPoints.push(...realSplitPoints); | ||||||
|  |     allPoints.sort(comparePointDistance); | ||||||
|  | 
 | ||||||
|  |     // The changeset that will contain the operations to split the road
 | ||||||
|  |     let changes = new Changes(); | ||||||
|  | 
 | ||||||
|  |     // Download the data of the current road from Osm to get the ID's of the coordinates
 | ||||||
|  |     let osmRoad: UIEventSource<OsmWay> = OsmObject.DownloadObject(roadID); | ||||||
|  | 
 | ||||||
|  |     // TODO: Remove delay, use a callback on odmRoad instead and execute all code below in callback function
 | ||||||
|  |     function delay(ms: number) { | ||||||
|  |         return new Promise(resolve => setTimeout(resolve, ms)); | ||||||
|  |     } | ||||||
|  |     await delay(3000); | ||||||
|  | 
 | ||||||
|  |     // Dict to quickly convert a coordinate to a nodeID
 | ||||||
|  |     let coordToIDMap = {}; | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Converts a coordinate to a string, so it's hashable (e.g. for using it in a dict) | ||||||
|  |      * @param coord [lon, lat] point | ||||||
|  |      */ | ||||||
|  |     function getCoordKey(coord: [number, number]) { | ||||||
|  |         return coord[0] + "," + coord[1]; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     osmRoad.data.coordinates.forEach((coord, i) => coordToIDMap[getCoordKey([coord[1], coord[0]])] = osmRoad.data.nodes[i]); | ||||||
|  | 
 | ||||||
|  |     let currentRoadPoints: number[] = []; | ||||||
|  |     let currentRoadCoordinates: [number, number][] = [] | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Given a coordinate, check whether there is already a node in osm created (on the road or cutpoints) or create | ||||||
|  |      * such point if it doesn't exist yet and return the id of this coordinate | ||||||
|  |      * @param coord [lon, lat] point | ||||||
|  |      * @return pointID The ID of the existing/created node on given coordinates | ||||||
|  |      */ | ||||||
|  |     function getOrCreateNodeID(coord) { | ||||||
|  |         console.log(coordToIDMap) | ||||||
|  |         let poinID = coordToIDMap[getCoordKey(coord)]; | ||||||
|  |         if (poinID == undefined) { | ||||||
|  |             console.log(getCoordKey(coord) + " not in map") | ||||||
|  |             // TODO: Check if lat, lon is correct
 | ||||||
|  |             let newNode = changes.createElement([], coord[1], coord[0]); | ||||||
|  | 
 | ||||||
|  |             coordToIDMap[coord] = newNode.id; | ||||||
|  |             poinID = newNode.id; | ||||||
|  | 
 | ||||||
|  |             console.log("New point created "); | ||||||
|  |         } | ||||||
|  |         return poinID; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Creates a new road in OSM, while copying the tags from osmRoad and using currentRoadPoints as points | ||||||
|  |      * @param currentRoadPoints List of id's of nodes the road should exist of | ||||||
|  |      * @param osmRoad The road to copy the tags from | ||||||
|  |      */ | ||||||
|  |     function createNewRoadSegment(currentRoadPoints, osmRoad) { | ||||||
|  |         changes.createRoad(osmRoad.data.tags, currentRoadPoints, currentRoadCoordinates); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     for (let coord of allPoints) { | ||||||
|  |         console.log("Handling coord") | ||||||
|  |         let pointID = getOrCreateNodeID(coord); | ||||||
|  |         currentRoadPoints.push(pointID); | ||||||
|  |         currentRoadCoordinates.push(coord); | ||||||
|  |         if (realSplitPoints.includes(coord)) { | ||||||
|  |             console.log("Handling split") | ||||||
|  |             // Should split here
 | ||||||
|  |             // currentRoadPoints contains a list containing all points for this road segment
 | ||||||
|  |             createNewRoadSegment(currentRoadPoints, osmRoad); | ||||||
|  | 
 | ||||||
|  |             // Cleanup for next split
 | ||||||
|  |             currentRoadPoints = [pointID]; | ||||||
|  |             currentRoadCoordinates = [coord]; | ||||||
|  |             console.log("Splitting here...") | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // Update the road to contain only the points of the last segment
 | ||||||
|  |     // changes.updateRoadCoordinates(roadID, currentRoadPoints);
 | ||||||
|  | 
 | ||||||
|  |     // push the applied changes
 | ||||||
|  |     changes.flushChanges(); | ||||||
|  | 
 | ||||||
|  |     return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | // TODO: Vlakbij bestaand punt geklikt? Bestaand punt hergebruiken
 | ||||||
|  | //  Nieuw wegobject aanmaken, en oude hergebruiken voor andere helft van de weg
 | ||||||
|  | // TODO: CHeck if relation exist to the road -> Delete them when splitted, because they might be outdated after the split
 | ||||||
|  | @ -11,6 +11,7 @@ import LayerConfig from "../../Customizations/JSON/LayerConfig"; | ||||||
| import Combine from "../Base/Combine"; | import Combine from "../Base/Combine"; | ||||||
| import {Button} from "../Base/Button"; | import {Button} from "../Base/Button"; | ||||||
| import Translations from "../i18n/Translations"; | import Translations from "../i18n/Translations"; | ||||||
|  | import {splitRoad} from "../../Logic/Osm/SplitAction"; | ||||||
| 
 | 
 | ||||||
| export default class SplitRoadWizard extends Toggle { | export default class SplitRoadWizard extends Toggle { | ||||||
|     /** |     /** | ||||||
|  | @ -87,7 +88,7 @@ export default class SplitRoadWizard extends Toggle { | ||||||
|             State.state.osmConnection.isLoggedIn) |             State.state.osmConnection.isLoggedIn) | ||||||
| 
 | 
 | ||||||
|         // Save button
 |         // Save button
 | ||||||
|         const saveButton =  new Button("Split here", () => window.alert("Splitting...")); |         const saveButton =  new Button("Split here", () => splitRoad(id, splitPositions.data)); | ||||||
|         saveButton.SetClass("block btn btn-primary"); |         saveButton.SetClass("block btn btn-primary"); | ||||||
|         const disabledSaveButton = new Button("Split here", undefined); |         const disabledSaveButton = new Button("Split here", undefined); | ||||||
|         disabledSaveButton.SetClass("block btn btn-disabled"); |         disabledSaveButton.SetClass("block btn btn-disabled"); | ||||||
|  | @ -98,6 +99,7 @@ export default class SplitRoadWizard extends Toggle { | ||||||
|             splitClicked.setData(false); |             splitClicked.setData(false); | ||||||
| 
 | 
 | ||||||
|             splitPositions.setData([]); |             splitPositions.setData([]); | ||||||
|  |             // Only keep showing the road, the cutpoints must be removed from the map
 | ||||||
|             roadEventSource.setData([roadEventSource.data[0]]) |             roadEventSource.setData([roadEventSource.data[0]]) | ||||||
|         }); |         }); | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
							
								
								
									
										37
									
								
								test.ts
									
										
									
									
									
								
							
							
						
						
									
										37
									
								
								test.ts
									
										
									
									
									
								
							|  | @ -5,32 +5,43 @@ import {AllKnownLayouts} from "./Customizations/AllKnownLayouts"; | ||||||
| const way = { | const way = { | ||||||
|     "type": "Feature", |     "type": "Feature", | ||||||
|     "properties": { |     "properties": { | ||||||
|         "id": "way/1234", |  | ||||||
|         "highway": "residential", |         "highway": "residential", | ||||||
|         "cyclestreet": "yes" |         "maxweight": "3.5", | ||||||
|  |         "maxweight:conditional": "none @ delivery", | ||||||
|  |         "name": "Silsstraat", | ||||||
|  |         "_last_edit:contributor": "Jorisbo", | ||||||
|  |         "_last_edit:contributor:uid": 1983103, | ||||||
|  |         "_last_edit:changeset": 70963524, | ||||||
|  |         "_last_edit:timestamp": "2019-06-05T18:20:44Z", | ||||||
|  |         "_version_number": 9, | ||||||
|  |         "id": "way/23583625" | ||||||
|     }, |     }, | ||||||
|     "geometry": { |     "geometry": { | ||||||
|         "type": "LineString", |         "type": "LineString", | ||||||
|         "coordinates": [ |         "coordinates": [ | ||||||
|             [ |             [ | ||||||
|                 4.488961100578308, |                 4.4889691, | ||||||
|                 51.204971024401374 |                 51.2049831 | ||||||
|             ], |             ], | ||||||
|             [ |             [ | ||||||
|                 4.4896745681762695, |                 4.4895496, | ||||||
|                 51.204712226516435 |                 51.2047718 | ||||||
|             ], |             ], | ||||||
|             [ |             [ | ||||||
|                 4.489814043045044, |                 4.48966, | ||||||
|                 51.20459459063348 |                 51.2047147 | ||||||
|             ], |             ], | ||||||
|             [ |             [ | ||||||
|                 4.48991060256958, |                 4.4897439, | ||||||
|                 51.204439983016115 |                 51.2046548 | ||||||
|             ], |             ], | ||||||
|             [ |             [ | ||||||
|                 4.490291476249695, |                 4.4898162, | ||||||
|                 51.203845074952376 |                 51.2045921 | ||||||
|  |             ], | ||||||
|  |             [ | ||||||
|  |                 4.4902997, | ||||||
|  |                 51.2038418 | ||||||
|             ] |             ] | ||||||
|         ] |         ] | ||||||
|     } |     } | ||||||
|  | @ -39,4 +50,4 @@ const way = { | ||||||
| State.state = new State(AllKnownLayouts.allKnownLayouts.get("fietsstraten")); | State.state = new State(AllKnownLayouts.allKnownLayouts.get("fietsstraten")); | ||||||
| // add road to state
 | // add road to state
 | ||||||
| State.state.allElements.addOrGetElement(way); | State.state.allElements.addOrGetElement(way); | ||||||
| new SplitRoadWizard("way/1234").AttachTo("maindiv") | new SplitRoadWizard("way/23583625").AttachTo("maindiv") | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue