forked from MapComplete/MapComplete
248 lines
9 KiB
TypeScript
248 lines
9 KiB
TypeScript
import FeatureSource, { FeatureSourceForLayer, IndexedFeatureSource, Tiled } from "../FeatureSource"
|
|
import { Store, UIEventSource } from "../../UIEventSource"
|
|
import FilteredLayer from "../../../Models/FilteredLayer"
|
|
import TileHierarchy from "./TileHierarchy"
|
|
import { Tiles } from "../../../Models/TileRange"
|
|
import { BBox } from "../../BBox"
|
|
|
|
/**
|
|
* Contains all features in a tiled fashion.
|
|
* The data will be automatically broken down into subtiles when there are too much features in a single tile or if the zoomlevel is too high
|
|
*/
|
|
export default class TiledFeatureSource
|
|
implements
|
|
Tiled,
|
|
IndexedFeatureSource,
|
|
FeatureSourceForLayer,
|
|
TileHierarchy<IndexedFeatureSource & FeatureSourceForLayer & Tiled>
|
|
{
|
|
public readonly z: number
|
|
public readonly x: number
|
|
public readonly y: number
|
|
public readonly parent: TiledFeatureSource
|
|
public readonly root: TiledFeatureSource
|
|
public readonly layer: FilteredLayer
|
|
/* An index of all known tiles. allTiles[z][x][y].get('layerid') will yield the corresponding tile.
|
|
* Only defined on the root element!
|
|
*/
|
|
public readonly loadedTiles: Map<number, TiledFeatureSource & FeatureSourceForLayer> = undefined
|
|
|
|
public readonly maxFeatureCount: number
|
|
public readonly name
|
|
public readonly features: UIEventSource<{ feature: any; freshness: Date }[]>
|
|
public readonly containedIds: Store<Set<string>>
|
|
|
|
public readonly bbox: BBox
|
|
public readonly tileIndex: number
|
|
private upper_left: TiledFeatureSource
|
|
private upper_right: TiledFeatureSource
|
|
private lower_left: TiledFeatureSource
|
|
private lower_right: TiledFeatureSource
|
|
private readonly maxzoom: number
|
|
private readonly options: TiledFeatureSourceOptions
|
|
|
|
private constructor(
|
|
z: number,
|
|
x: number,
|
|
y: number,
|
|
parent: TiledFeatureSource,
|
|
options?: TiledFeatureSourceOptions
|
|
) {
|
|
this.z = z
|
|
this.x = x
|
|
this.y = y
|
|
this.bbox = BBox.fromTile(z, x, y)
|
|
this.tileIndex = Tiles.tile_index(z, x, y)
|
|
this.name = `TiledFeatureSource(${z},${x},${y})`
|
|
this.parent = parent
|
|
this.layer = options.layer
|
|
options = options ?? {}
|
|
this.maxFeatureCount = options?.maxFeatureCount ?? 250
|
|
this.maxzoom = options.maxZoomLevel ?? 18
|
|
this.options = options
|
|
if (parent === undefined) {
|
|
throw "Parent is not allowed to be undefined. Use null instead"
|
|
}
|
|
if (parent === null && z !== 0 && x !== 0 && y !== 0) {
|
|
throw "Invalid root tile: z, x and y should all be null"
|
|
}
|
|
if (parent === null) {
|
|
this.root = this
|
|
this.loadedTiles = new Map()
|
|
} else {
|
|
this.root = this.parent.root
|
|
this.loadedTiles = this.root.loadedTiles
|
|
const i = Tiles.tile_index(z, x, y)
|
|
this.root.loadedTiles.set(i, this)
|
|
}
|
|
this.features = new UIEventSource<any[]>([])
|
|
this.containedIds = this.features.map((features) => {
|
|
if (features === undefined) {
|
|
return undefined
|
|
}
|
|
return new Set(features.map((f) => f.feature.properties.id))
|
|
})
|
|
|
|
// We register this tile, but only when there is some data in it
|
|
if (this.options.registerTile !== undefined) {
|
|
this.features.addCallbackAndRunD((features) => {
|
|
if (features.length === 0) {
|
|
return
|
|
}
|
|
this.options.registerTile(this)
|
|
return true
|
|
})
|
|
}
|
|
}
|
|
|
|
public static createHierarchy(
|
|
features: FeatureSource,
|
|
options?: TiledFeatureSourceOptions
|
|
): TiledFeatureSource {
|
|
options = {
|
|
...options,
|
|
layer: features["layer"] ?? options.layer,
|
|
}
|
|
const root = new TiledFeatureSource(0, 0, 0, null, options)
|
|
features.features?.addCallbackAndRunD((feats) => root.addFeatures(feats))
|
|
return root
|
|
}
|
|
|
|
private isSplitNeeded(featureCount: number) {
|
|
if (this.upper_left !== undefined) {
|
|
// This tile has been split previously, so we keep on splitting
|
|
return true
|
|
}
|
|
if (this.z >= this.maxzoom) {
|
|
// We are not allowed to split any further
|
|
return false
|
|
}
|
|
if (this.options.minZoomLevel !== undefined && this.z < this.options.minZoomLevel) {
|
|
// We must have at least this zoom level before we are allowed to start splitting
|
|
return true
|
|
}
|
|
|
|
// To much features - we split
|
|
return featureCount > this.maxFeatureCount
|
|
}
|
|
|
|
/***
|
|
* Adds the list of features to this hierarchy.
|
|
* If there are too much features, the list will be broken down and distributed over the subtiles (only retaining features that don't fit a subtile on this level)
|
|
* @param features
|
|
* @private
|
|
*/
|
|
private addFeatures(features: { feature: any; freshness: Date }[]) {
|
|
if (features === undefined || features.length === 0) {
|
|
return
|
|
}
|
|
|
|
if (!this.isSplitNeeded(features.length)) {
|
|
this.features.setData(features)
|
|
return
|
|
}
|
|
|
|
if (this.upper_left === undefined) {
|
|
this.upper_left = new TiledFeatureSource(
|
|
this.z + 1,
|
|
this.x * 2,
|
|
this.y * 2,
|
|
this,
|
|
this.options
|
|
)
|
|
this.upper_right = new TiledFeatureSource(
|
|
this.z + 1,
|
|
this.x * 2 + 1,
|
|
this.y * 2,
|
|
this,
|
|
this.options
|
|
)
|
|
this.lower_left = new TiledFeatureSource(
|
|
this.z + 1,
|
|
this.x * 2,
|
|
this.y * 2 + 1,
|
|
this,
|
|
this.options
|
|
)
|
|
this.lower_right = new TiledFeatureSource(
|
|
this.z + 1,
|
|
this.x * 2 + 1,
|
|
this.y * 2 + 1,
|
|
this,
|
|
this.options
|
|
)
|
|
}
|
|
|
|
const ulf = []
|
|
const urf = []
|
|
const llf = []
|
|
const lrf = []
|
|
const overlapsboundary = []
|
|
|
|
for (const feature of features) {
|
|
const bbox = BBox.get(feature.feature)
|
|
|
|
// There are a few strategies to deal with features that cross tile boundaries
|
|
|
|
if (this.options.noDuplicates) {
|
|
// Strategy 1: We put the feature into a somewhat matching tile
|
|
if (bbox.overlapsWith(this.upper_left.bbox)) {
|
|
ulf.push(feature)
|
|
} else if (bbox.overlapsWith(this.upper_right.bbox)) {
|
|
urf.push(feature)
|
|
} else if (bbox.overlapsWith(this.lower_left.bbox)) {
|
|
llf.push(feature)
|
|
} else if (bbox.overlapsWith(this.lower_right.bbox)) {
|
|
lrf.push(feature)
|
|
} else {
|
|
overlapsboundary.push(feature)
|
|
}
|
|
} else if (this.options.minZoomLevel === undefined) {
|
|
// Strategy 2: put it into a strictly matching tile (or in this tile, which is slightly too big)
|
|
if (bbox.isContainedIn(this.upper_left.bbox)) {
|
|
ulf.push(feature)
|
|
} else if (bbox.isContainedIn(this.upper_right.bbox)) {
|
|
urf.push(feature)
|
|
} else if (bbox.isContainedIn(this.lower_left.bbox)) {
|
|
llf.push(feature)
|
|
} else if (bbox.isContainedIn(this.lower_right.bbox)) {
|
|
lrf.push(feature)
|
|
} else {
|
|
overlapsboundary.push(feature)
|
|
}
|
|
} else {
|
|
// Strategy 3: We duplicate a feature on a boundary into every tile as we need to get to the minZoomLevel
|
|
if (bbox.overlapsWith(this.upper_left.bbox)) {
|
|
ulf.push(feature)
|
|
}
|
|
if (bbox.overlapsWith(this.upper_right.bbox)) {
|
|
urf.push(feature)
|
|
}
|
|
if (bbox.overlapsWith(this.lower_left.bbox)) {
|
|
llf.push(feature)
|
|
}
|
|
if (bbox.overlapsWith(this.lower_right.bbox)) {
|
|
lrf.push(feature)
|
|
}
|
|
}
|
|
}
|
|
this.upper_left.addFeatures(ulf)
|
|
this.upper_right.addFeatures(urf)
|
|
this.lower_left.addFeatures(llf)
|
|
this.lower_right.addFeatures(lrf)
|
|
this.features.setData(overlapsboundary)
|
|
}
|
|
}
|
|
|
|
export interface TiledFeatureSourceOptions {
|
|
readonly maxFeatureCount?: number
|
|
readonly maxZoomLevel?: number
|
|
readonly minZoomLevel?: number
|
|
/**
|
|
* IF minZoomLevel is set, and if a feature runs through a tile boundary, it would normally be duplicated.
|
|
* Setting 'dontEnforceMinZoomLevel' will assign to feature to some matching subtile.
|
|
*/
|
|
readonly noDuplicates?: boolean
|
|
readonly registerTile?: (tile: TiledFeatureSource & FeatureSourceForLayer & Tiled) => void
|
|
readonly layer?: FilteredLayer
|
|
}
|