mapcomplete/Logic/FeatureSource/Sources/FeatureSourceMerger.ts

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

83 lines
2.8 KiB
TypeScript
Raw Normal View History

import { Store, UIEventSource } from "../../UIEventSource"
import FeatureSource, { IndexedFeatureSource } from "../FeatureSource"
import { Feature } from "geojson"
/**
*
*/
export default class FeatureSourceMerger implements IndexedFeatureSource {
public features: UIEventSource<Feature[]> = new UIEventSource([])
public readonly featuresById: Store<Map<string, Feature>>
private readonly _featuresById: UIEventSource<Map<string, Feature>>
private readonly _sources: FeatureSource[] = []
/**
* Merges features from different featureSources.
* In case that multiple features have the same id, the latest `_version_number` will be used. Otherwise, we will take the last one
*/
constructor(...sources: FeatureSource[]) {
this._featuresById = new UIEventSource<Map<string, Feature>>(undefined)
this.featuresById = this._featuresById
const self = this
for (let source of sources) {
source.features.addCallback(() => {
self.addData(sources.map((s) => s.features.data))
})
}
this.addData(sources.map((s) => s.features.data))
this._sources = sources
}
protected addSource(source: FeatureSource) {
this._sources.push(source)
source.features.addCallbackAndRun(() => {
this.addData(this._sources.map((s) => s.features.data))
})
}
protected addData(featuress: Feature[][]) {
2021-05-07 01:43:32 +02:00
let somethingChanged = false
const all: Map<string, Feature> = new Map()
2021-05-07 01:43:32 +02:00
// We seed the dictionary with the previously loaded features
const oldValues = this.features.data ?? []
for (const oldValue of oldValues) {
all.set(oldValue.properties.id, oldValue)
2021-05-07 01:43:32 +02:00
}
for (const features of featuress) {
if (features === undefined) {
2021-01-04 22:59:11 +01:00
continue
}
for (const f of features) {
const id = f.properties.id
2021-05-07 01:43:32 +02:00
if (!all.has(id)) {
// This is a new feature
somethingChanged = true
all.set(id, f)
continue
}
// This value has been seen already, either in a previous run or by a previous datasource
// Let's figure out if something changed
const oldV = all.get(id)
if (oldV == f) {
continue
}
all.set(id, f)
somethingChanged = true
}
}
if (!somethingChanged) {
2021-05-07 01:43:32 +02:00
// We don't bother triggering an update
return
}
2021-05-07 01:43:32 +02:00
const newList = []
all.forEach((value, key) => {
2021-05-07 01:43:32 +02:00
newList.push(value)
})
this.features.setData(newList)
this._featuresById.setData(all)
}
}