forked from MapComplete/MapComplete
74 lines
No EOL
2.5 KiB
TypeScript
74 lines
No EOL
2.5 KiB
TypeScript
import FeatureSource from "./FeatureSource";
|
|
import {UIEventSource} from "../UIEventSource";
|
|
|
|
/**
|
|
* Merges features from different featureSources
|
|
* Uses the freshest feature available in the case multiple sources offer data with the same identifier
|
|
*/
|
|
export default class FeatureSourceMerger implements FeatureSource {
|
|
|
|
public features: UIEventSource<{ feature: any; freshness: Date }[]> = new UIEventSource<{ feature: any; freshness: Date }[]>([]);
|
|
public readonly name;
|
|
private readonly _sources: FeatureSource[];
|
|
|
|
constructor(sources: FeatureSource[]) {
|
|
this._sources = sources;
|
|
this.name = "SourceMerger of (" + sources.map(s => s.name).join(", ") + ")"
|
|
const self = this;
|
|
for (let i = 0; i < sources.length; i++) {
|
|
let source = sources[i];
|
|
source.features.addCallback(() => {
|
|
self.Update();
|
|
});
|
|
}
|
|
this.Update();
|
|
}
|
|
|
|
private Update() {
|
|
|
|
let somethingChanged = false;
|
|
const all: Map<string, { feature: any, freshness: Date }> = new Map<string, { feature: any; freshness: Date }>();
|
|
// We seed the dictionary with the previously loaded features
|
|
const oldValues = this.features.data ?? [];
|
|
for (const oldValue of oldValues) {
|
|
all.set(oldValue.feature.id + oldValue.feature._matching_layer_id, oldValue)
|
|
}
|
|
|
|
for (const source of this._sources) {
|
|
if (source?.features?.data === undefined) {
|
|
continue;
|
|
}
|
|
for (const f of source.features.data) {
|
|
const id = f.feature.properties.id + f.feature._matching_layer_id;
|
|
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.freshness < f.freshness) {
|
|
// Jup, this feature is fresher
|
|
all.set(id, f);
|
|
somethingChanged = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!somethingChanged) {
|
|
// We don't bother triggering an update
|
|
return;
|
|
}
|
|
|
|
const newList = [];
|
|
all.forEach((value, key) => {
|
|
newList.push(value)
|
|
})
|
|
this.features.setData(newList);
|
|
}
|
|
|
|
|
|
} |