mapcomplete/Logic/FeatureSource/TiledFeatureSource/DynamicTileSource.ts

89 lines
3.4 KiB
TypeScript
Raw Normal View History

2022-09-08 21:40:48 +02:00
import FilteredLayer from "../../../Models/FilteredLayer"
import { FeatureSourceForLayer, Tiled } from "../FeatureSource"
import { UIEventSource } from "../../UIEventSource"
import TileHierarchy from "./TileHierarchy"
import { Tiles } from "../../../Models/TileRange"
import { BBox } from "../../BBox"
/***
* A tiled source which dynamically loads the required tiles at a fixed zoom level
*/
export default class DynamicTileSource implements TileHierarchy<FeatureSourceForLayer & Tiled> {
2022-09-08 21:40:48 +02:00
public readonly loadedTiles: Map<number, FeatureSourceForLayer & Tiled>
private readonly _loadedTiles = new Set<number>()
constructor(
layer: FilteredLayer,
zoomlevel: number,
2022-09-08 21:40:48 +02:00
constructTile: (zxy: [number, number, number]) => FeatureSourceForLayer & Tiled,
state: {
2022-09-08 21:40:48 +02:00
currentBounds: UIEventSource<BBox>
locationControl?: UIEventSource<{ zoom?: number }>
}
) {
2022-09-08 21:40:48 +02:00
const self = this
2021-11-07 16:34:51 +01:00
this.loadedTiles = new Map<number, FeatureSourceForLayer & Tiled>()
2022-09-08 21:40:48 +02:00
const neededTiles = state.currentBounds
.map(
(bounds) => {
if (bounds === undefined) {
// We'll retry later
return undefined
}
2022-09-08 21:40:48 +02:00
if (!layer.isDisplayed.data && !layer.layerDef.forceLoad) {
// No need to download! - the layer is disabled
return undefined
}
2022-09-08 21:40:48 +02:00
if (
state.locationControl?.data?.zoom !== undefined &&
state.locationControl.data.zoom < layer.layerDef.minzoom
) {
// No need to download! - the layer is disabled
return undefined
}
2022-01-26 21:40:38 +01:00
2022-09-08 21:40:48 +02:00
const tileRange = Tiles.TileRangeBetween(
zoomlevel,
bounds.getNorth(),
bounds.getEast(),
bounds.getSouth(),
bounds.getWest()
)
if (tileRange.total > 10000) {
console.error(
"Got a really big tilerange, bounds and location might be out of sync"
)
return undefined
}
2022-09-08 21:40:48 +02:00
const needed = Tiles.MapRange(tileRange, (x, y) =>
Tiles.tile_index(zoomlevel, x, y)
).filter((i) => !self._loadedTiles.has(i))
if (needed.length === 0) {
return undefined
}
return needed
},
[layer.isDisplayed, state.locationControl]
)
.stabilized(250)
neededTiles.addCallbackAndRunD((neededIndexes) => {
2021-11-07 16:34:51 +01:00
console.log("Tiled geojson source ", layer.layerDef.id, " needs", neededIndexes)
if (neededIndexes === undefined) {
2022-09-08 21:40:48 +02:00
return
}
for (const neededIndex of neededIndexes) {
self._loadedTiles.add(neededIndex)
const src = constructTile(Tiles.tile_from_index(neededIndex))
2021-11-07 16:34:51 +01:00
if (src !== undefined) {
self.loadedTiles.set(neededIndex, src)
}
}
})
}
}