forked from MapComplete/MapComplete
196 lines
No EOL
6.3 KiB
TypeScript
196 lines
No EOL
6.3 KiB
TypeScript
import {OsmNode, OsmRelation, OsmWay} from "../OsmObject";
|
|
|
|
/**
|
|
* Represents a single change to an object
|
|
*/
|
|
export interface ChangeDescription {
|
|
|
|
/**
|
|
* Metadata to be included in the changeset
|
|
*/
|
|
meta: {
|
|
/*
|
|
* The theme with which this changeset was made
|
|
*/
|
|
theme: string,
|
|
/**
|
|
* The type of the change
|
|
*/
|
|
changeType: "answer" | "create" | "split" | "delete" | "move" | "import" | string | null
|
|
/**
|
|
* THe motivation for the change, e.g. 'deleted because does not exist anymore'
|
|
*/
|
|
specialMotivation?: string,
|
|
/**
|
|
* Added by Changes.ts
|
|
*/
|
|
distanceToObject?: number
|
|
},
|
|
|
|
/**
|
|
* Identifier of the object
|
|
*/
|
|
type: "node" | "way" | "relation",
|
|
/**
|
|
* Identifier of the object
|
|
* Negative for new objects
|
|
*/
|
|
id: number,
|
|
|
|
/**
|
|
* All changes to tags
|
|
* v = "" or v = undefined to erase this tag
|
|
*
|
|
* Note that this list will only contain the _changes_ to the tags, not the full set of tags
|
|
*/
|
|
tags?: { k: string, v: string }[],
|
|
|
|
/**
|
|
* A change to the geometry:
|
|
* 1) Change of node location
|
|
* 2) Change of way geometry
|
|
* 3) Change of relation members (untested)
|
|
*/
|
|
changes?: {
|
|
lat: number,
|
|
lon: number
|
|
} | {
|
|
/* Coordinates are only used for rendering. They should be LON, LAT
|
|
* */
|
|
coordinates: [number, number][]
|
|
nodes: number[],
|
|
} | {
|
|
members: { type: "node" | "way" | "relation", ref: number, role: string }[]
|
|
}
|
|
|
|
/*
|
|
Set to delete the object
|
|
*/
|
|
doDelete?: boolean
|
|
}
|
|
|
|
export class ChangeDescriptionTools {
|
|
|
|
/**
|
|
* Rewrites all the ids in a changeDescription
|
|
*
|
|
* // should rewrite the id of the changed object
|
|
* const change = <ChangeDescription> {
|
|
* id: -1234,
|
|
* type: "node",
|
|
* meta:{
|
|
* theme:"test",
|
|
* changeType: "answer"
|
|
* },
|
|
* tags:[
|
|
* {
|
|
* k: "key",
|
|
* v: "value"
|
|
* }
|
|
* ]
|
|
* }
|
|
* }
|
|
* const mapping = new Map<string, string>([["node/-1234", "node/42"]])
|
|
* const rewritten = ChangeDescriptionTools.rewriteIds(change, mapping)
|
|
* rewritten.id // => 42
|
|
*
|
|
* // should rewrite ids in nodes of a way
|
|
* const change = <ChangeDescription> {
|
|
* type: "way",
|
|
* id: 789,
|
|
* changes: {
|
|
* nodes: [-1, -2, -3, 68453],
|
|
* coordinates: []
|
|
* },
|
|
* meta:{
|
|
* theme:"test",
|
|
* changeType: "create"
|
|
* }
|
|
* }
|
|
* const mapping = new Map<string, string>([["node/-1", "node/42"],["node/-2", "node/43"],["node/-3", "node/44"]])
|
|
* const rewritten = ChangeDescriptionTools.rewriteIds(change, mapping)
|
|
* rewritten.id // => 789
|
|
* rewritten.changes["nodes"] // => [42,43,44, 68453]
|
|
*
|
|
* // should rewrite ids in relationship members
|
|
* const change = <ChangeDescription> {
|
|
* type: "way",
|
|
* id: 789,
|
|
* changes: {
|
|
* members: [{type: "way", ref: -1, role: "outer"},{type: "way", ref: 48, role: "outer"}],
|
|
* },
|
|
* meta:{
|
|
* theme:"test",
|
|
* changeType: "create"
|
|
* }
|
|
* }
|
|
* const mapping = new Map<string, string>([["way/-1", "way/42"],["node/-2", "node/43"],["node/-3", "node/44"]])
|
|
* const rewritten = ChangeDescriptionTools.rewriteIds(change, mapping)
|
|
* rewritten.id // => 789
|
|
* rewritten.changes["members"] // => [{type: "way", ref: 42, role: "outer"},{type: "way", ref: 48, role: "outer"}]
|
|
*
|
|
*/
|
|
public static rewriteIds(change: ChangeDescription, mappings: Map<string, string>): ChangeDescription {
|
|
const key = change.type + "/" + change.id
|
|
|
|
const wayHasChangedNode = ((change.changes ?? {})["nodes"] ?? []).some(id => mappings.has("node/" + id));
|
|
const relationHasChangedMembers = ((change.changes ?? {})["members"] ?? [])
|
|
.some((obj:{type: string, ref: number}) => mappings.has(obj.type+"/" + obj.ref));
|
|
|
|
const hasSomeChange = mappings.has(key)
|
|
|| wayHasChangedNode || relationHasChangedMembers
|
|
if(hasSomeChange){
|
|
change = {...change}
|
|
}
|
|
|
|
if (mappings.has(key)) {
|
|
const [_, newId] = mappings.get(key).split("/")
|
|
change.id = Number.parseInt(newId)
|
|
}
|
|
if(wayHasChangedNode){
|
|
change.changes = {...change.changes}
|
|
change.changes["nodes"] = change.changes["nodes"].map(id => {
|
|
const key = "node/"+id
|
|
if(!mappings.has(key)){
|
|
return id
|
|
}
|
|
const [_, newId] = mappings.get(key).split("/")
|
|
return Number.parseInt(newId)
|
|
})
|
|
}
|
|
if(relationHasChangedMembers){
|
|
change.changes = {...change.changes}
|
|
change.changes["members"] = change.changes["members"].map(
|
|
(obj:{type: string, ref: number}) => {
|
|
const key = obj.type+"/"+obj.ref;
|
|
if(!mappings.has(key)){
|
|
return obj
|
|
}
|
|
const [_, newId] = mappings.get(key).split("/")
|
|
return {...obj, ref: Number.parseInt(newId)}
|
|
}
|
|
)
|
|
}
|
|
|
|
return change
|
|
}
|
|
|
|
public static getGeojsonGeometry(change: ChangeDescription): any {
|
|
switch (change.type) {
|
|
case "node":
|
|
const n = new OsmNode(change.id)
|
|
n.lat = change.changes["lat"]
|
|
n.lon = change.changes["lon"]
|
|
return n.asGeoJson().geometry
|
|
case "way":
|
|
const w = new OsmWay(change.id)
|
|
w.nodes = change.changes["nodes"]
|
|
w.coordinates = change.changes["coordinates"].map(([lon, lat]) => [lat, lon])
|
|
return w.asGeoJson().geometry
|
|
case "relation":
|
|
const r = new OsmRelation(change.id)
|
|
r.members = change.changes["members"]
|
|
return r.asGeoJson().geometry
|
|
}
|
|
}
|
|
} |