import DynamicTileSource from "./DynamicTileSource" import { Store, UIEventSource } from "../../UIEventSource" import { BBox } from "../../BBox" import StaticFeatureSource from "../Sources/StaticFeatureSource" import { Feature, Point } from "geojson" import { Utils } from "../../../Utils" import { Tiles } from "../../../Models/TileRange" import { FeatureSource } from "../FeatureSource" import FilteredLayer from "../../../Models/FilteredLayer" import Constants from "../../../Models/Constants" export class SummaryTileSourceRewriter implements FeatureSource { private readonly _features: UIEventSource = new UIEventSource([]) private filteredLayers: FilteredLayer[] public readonly features: Store = this._features private readonly _summarySource: SummaryTileSource private readonly _totalNumberOfFeatures: UIEventSource = new UIEventSource( undefined ) public readonly totalNumberOfFeatures: Store = this._totalNumberOfFeatures constructor( summarySource: SummaryTileSource, filteredLayers: ReadonlyMap ) { this.filteredLayers = Array.from(filteredLayers.values()).filter( (l) => Constants.priviliged_layers.indexOf(l.layerDef.id) < 0 && !l.layerDef.id.startsWith("note_import") ) this._summarySource = summarySource filteredLayers.forEach((v, k) => { v.isDisplayed.addCallback((_) => this.update()) }) this._summarySource.features.addCallbackAndRunD((_) => this.update()) } private update() { let fullTotal = 0 const newFeatures: Feature[] = [] const layersToCount = this.filteredLayers.filter((fl) => fl.isDisplayed.data) const bitmap = layersToCount.map((l) => (l.isDisplayed.data ? "1" : "0")).join("") const ids = layersToCount.map((l) => l.layerDef.id) for (const f of this._summarySource.features.data ?? []) { let newTotal = 0 for (const id of ids) { newTotal += Number(f.properties[id] ?? 0) } newFeatures.push({ ...f, properties: { ...f.properties, id: f.properties.id + bitmap, total: newTotal, total_metric: Utils.numberWithMetricPrefix(newTotal), }, }) fullTotal += newTotal } this._features.setData(newFeatures) this._totalNumberOfFeatures.setData(fullTotal) } } /** * Provides features summarizing the total amount of features at a given location */ export class SummaryTileSource extends DynamicTileSource { private static readonly empty = [] constructor( cacheserver: string, layers: string[], zoomRounded: Store, mapProperties: { bounds: Store zoom: Store }, options?: { isActive?: Store } ) { const layersSummed = layers.join("+") const zDiff = 2 super( zoomRounded, 0, // minzoom (tileIndex) => { const features = SummaryTileSource.downloadTile( tileIndex, cacheserver, layersSummed ) const [z] = Tiles.tile_from_index(tileIndex) return new StaticFeatureSource( features.map( (f) => { if (z - zDiff !== zoomRounded.data) { return SummaryTileSource.empty } return f }, [zoomRounded] ) ) }, mapProperties, { ...options, zDiff } ) } public static downloadTile( tileIndex: number, cacheserver: string, layersSummed: string ): Store[]> { const [z, x, y] = Tiles.tile_from_index(tileIndex) let coordinates = Tiles.centerPointOf(z, x, y) const url = `${cacheserver}/${layersSummed}/${z}/${x}/${y}.json` const count = UIEventSource.FromPromiseWithErr(Utils.downloadJson(url)) return count.mapD((count) => { if (count["error"] !== undefined) { console.error( "Could not download count for tile", z, x, y, "due to", count["error"] ) return SummaryTileSource.empty } const counts = count["success"] const total = Number(counts?.["total"] ?? 0) if (total === 0) { return SummaryTileSource.empty } const lat = counts["lat"] const lon = counts["lon"] const tileBbox = new BBox(Tiles.tile_bounds_lon_lat(z, x, y)) if (!tileBbox.contains([lon, lat])) { console.error( "Average coordinate is outside of bbox!?", lon, lat, tileBbox, counts, url ) } else { coordinates = [lon, lat] } return [ { type: "Feature", properties: { id: "summary_" + tileIndex, summary: "yes", ...counts, total, total_metric: Utils.numberWithMetricPrefix(total), layers: layersSummed, }, geometry: { type: "Point", coordinates, }, }, ] }) } }