2023-03-28 05:13:48 +02:00
|
|
|
import GeoJsonSource from "./GeoJsonSource"
|
|
|
|
import LayerConfig from "../../../Models/ThemeConfig/LayerConfig"
|
2024-11-01 18:51:31 +01:00
|
|
|
import { FeatureSource, UpdatableFeatureSource } from "../FeatureSource"
|
2023-06-14 20:39:36 +02:00
|
|
|
import { Or } from "../../Tags/Or"
|
2023-03-28 05:13:48 +02:00
|
|
|
import FeatureSwitchState from "../../State/FeatureSwitchState"
|
|
|
|
import OverpassFeatureSource from "./OverpassFeatureSource"
|
2024-11-12 13:17:36 +01:00
|
|
|
import { ImmutableStore, Store, UIEventSource } from "../../UIEventSource"
|
2023-03-28 05:13:48 +02:00
|
|
|
import OsmFeatureSource from "./OsmFeatureSource"
|
|
|
|
import DynamicGeoJsonTileSource from "../TiledFeatureSource/DynamicGeoJsonTileSource"
|
2023-06-14 20:39:36 +02:00
|
|
|
import { BBox } from "../../BBox"
|
2023-03-28 05:13:48 +02:00
|
|
|
import LocalStorageFeatureSource from "../TiledFeatureSource/LocalStorageFeatureSource"
|
2023-06-14 20:39:36 +02:00
|
|
|
import FullNodeDatabaseSource from "../TiledFeatureSource/FullNodeDatabaseSource"
|
2024-01-22 01:42:05 +01:00
|
|
|
import DynamicMvtileSource from "../TiledFeatureSource/DynamicMvtTileSource"
|
2024-02-21 16:35:49 +01:00
|
|
|
import FeatureSourceMerger from "./FeatureSourceMerger"
|
2023-03-26 05:58:28 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* This source will fetch the needed data from various sources for the given layout.
|
|
|
|
*
|
|
|
|
* Note that special layers (with `source=null` will be ignored)
|
|
|
|
*/
|
2024-10-17 04:06:03 +02:00
|
|
|
export default class ThemeSource extends FeatureSourceMerger {
|
2023-04-06 02:20:25 +02:00
|
|
|
/**
|
|
|
|
* Indicates if a data source is loading something
|
|
|
|
*/
|
2024-02-18 15:59:28 +01:00
|
|
|
public readonly isLoading: Store<boolean>
|
|
|
|
|
2024-02-21 16:35:49 +01:00
|
|
|
private readonly supportsForceDownload: UpdatableFeatureSource[]
|
|
|
|
|
2024-02-26 15:52:40 +01:00
|
|
|
public static readonly fromCacheZoomLevel = 15
|
2024-08-02 13:32:47 +02:00
|
|
|
|
2024-11-12 13:17:36 +01:00
|
|
|
/**
|
|
|
|
* This source is _only_ triggered when the data is downloaded for CSV export
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
private readonly _downloadAll: OverpassFeatureSource
|
|
|
|
private readonly _mapBounds: Store<BBox>
|
|
|
|
|
2023-03-26 05:58:28 +02:00
|
|
|
constructor(
|
2023-03-28 05:13:48 +02:00
|
|
|
layers: LayerConfig[],
|
2023-03-26 05:58:28 +02:00
|
|
|
featureSwitches: FeatureSwitchState,
|
|
|
|
mapProperties: { bounds: Store<BBox>; zoom: Store<number> },
|
|
|
|
backend: string,
|
2023-06-01 02:52:21 +02:00
|
|
|
isDisplayed: (id: string) => Store<boolean>,
|
2024-02-18 15:59:28 +01:00
|
|
|
mvtAvailableLayers: Set<string>,
|
2024-11-14 02:21:10 +01:00
|
|
|
fullNodeDatabaseSource?: FullNodeDatabaseSource
|
2023-03-26 05:58:28 +02:00
|
|
|
) {
|
2024-02-21 16:35:49 +01:00
|
|
|
const supportsForceDownload: UpdatableFeatureSource[] = []
|
|
|
|
|
2023-03-26 05:58:28 +02:00
|
|
|
const { bounds, zoom } = mapProperties
|
|
|
|
// remove all 'special' layers
|
2023-03-30 04:51:56 +02:00
|
|
|
layers = layers.filter((layer) => layer.source !== null && layer.source !== undefined)
|
2023-03-26 05:58:28 +02:00
|
|
|
|
2023-03-28 05:13:48 +02:00
|
|
|
const geojsonlayers = layers.filter((layer) => layer.source.geojsonSource !== undefined)
|
|
|
|
const osmLayers = layers.filter((layer) => layer.source.geojsonSource === undefined)
|
2024-02-26 15:08:07 +01:00
|
|
|
const fromCache = new Map<string, LocalStorageFeatureSource>()
|
2024-08-02 13:32:47 +02:00
|
|
|
if (featureSwitches.featureSwitchCache.data) {
|
|
|
|
for (const layer of osmLayers) {
|
|
|
|
const src = new LocalStorageFeatureSource(
|
|
|
|
backend,
|
|
|
|
layer,
|
2024-10-17 04:06:03 +02:00
|
|
|
ThemeSource.fromCacheZoomLevel,
|
2024-08-02 13:32:47 +02:00
|
|
|
mapProperties,
|
|
|
|
{
|
|
|
|
isActive: isDisplayed(layer.id),
|
2024-08-14 13:53:56 +02:00
|
|
|
maxAge: layer.maxAgeOfCache,
|
2024-11-14 02:21:10 +01:00
|
|
|
}
|
2024-08-02 13:32:47 +02:00
|
|
|
)
|
|
|
|
fromCache.set(layer.id, src)
|
|
|
|
}
|
|
|
|
}
|
2024-02-21 16:35:49 +01:00
|
|
|
const mvtSources: UpdatableFeatureSource[] = osmLayers
|
2024-02-18 15:59:28 +01:00
|
|
|
.filter((f) => mvtAvailableLayers.has(f.id))
|
2024-10-17 04:06:03 +02:00
|
|
|
.map((l) => ThemeSource.setupMvtSource(l, mapProperties, isDisplayed(l.id)))
|
2024-11-01 18:51:31 +01:00
|
|
|
const nonMvtSources: FeatureSource[] = []
|
|
|
|
const nonMvtLayers: LayerConfig[] = osmLayers.filter((l) => !mvtAvailableLayers.has(l.id))
|
2023-04-20 17:42:07 +02:00
|
|
|
|
2024-02-18 15:59:28 +01:00
|
|
|
const isLoading = new UIEventSource(false)
|
2024-02-24 21:05:46 +01:00
|
|
|
|
2024-10-17 04:06:03 +02:00
|
|
|
const osmApiSource = ThemeSource.setupOsmApiSource(
|
2024-02-24 21:05:46 +01:00
|
|
|
osmLayers,
|
|
|
|
bounds,
|
|
|
|
zoom,
|
|
|
|
backend,
|
|
|
|
featureSwitches,
|
2024-11-14 02:21:10 +01:00
|
|
|
fullNodeDatabaseSource
|
2024-02-24 21:05:46 +01:00
|
|
|
)
|
|
|
|
nonMvtSources.push(osmApiSource)
|
|
|
|
|
|
|
|
let overpassSource: OverpassFeatureSource = undefined
|
2024-02-18 15:59:28 +01:00
|
|
|
if (nonMvtLayers.length > 0) {
|
|
|
|
console.log(
|
|
|
|
"Layers ",
|
|
|
|
nonMvtLayers.map((l) => l.id),
|
2024-11-14 02:21:10 +01:00
|
|
|
" cannot be fetched from the cache server, defaulting to overpass/OSM-api"
|
2024-02-18 15:59:28 +01:00
|
|
|
)
|
2024-10-17 04:06:03 +02:00
|
|
|
overpassSource = ThemeSource.setupOverpass(osmLayers, bounds, zoom, featureSwitches)
|
2024-02-24 21:05:46 +01:00
|
|
|
nonMvtSources.push(overpassSource)
|
2024-02-21 16:35:49 +01:00
|
|
|
supportsForceDownload.push(overpassSource)
|
2024-02-18 15:59:28 +01:00
|
|
|
}
|
2024-01-22 01:42:05 +01:00
|
|
|
|
2024-02-24 21:05:46 +01:00
|
|
|
function setIsLoading() {
|
|
|
|
const loading = overpassSource?.runningQuery?.data || osmApiSource?.isRunning?.data
|
|
|
|
isLoading.setData(loading)
|
|
|
|
}
|
|
|
|
|
2024-04-10 14:09:32 +02:00
|
|
|
overpassSource?.runningQuery?.addCallbackAndRun(() => setIsLoading())
|
|
|
|
osmApiSource?.isRunning?.addCallbackAndRun(() => setIsLoading())
|
2024-02-24 21:05:46 +01:00
|
|
|
|
2024-02-21 16:35:49 +01:00
|
|
|
const geojsonSources: UpdatableFeatureSource[] = geojsonlayers.map((l) =>
|
2024-11-14 02:21:10 +01:00
|
|
|
ThemeSource.setupGeojsonSource(l, mapProperties, isDisplayed(l.id))
|
2023-03-26 05:58:28 +02:00
|
|
|
)
|
|
|
|
|
2024-11-12 13:17:36 +01:00
|
|
|
const downloadAllBounds: UIEventSource<BBox> = new UIEventSource<BBox>(undefined)
|
2024-11-14 02:21:10 +01:00
|
|
|
const downloadAll = new OverpassFeatureSource(
|
|
|
|
{
|
|
|
|
layers: layers.filter((l) => l.isNormal()),
|
|
|
|
bounds: mapProperties.bounds,
|
|
|
|
zoom: mapProperties.zoom,
|
|
|
|
overpassUrl: featureSwitches.overpassUrl,
|
|
|
|
overpassTimeout: featureSwitches.overpassTimeout,
|
|
|
|
overpassMaxZoom: new ImmutableStore(99),
|
|
|
|
widenFactor: 0,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
ignoreZoom: true,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
super(
|
|
|
|
...geojsonSources,
|
|
|
|
...Array.from(fromCache.values()),
|
|
|
|
...mvtSources,
|
|
|
|
...nonMvtSources,
|
|
|
|
downloadAll
|
|
|
|
)
|
2024-01-22 01:42:05 +01:00
|
|
|
|
2024-02-18 15:59:28 +01:00
|
|
|
this.isLoading = isLoading
|
2024-02-21 16:35:49 +01:00
|
|
|
supportsForceDownload.push(...geojsonSources)
|
|
|
|
supportsForceDownload.push(...mvtSources) // Non-mvt sources are handled by overpass
|
2024-11-12 13:17:36 +01:00
|
|
|
|
|
|
|
this._mapBounds = mapProperties.bounds
|
|
|
|
this._downloadAll = downloadAll
|
|
|
|
|
2024-02-21 16:35:49 +01:00
|
|
|
this.supportsForceDownload = supportsForceDownload
|
2023-03-26 05:58:28 +02:00
|
|
|
}
|
|
|
|
|
2024-02-18 15:59:28 +01:00
|
|
|
private static setupMvtSource(
|
|
|
|
layer: LayerConfig,
|
|
|
|
mapProperties: { zoom: Store<number>; bounds: Store<BBox> },
|
2024-11-14 02:21:10 +01:00
|
|
|
isActive?: Store<boolean>
|
2024-02-21 16:35:49 +01:00
|
|
|
): UpdatableFeatureSource {
|
2024-01-22 01:42:05 +01:00
|
|
|
return new DynamicMvtileSource(layer, mapProperties, { isActive })
|
|
|
|
}
|
2024-02-18 15:59:28 +01:00
|
|
|
|
2023-03-26 05:58:28 +02:00
|
|
|
private static setupGeojsonSource(
|
|
|
|
layer: LayerConfig,
|
|
|
|
mapProperties: { zoom: Store<number>; bounds: Store<BBox> },
|
2024-11-14 02:21:10 +01:00
|
|
|
isActiveByFilter?: Store<boolean>
|
2024-02-21 16:35:49 +01:00
|
|
|
): UpdatableFeatureSource {
|
2023-03-26 05:58:28 +02:00
|
|
|
const source = layer.source
|
2024-04-10 14:09:32 +02:00
|
|
|
const isActive = mapProperties.zoom.map(
|
|
|
|
(z) => (isActiveByFilter?.data ?? true) && z >= layer.minzoom,
|
2024-11-14 02:21:10 +01:00
|
|
|
[isActiveByFilter]
|
2023-03-28 05:13:48 +02:00
|
|
|
)
|
2023-03-26 05:58:28 +02:00
|
|
|
if (source.geojsonZoomLevel === undefined) {
|
|
|
|
// This is a 'load everything at once' geojson layer
|
|
|
|
return new GeoJsonSource(layer, { isActive })
|
|
|
|
} else {
|
|
|
|
return new DynamicGeoJsonTileSource(layer, mapProperties, { isActive })
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static setupOsmApiSource(
|
|
|
|
osmLayers: LayerConfig[],
|
|
|
|
bounds: Store<BBox>,
|
|
|
|
zoom: Store<number>,
|
|
|
|
backend: string,
|
2023-06-01 02:52:21 +02:00
|
|
|
featureSwitches: FeatureSwitchState,
|
2024-11-14 02:21:10 +01:00
|
|
|
fullNodeDatabase: FullNodeDatabaseSource
|
2023-04-20 01:52:23 +02:00
|
|
|
): OsmFeatureSource | undefined {
|
2023-04-06 01:33:08 +02:00
|
|
|
if (osmLayers.length == 0) {
|
2023-04-20 01:52:23 +02:00
|
|
|
return undefined
|
2023-04-06 01:33:08 +02:00
|
|
|
}
|
2023-03-26 05:58:28 +02:00
|
|
|
const minzoom = Math.min(...osmLayers.map((layer) => layer.minzoom))
|
|
|
|
const isActive = zoom.mapD((z) => {
|
|
|
|
if (z < minzoom) {
|
|
|
|
// We are zoomed out over the zoomlevel of any layer
|
|
|
|
console.debug("Disabling overpass source: zoom < minzoom")
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
// Overpass should handle this if zoomed out a bit
|
|
|
|
return z > featureSwitches.overpassMaxZoom.data
|
|
|
|
})
|
|
|
|
const allowedFeatures = new Or(osmLayers.map((l) => l.source.osmTags)).optimize()
|
|
|
|
if (typeof allowedFeatures === "boolean") {
|
|
|
|
throw "Invalid filter to init OsmFeatureSource: it optimizes away to " + allowedFeatures
|
|
|
|
}
|
|
|
|
return new OsmFeatureSource({
|
|
|
|
allowedFeatures,
|
|
|
|
bounds,
|
|
|
|
backend,
|
|
|
|
isActive,
|
2023-06-01 02:52:21 +02:00
|
|
|
patchRelations: true,
|
2024-08-14 13:53:56 +02:00
|
|
|
fullNodeDatabase,
|
2023-03-26 05:58:28 +02:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
private static setupOverpass(
|
|
|
|
osmLayers: LayerConfig[],
|
|
|
|
bounds: Store<BBox>,
|
|
|
|
zoom: Store<number>,
|
2024-11-14 02:21:10 +01:00
|
|
|
featureSwitches: FeatureSwitchState
|
2023-04-20 01:52:23 +02:00
|
|
|
): OverpassFeatureSource | undefined {
|
2023-04-06 01:33:08 +02:00
|
|
|
if (osmLayers.length == 0) {
|
2023-04-20 01:52:23 +02:00
|
|
|
return undefined
|
2023-04-06 01:33:08 +02:00
|
|
|
}
|
2023-03-26 05:58:28 +02:00
|
|
|
const minzoom = Math.min(...osmLayers.map((layer) => layer.minzoom))
|
|
|
|
const isActive = zoom.mapD((z) => {
|
|
|
|
if (z < minzoom) {
|
|
|
|
// We are zoomed out over the zoomlevel of any layer
|
|
|
|
console.debug("Disabling overpass source: zoom < minzoom")
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
return z <= featureSwitches.overpassMaxZoom.data
|
|
|
|
})
|
|
|
|
|
|
|
|
return new OverpassFeatureSource(
|
|
|
|
{
|
|
|
|
zoom,
|
|
|
|
bounds,
|
2023-03-30 04:51:56 +02:00
|
|
|
layers: osmLayers,
|
2024-10-17 04:06:03 +02:00
|
|
|
widenFactor: 1.5,
|
2023-03-26 05:58:28 +02:00
|
|
|
overpassUrl: featureSwitches.overpassUrl,
|
|
|
|
overpassTimeout: featureSwitches.overpassTimeout,
|
2024-08-14 13:53:56 +02:00
|
|
|
overpassMaxZoom: featureSwitches.overpassMaxZoom,
|
2023-03-26 05:58:28 +02:00
|
|
|
},
|
|
|
|
{
|
|
|
|
padToTiles: zoom.map((zoom) => Math.min(15, zoom + 1)),
|
2024-08-14 13:53:56 +02:00
|
|
|
isActive,
|
2024-11-14 02:21:10 +01:00
|
|
|
}
|
2023-03-26 05:58:28 +02:00
|
|
|
)
|
|
|
|
}
|
2024-02-21 16:35:49 +01:00
|
|
|
|
|
|
|
public async downloadAll() {
|
2024-11-12 13:17:36 +01:00
|
|
|
console.log("Downloading all data:")
|
|
|
|
await this._downloadAll.updateAsync(this._mapBounds.data)
|
2024-11-14 02:21:10 +01:00
|
|
|
// await Promise.all(this.supportsForceDownload.map((i) => i.updateAsync()))
|
2024-02-21 16:35:49 +01:00
|
|
|
console.log("Done")
|
|
|
|
}
|
2023-03-26 05:58:28 +02:00
|
|
|
}
|