MapComplete/Logic/Osm/OsmObject.ts

220 lines
6 KiB
TypeScript
Raw Normal View History

2020-06-24 00:35:19 +02:00
import * as $ from "jquery"
2020-09-30 22:22:58 +02:00
import {Utils} from "../../Utils";
2020-06-24 00:35:19 +02:00
export abstract class OsmObject {
type: string;
id: number;
tags: {} = {};
version: number;
public changed: boolean = false;
protected constructor(type: string, id: number) {
this.id = id;
this.type = type;
}
2021-04-21 01:23:28 +02:00
static DownloadObject(id, continuation: ((element: OsmObject, meta: OsmObjectMeta) => void)) {
2020-06-24 00:35:19 +02:00
const splitted = id.split("/");
const type = splitted[0];
const idN = splitted[1];
2021-04-21 01:23:28 +02:00
const newContinuation = (element: OsmObject, meta :OsmObjectMeta) => {
console.log("Received: ", element, "with meta", meta);
continuation(element, meta);
}
2021-04-21 01:23:28 +02:00
2020-06-24 00:35:19 +02:00
switch (type) {
case("node"):
return new OsmNode(idN).Download(newContinuation);
2020-06-24 00:35:19 +02:00
case("way"):
return new OsmWay(idN).Download(newContinuation);
2020-06-24 00:35:19 +02:00
case("relation"):
return new OsmRelation(idN).Download(newContinuation);
2020-06-24 00:35:19 +02:00
}
}
2021-04-21 01:23:28 +02:00
public static DownloadAll(neededIds, knownElements: any = {}, continuation: ((knownObjects: any) => void)) {
// local function which downloads all the objects one by one
// this is one big loop, running one download, then rerunning the entire function
if (neededIds.length == 0) {
continuation(knownElements);
return;
}
const neededId = neededIds.pop();
2020-06-24 00:35:19 +02:00
2021-04-21 01:23:28 +02:00
if (neededId in knownElements) {
OsmObject.DownloadAll(neededIds, knownElements, continuation);
return;
}
2020-06-24 00:35:19 +02:00
2021-04-21 01:23:28 +02:00
console.log("Downloading ", neededId);
OsmObject.DownloadObject(neededId,
function (element) {
knownElements[neededId] = element; // assign the element for later, continue downloading the next element
OsmObject.DownloadAll(neededIds, knownElements, continuation);
}
);
}
abstract SaveExtraData(element);
2020-07-05 18:59:47 +02:00
2020-06-24 00:35:19 +02:00
/**
* Generates the changeset-XML for tags
* @constructor
*/
TagsXML(): string {
let tags = "";
for (const key in this.tags) {
const v = this.tags[key];
if (v !== "") {
2020-09-30 22:22:58 +02:00
tags += ' <tag k="' + Utils.EncodeXmlValue(key) + '" v="' + Utils.EncodeXmlValue(this.tags[key]) + '"/>\n'
2020-06-24 00:35:19 +02:00
}
}
return tags;
}
2021-04-21 01:23:28 +02:00
Download(continuation: ((element: OsmObject, meta: OsmObjectMeta) => void)) {
2020-06-24 00:35:19 +02:00
const self = this;
$.getJSON("https://www.openstreetmap.org/api/0.6/" + this.type + "/" + this.id,
function (data) {
const element = data.elements[0];
self.tags = element.tags;
self.version = element.version;
self.SaveExtraData(element);
2021-04-21 01:23:28 +02:00
continuation(self, {
"_last_edit:contributor": element.user,
"_last_edit:contributor:uid": element.uid,
"_last_edit:changeset": element.changeset,
"_last_edit:timestamp": new Date(element.timestamp),
"_version_number": element.version
});
2020-06-24 00:35:19 +02:00
}
);
return this;
}
public addTag(k: string, v: string): void {
if (k in this.tags) {
const oldV = this.tags[k];
if (oldV == v) {
return;
}
2021-04-21 01:23:28 +02:00
console.log("WARNING: overwriting ", oldV, " with ", v, " for key ", k)
2020-06-24 00:35:19 +02:00
}
this.tags[k] = v;
2021-04-21 01:23:28 +02:00
if (v === undefined || v === "") {
delete this.tags[k];
}
2020-06-24 00:35:19 +02:00
this.changed = true;
}
abstract ChangesetXML(changesetId: string): string;
2021-04-21 01:23:28 +02:00
protected VersionXML() {
if (this.version === undefined) {
return "";
}
2021-04-21 01:23:28 +02:00
return 'version="' + this.version + '"';
}
2020-06-24 00:35:19 +02:00
}
export class OsmNode extends OsmObject {
lat: number;
lon: number;
constructor(id) {
super("node", id);
}
ChangesetXML(changesetId: string): string {
let tags = this.TagsXML();
let change =
' <node id="' + this.id + '" changeset="' + changesetId + '" ' + this.VersionXML() + ' lat="' + this.lat + '" lon="' + this.lon + '">\n' +
tags +
' </node>\n';
return change;
}
SaveExtraData(element) {
this.lat = element.lat;
this.lon = element.lon;
}
}
2021-04-21 01:23:28 +02:00
export interface OsmObjectMeta{
"_last_edit:contributor": string,
"_last_edit:contributor:uid": number,
"_last_edit:changeset": number,
"_last_edit:timestamp": Date,
"_version_number": number
}
2020-06-24 00:35:19 +02:00
export class OsmWay extends OsmObject {
nodes: number[];
constructor(id) {
super("way", id);
}
ChangesetXML(changesetId: string): string {
let tags = this.TagsXML();
let nds = "";
for (const node in this.nodes) {
nds += ' <nd ref="' + this.nodes[node] + '"/>\n';
}
let change =
' <way id="' + this.id + '" changeset="' + changesetId + '" ' + this.VersionXML() + '>\n' +
nds +
tags +
' </way>\n';
return change;
}
SaveExtraData(element) {
this.nodes = element.nodes;
}
}
export class OsmRelation extends OsmObject {
members;
constructor(id) {
super("relation", id);
}
ChangesetXML(changesetId: string): string {
let members = "";
for (const memberI in this.members) {
const member = this.members[memberI];
members += ' <member type="' + member.type + '" ref="' + member.ref + '" role="' + member.role + '"/>\n';
}
let tags = this.TagsXML();
let change =
' <relation id="' + this.id + '" changeset="' + changesetId + '" ' + this.VersionXML() + '>\n' +
members +
tags +
' </relation>\n';
return change;
}
SaveExtraData(element) {
this.members = element.members;
}
}