forked from MapComplete/MapComplete
92 lines
No EOL
3.2 KiB
TypeScript
92 lines
No EOL
3.2 KiB
TypeScript
import {ChangeDescription} from "./ChangeDescription";
|
|
import OsmChangeAction from "./OsmChangeAction";
|
|
import {Changes} from "../Changes";
|
|
import {Tag} from "../../Tags/Tag";
|
|
import CreateNewNodeAction from "./CreateNewNodeAction";
|
|
import {And} from "../../Tags/And";
|
|
import {TagsFilter} from "../../Tags/TagsFilter";
|
|
|
|
export default class CreateNewWayAction extends OsmChangeAction {
|
|
public newElementId: string = undefined
|
|
private readonly coordinates: ({ nodeId?: number, lat: number, lon: number })[];
|
|
private readonly tags: Tag[];
|
|
private readonly _options: {
|
|
theme: string, existingPointHandling?: {
|
|
withinRangeOfM: number,
|
|
ifMatches?: TagsFilter,
|
|
mode: "reuse_osm_point" | "move_osm_point"
|
|
} []
|
|
};
|
|
|
|
|
|
/***
|
|
* Creates a new way to upload to OSM
|
|
* @param tags: the tags to apply to the wya
|
|
* @param coordinates: the coordinates. Might have a nodeId, in this case, this node will be used
|
|
* @param options
|
|
*/
|
|
constructor(tags: Tag[], coordinates: ({ nodeId?: number, lat: number, lon: number })[],
|
|
options: {
|
|
theme: string,
|
|
/**
|
|
* IF specified, an existing OSM-point within this range and satisfying the condition 'ifMatches' will be used instead of a new coordinate.
|
|
* If multiple points are possible, only the closest point is considered
|
|
*/
|
|
existingPointHandling?: {
|
|
withinRangeOfM: number,
|
|
ifMatches?: TagsFilter,
|
|
mode: "reuse_osm_point" | "move_osm_point"
|
|
} []
|
|
}) {
|
|
super()
|
|
this.coordinates = coordinates;
|
|
this.tags = tags;
|
|
this._options = options;
|
|
}
|
|
|
|
protected async CreateChangeDescriptions(changes: Changes): Promise<ChangeDescription[]> {
|
|
|
|
const newElements: ChangeDescription[] = []
|
|
|
|
const pointIds: number[] = []
|
|
for (const coordinate of this.coordinates) {
|
|
if (coordinate.nodeId !== undefined) {
|
|
pointIds.push(coordinate.nodeId)
|
|
continue
|
|
}
|
|
|
|
const newPoint = new CreateNewNodeAction([], coordinate.lat, coordinate.lon, {
|
|
allowReuseOfPreviouslyCreatedPoints: true,
|
|
changeType: null,
|
|
theme: this._options.theme
|
|
})
|
|
await changes.applyAction(newPoint)
|
|
pointIds.push(newPoint.newElementIdNumber)
|
|
}
|
|
|
|
// We have all created (or reused) all the points!
|
|
// Time to create the actual way
|
|
|
|
|
|
const id = changes.getNewID()
|
|
|
|
const newWay = <ChangeDescription>{
|
|
id,
|
|
type: "way",
|
|
meta: {
|
|
theme: this._options.theme,
|
|
changeType: "import"
|
|
},
|
|
tags: new And(this.tags).asChange({}),
|
|
changes: {
|
|
nodes: pointIds,
|
|
coordinates: this.coordinates.map(c => [c.lon, c.lat])
|
|
}
|
|
}
|
|
newElements.push(newWay)
|
|
this.newElementId = "way/" + id
|
|
return newElements
|
|
}
|
|
|
|
|
|
} |