MapComplete/Logic/FeatureSource/TiledFeatureSource/FullNodeDatabaseSource.ts

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

96 lines
3.6 KiB
TypeScript
Raw Normal View History

import { OsmNode, OsmObject, OsmWay } from "../../Osm/OsmObject"
import { UIEventSource } from "../../UIEventSource"
import { BBox } from "../../BBox"
import StaticFeatureSource from "../Sources/StaticFeatureSource"
import { Tiles } from "../../../Models/TileRange"
2023-03-28 05:13:48 +02:00
export default class FullNodeDatabaseSource {
2023-06-01 02:52:21 +02:00
private readonly loadedTiles = new Map<number, Map<number, OsmNode>>()
2021-12-23 03:36:03 +01:00
private readonly nodeByIds = new Map<number, OsmNode>()
2021-12-30 20:41:45 +01:00
private readonly parentWays = new Map<number, UIEventSource<OsmWay[]>>()
2023-06-01 02:52:21 +02:00
private smallestZoom = 99
private largestZoom = 0
2021-11-07 16:34:51 +01:00
public handleOsmJson(osmJson: any, z: number, x: number, y: number): void {
const allObjects = OsmObject.ParseObjects(osmJson.elements)
const nodesById = new Map<number, OsmNode>()
2023-06-01 02:52:21 +02:00
this.smallestZoom = Math.min(this.smallestZoom, z)
this.largestZoom = Math.max(this.largestZoom, z)
for (const osmObj of allObjects) {
if (osmObj.type !== "node") {
continue
}
const osmNode = <OsmNode>osmObj
nodesById.set(osmNode.id, osmNode)
2021-12-23 03:36:03 +01:00
this.nodeByIds.set(osmNode.id, osmNode)
}
for (const osmObj of allObjects) {
if (osmObj.type !== "way") {
continue
}
const osmWay = <OsmWay>osmObj
for (const nodeId of osmWay.nodes) {
2021-12-30 20:41:45 +01:00
if (!this.parentWays.has(nodeId)) {
2022-01-26 21:40:38 +01:00
const src = new UIEventSource<OsmWay[]>([])
this.parentWays.set(nodeId, src)
2021-12-30 20:41:45 +01:00
src.addCallback((parentWays) => {
const tgs = nodesById.get(nodeId).tags
tgs["parent_ways"] = JSON.stringify(parentWays.map((w) => w.tags))
tgs["parent_way_ids"] = JSON.stringify(parentWays.map((w) => w.id))
})
}
2021-12-30 20:41:45 +01:00
const src = this.parentWays.get(nodeId)
src.data.push(osmWay)
src.ping()
}
}
const asGeojsonFeatures = Array.from(nodesById.values()).map((osmNode) =>
osmNode.asGeoJson()
)
2023-06-01 02:52:21 +02:00
const featureSource = new StaticFeatureSource(asGeojsonFeatures)
const tileId = Tiles.tile_index(z, x, y)
this.loadedTiles.set(tileId, nodesById)
}
2021-12-23 03:36:03 +01:00
/**
* Returns the OsmNode with the corresponding id (undefined if not found)
* Note that this OsmNode will have a calculated tag 'parent_ways' and 'parent_way_ids', which are resp. stringified lists of parent way tags and ids
* @param id
* @constructor
*/
2021-12-30 20:41:45 +01:00
public GetNode(id: number): OsmNode {
2021-12-23 03:36:03 +01:00
return this.nodeByIds.get(id)
}
2021-12-30 20:41:45 +01:00
/**
2023-06-01 02:52:21 +02:00
* Gets all the ways that the given node is a part of
2021-12-30 20:41:45 +01:00
* @param nodeId
* @constructor
*/
public GetParentWays(nodeId: number): UIEventSource<OsmWay[]> {
return this.parentWays.get(nodeId)
}
2023-03-28 05:13:48 +02:00
2023-06-01 02:52:21 +02:00
/**
* Gets (at least) all nodes which are part of this BBOX; might also return some nodes that fall outside of the bbox but are closeby
* @param bbox
*/
getNodesWithin(bbox: BBox): Map<number, OsmNode> {
2023-06-01 02:52:21 +02:00
const allById = new Map<number, OsmNode>()
for (let z = this.smallestZoom; z < this.largestZoom; z++) {
const range = Tiles.tileRangeFrom(bbox, z)
Tiles.MapRange(range, (x, y) => {
2023-06-01 02:52:21 +02:00
const tileId = Tiles.tile_index(z, x, y)
const nodesById = this.loadedTiles.get(tileId)
nodesById?.forEach((v, k) => allById.set(k, v))
2023-06-01 02:52:21 +02:00
})
}
return allById
2023-03-28 05:13:48 +02:00
}
}