forked from MapComplete/MapComplete
Partial fix of opening the selected element
This commit is contained in:
parent
9e21ec1182
commit
e374bb355c
7 changed files with 100 additions and 259 deletions
|
@ -35,6 +35,7 @@ import {LayoutConfigJson} from "./Models/ThemeConfig/Json/LayoutConfigJson";
|
||||||
import LayoutConfig from "./Models/ThemeConfig/LayoutConfig";
|
import LayoutConfig from "./Models/ThemeConfig/LayoutConfig";
|
||||||
import LayerConfig from "./Models/ThemeConfig/LayerConfig";
|
import LayerConfig from "./Models/ThemeConfig/LayerConfig";
|
||||||
import Minimap from "./UI/Base/Minimap";
|
import Minimap from "./UI/Base/Minimap";
|
||||||
|
import SelectedFeatureHandler from "./Logic/Actors/SelectedFeatureHandler";
|
||||||
|
|
||||||
export class InitUiElements {
|
export class InitUiElements {
|
||||||
static InitAll(
|
static InitAll(
|
||||||
|
@ -194,6 +195,8 @@ export class InitUiElements {
|
||||||
State.state.locationControl.ping();
|
State.state.locationControl.ping();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
new SelectedFeatureHandler(Hash.hash, State.state)
|
||||||
|
|
||||||
// Reset the loading message once things are loaded
|
// Reset the loading message once things are loaded
|
||||||
new CenterMessageBox().AttachTo("centermessage");
|
new CenterMessageBox().AttachTo("centermessage");
|
||||||
document
|
document
|
||||||
|
@ -404,15 +407,6 @@ export class InitUiElements {
|
||||||
}, state
|
}, state
|
||||||
);
|
);
|
||||||
|
|
||||||
/* const selectedFeatureHandler = new SelectedFeatureHandler(
|
|
||||||
Hash.hash,
|
|
||||||
State.state.selectedElement,
|
|
||||||
source,
|
|
||||||
State.state.osmApiFeatureSource
|
|
||||||
);
|
|
||||||
selectedFeatureHandler.zoomToSelectedFeature(
|
|
||||||
State.state.locationControl
|
|
||||||
);*/
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private static setupAllLayerElements() {
|
private static setupAllLayerElements() {
|
||||||
|
|
|
@ -1,49 +1,45 @@
|
||||||
import {UIEventSource} from "../UIEventSource";
|
import {UIEventSource} from "../UIEventSource";
|
||||||
import FeatureSource from "../FeatureSource/FeatureSource";
|
|
||||||
import {OsmObject} from "../Osm/OsmObject";
|
import {OsmObject} from "../Osm/OsmObject";
|
||||||
import Loc from "../../Models/Loc";
|
import Loc from "../../Models/Loc";
|
||||||
import FeaturePipeline from "../FeatureSource/FeaturePipeline";
|
import {ElementStorage} from "../ElementStorage";
|
||||||
import OsmApiFeatureSource from "../FeatureSource/Sources/OsmApiFeatureSource";
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Makes sure the hash shows the selected element and vice-versa.
|
* Makes sure the hash shows the selected element and vice-versa.
|
||||||
*/
|
*/
|
||||||
export default class SelectedFeatureHandler {
|
export default class SelectedFeatureHandler {
|
||||||
private static readonly _no_trigger_on = ["welcome", "copyright", "layers", "new"]
|
private static readonly _no_trigger_on = ["welcome", "copyright", "layers", "new"]
|
||||||
private readonly _featureSource: FeatureSource;
|
hash: UIEventSource<string>;
|
||||||
private readonly _hash: UIEventSource<string>;
|
private readonly state: {
|
||||||
private readonly _selectedFeature: UIEventSource<any>;
|
selectedElement: UIEventSource<any>
|
||||||
private readonly _osmApiSource: OsmApiFeatureSource;
|
}
|
||||||
|
|
||||||
constructor(hash: UIEventSource<string>,
|
constructor(
|
||||||
selectedFeature: UIEventSource<any>,
|
hash: UIEventSource<string>,
|
||||||
featureSource: FeaturePipeline,
|
state: {
|
||||||
osmApiSource: OsmApiFeatureSource) {
|
selectedElement: UIEventSource<any>,
|
||||||
this._hash = hash;
|
allElements: ElementStorage;
|
||||||
this._selectedFeature = selectedFeature;
|
}
|
||||||
this._featureSource = featureSource;
|
) {
|
||||||
this._osmApiSource = osmApiSource;
|
this.hash = hash;
|
||||||
const self = this;
|
|
||||||
|
this.state = state
|
||||||
|
|
||||||
|
// Getting a blank hash clears the selected element
|
||||||
hash.addCallback(h => {
|
hash.addCallback(h => {
|
||||||
if (h === undefined || h === "") {
|
if (h === undefined || h === "") {
|
||||||
selectedFeature.setData(undefined);
|
state.selectedElement.setData(undefined);
|
||||||
} else {
|
}else{
|
||||||
self.selectFeature();
|
const feature = state.allElements.ContainingFeatures.get(h)
|
||||||
|
if(feature !== undefined){
|
||||||
|
state.selectedElement.setData(feature)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
hash.addCallbackAndRunD(h => {
|
|
||||||
try {
|
|
||||||
self.downloadFeature(h)
|
|
||||||
} catch (e) {
|
|
||||||
console.error("Could not download feature, probably a weird hash")
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
featureSource.features.addCallback(_ => self.selectFeature());
|
state.selectedElement.addCallback(feature => {
|
||||||
|
|
||||||
selectedFeature.addCallback(feature => {
|
|
||||||
if (feature === undefined) {
|
if (feature === undefined) {
|
||||||
|
console.trace("Resetting hash")
|
||||||
if (SelectedFeatureHandler._no_trigger_on.indexOf(hash.data) < 0) {
|
if (SelectedFeatureHandler._no_trigger_on.indexOf(hash.data) < 0) {
|
||||||
hash.setData("")
|
hash.setData("")
|
||||||
}
|
}
|
||||||
|
@ -55,13 +51,11 @@ export default class SelectedFeatureHandler {
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
this.selectFeature();
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// If a feature is selected via the hash, zoom there
|
// If a feature is selected via the hash, zoom there
|
||||||
public zoomToSelectedFeature(location: UIEventSource<Loc>) {
|
public zoomToSelectedFeature(location: UIEventSource<Loc>) {
|
||||||
const hash = this._hash.data;
|
const hash = this.hash.data;
|
||||||
if (hash === undefined || SelectedFeatureHandler._no_trigger_on.indexOf(hash) >= 0) {
|
if (hash === undefined || SelectedFeatureHandler._no_trigger_on.indexOf(hash) >= 0) {
|
||||||
return; // No valid feature selected
|
return; // No valid feature selected
|
||||||
}
|
}
|
||||||
|
@ -80,42 +74,4 @@ export default class SelectedFeatureHandler {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private downloadFeature(hash: string) {
|
|
||||||
if (hash === undefined || hash === "") {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (SelectedFeatureHandler._no_trigger_on.indexOf(hash) >= 0) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
try {
|
|
||||||
|
|
||||||
this._osmApiSource.load(hash)
|
|
||||||
} catch (e) {
|
|
||||||
console.log("Could not download feature, probably a weird hash:", hash)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private selectFeature() {
|
|
||||||
const features = this._featureSource?.features?.data;
|
|
||||||
if (features === undefined) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (this._selectedFeature.data?.properties?.id === this._hash.data) {
|
|
||||||
// Feature already selected
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const hash = this._hash.data;
|
|
||||||
if (hash === undefined || hash === "" || hash === "#") {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
for (const feature of features) {
|
|
||||||
const id = feature.feature?.properties?.id;
|
|
||||||
if (id === hash) {
|
|
||||||
this._selectedFeature.setData(feature.feature);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
|
@ -1,9 +1,7 @@
|
||||||
/// Given a feature source, calculates a list of OSM-contributors who mapped the latest versions
|
/// Given a feature source, calculates a list of OSM-contributors who mapped the latest versions
|
||||||
import FeatureSource from "./FeatureSource/FeatureSource";
|
|
||||||
import {UIEventSource} from "./UIEventSource";
|
import {UIEventSource} from "./UIEventSource";
|
||||||
import FeaturePipeline from "./FeatureSource/FeaturePipeline";
|
import FeaturePipeline from "./FeatureSource/FeaturePipeline";
|
||||||
import Loc from "../Models/Loc";
|
import Loc from "../Models/Loc";
|
||||||
import State from "../State";
|
|
||||||
import {BBox} from "./GeoOperations";
|
import {BBox} from "./GeoOperations";
|
||||||
|
|
||||||
export default class ContributorCount {
|
export default class ContributorCount {
|
||||||
|
@ -33,7 +31,7 @@ export default class ContributorCount {
|
||||||
if (this.lastUpdate !== undefined && ((now.getTime() - this.lastUpdate.getTime()) < 1000 * 60)) {
|
if (this.lastUpdate !== undefined && ((now.getTime() - this.lastUpdate.getTime()) < 1000 * 60)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
console.log("Calculating contributors")
|
this.lastUpdate = now;
|
||||||
const featuresList = this.state.featurePipeline.GetAllFeaturesWithin(bbox)
|
const featuresList = this.state.featurePipeline.GetAllFeaturesWithin(bbox)
|
||||||
const hist = new Map<string, number>();
|
const hist = new Map<string, number>();
|
||||||
for (const list of featuresList) {
|
for (const list of featuresList) {
|
||||||
|
|
|
@ -36,7 +36,6 @@ export default class FeaturePipeline implements FeatureSourceState {
|
||||||
constructor(
|
constructor(
|
||||||
handleFeatureSource: (source: FeatureSourceForLayer) => void,
|
handleFeatureSource: (source: FeatureSourceForLayer) => void,
|
||||||
state: {
|
state: {
|
||||||
osmApiFeatureSource: FeatureSource,
|
|
||||||
filteredLayers: UIEventSource<FilteredLayer[]>,
|
filteredLayers: UIEventSource<FilteredLayer[]>,
|
||||||
locationControl: UIEventSource<Loc>,
|
locationControl: UIEventSource<Loc>,
|
||||||
selectedElement: UIEventSource<any>,
|
selectedElement: UIEventSource<any>,
|
||||||
|
|
|
@ -1,110 +0,0 @@
|
||||||
import FeatureSource from "../FeatureSource";
|
|
||||||
import {UIEventSource} from "../../UIEventSource";
|
|
||||||
import Loc from "../../../Models/Loc";
|
|
||||||
import FilteredLayer from "../../../Models/FilteredLayer";
|
|
||||||
import {Utils} from "../../../Utils";
|
|
||||||
import {OsmObject} from "../../Osm/OsmObject";
|
|
||||||
|
|
||||||
|
|
||||||
export default class OsmApiFeatureSource implements FeatureSource {
|
|
||||||
public readonly features: UIEventSource<{ feature: any; freshness: Date }[]> = new UIEventSource<{ feature: any; freshness: Date }[]>([]);
|
|
||||||
public readonly name: string = "OsmApiFeatureSource";
|
|
||||||
private readonly loadedTiles: Set<string> = new Set<string>();
|
|
||||||
private readonly _state: {
|
|
||||||
leafletMap: UIEventSource<any>;
|
|
||||||
locationControl: UIEventSource<Loc>, filteredLayers: UIEventSource<FilteredLayer[]>};
|
|
||||||
|
|
||||||
constructor(state: {locationControl: UIEventSource<Loc>, filteredLayers: UIEventSource<FilteredLayer[]>, leafletMap: UIEventSource<any>,
|
|
||||||
overpassMaxZoom: UIEventSource<number>}) {
|
|
||||||
this._state = state;
|
|
||||||
const self = this;
|
|
||||||
function update(){
|
|
||||||
const minZoom = state.overpassMaxZoom.data;
|
|
||||||
const location = state.locationControl.data
|
|
||||||
if(minZoom === undefined || location === undefined){
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if(minZoom < 14){
|
|
||||||
throw "MinZoom should be at least 14 or higher, OSM-api won't work otherwise"
|
|
||||||
}
|
|
||||||
if(location.zoom > minZoom){
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
self.loadArea()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
public load(id: string) {
|
|
||||||
if (id.indexOf("-") >= 0) {
|
|
||||||
// Newly added point - not yet in OSM
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
console.debug("Downloading", id, "from the OSM-API")
|
|
||||||
OsmObject.DownloadObject(id).addCallbackAndRunD(element => {
|
|
||||||
try {
|
|
||||||
const geojson = element.asGeoJson();
|
|
||||||
geojson.id = geojson.properties.id;
|
|
||||||
this.features.setData([{feature: geojson, freshness: element.timestamp}])
|
|
||||||
} catch (e) {
|
|
||||||
console.error(e)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Loads the current inview-area
|
|
||||||
*/
|
|
||||||
public loadArea(z: number = 14): boolean {
|
|
||||||
const layers = this._state.filteredLayers.data;
|
|
||||||
|
|
||||||
const disabledLayers = layers.filter(layer => layer.layerDef.source.overpassScript !== undefined || layer.layerDef.source.geojsonSource !== undefined)
|
|
||||||
if (disabledLayers.length > 0) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
if (this._state.leafletMap.data === undefined) {
|
|
||||||
return false; // Not yet inited
|
|
||||||
}
|
|
||||||
const bounds = this._state.leafletMap.data.getBounds()
|
|
||||||
const tileRange = Utils.TileRangeBetween(z, bounds.getNorth(), bounds.getEast(), bounds.getSouth(), bounds.getWest())
|
|
||||||
const self = this;
|
|
||||||
Utils.MapRange(tileRange, (x, y) => {
|
|
||||||
const key = x + "/" + y;
|
|
||||||
if (self.loadedTiles.has(key)) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
self.loadedTiles.add(key);
|
|
||||||
|
|
||||||
const bounds = Utils.tile_bounds(z, x, y);
|
|
||||||
console.log("Loading OSM data tile", z, x, y, " with bounds", bounds)
|
|
||||||
OsmObject.LoadArea(bounds, objects => {
|
|
||||||
const keptGeoJson: { feature: any, freshness: Date }[] = []
|
|
||||||
// Which layer does the object match?
|
|
||||||
for (const object of objects) {
|
|
||||||
|
|
||||||
for (const flayer of layers) {
|
|
||||||
const layer = flayer.layerDef;
|
|
||||||
const tags = object.tags
|
|
||||||
const doesMatch = layer.source.osmTags.matchesProperties(tags);
|
|
||||||
if (doesMatch) {
|
|
||||||
const geoJson = object.asGeoJson();
|
|
||||||
geoJson._matching_layer_id = layer.id
|
|
||||||
keptGeoJson.push({feature: geoJson, freshness: object.timestamp})
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
self.features.setData(keptGeoJson)
|
|
||||||
});
|
|
||||||
|
|
||||||
});
|
|
||||||
|
|
||||||
return true;
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
5
State.ts
5
State.ts
|
@ -13,7 +13,6 @@ import Loc from "./Models/Loc";
|
||||||
import Constants from "./Models/Constants";
|
import Constants from "./Models/Constants";
|
||||||
import TitleHandler from "./Logic/Actors/TitleHandler";
|
import TitleHandler from "./Logic/Actors/TitleHandler";
|
||||||
import PendingChangesUploader from "./Logic/Actors/PendingChangesUploader";
|
import PendingChangesUploader from "./Logic/Actors/PendingChangesUploader";
|
||||||
import OsmApiFeatureSource from "./Logic/FeatureSource/Sources/OsmApiFeatureSource";
|
|
||||||
import FeaturePipeline from "./Logic/FeatureSource/FeaturePipeline";
|
import FeaturePipeline from "./Logic/FeatureSource/FeaturePipeline";
|
||||||
import FilteredLayer from "./Models/FilteredLayer";
|
import FilteredLayer from "./Models/FilteredLayer";
|
||||||
import ChangeToElementsActor from "./Logic/Actors/ChangeToElementsActor";
|
import ChangeToElementsActor from "./Logic/Actors/ChangeToElementsActor";
|
||||||
|
@ -55,8 +54,6 @@ export default class State {
|
||||||
|
|
||||||
public favouriteLayers: UIEventSource<string[]>;
|
public favouriteLayers: UIEventSource<string[]>;
|
||||||
|
|
||||||
public osmApiFeatureSource: OsmApiFeatureSource;
|
|
||||||
|
|
||||||
public filteredLayers: UIEventSource<FilteredLayer[]> = new UIEventSource<FilteredLayer[]>([], "filteredLayers");
|
public filteredLayers: UIEventSource<FilteredLayer[]> = new UIEventSource<FilteredLayer[]>([], "filteredLayers");
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -395,8 +392,6 @@ export default class State {
|
||||||
|
|
||||||
new ChangeToElementsActor(this.changes, this.allElements)
|
new ChangeToElementsActor(this.changes, this.allElements)
|
||||||
|
|
||||||
this.osmApiFeatureSource = new OsmApiFeatureSource(this)
|
|
||||||
|
|
||||||
new PendingChangesUploader(this.changes, this.selectedElement);
|
new PendingChangesUploader(this.changes, this.selectedElement);
|
||||||
|
|
||||||
this.mangroveIdentity = new MangroveIdentity(
|
this.mangroveIdentity = new MangroveIdentity(
|
||||||
|
|
|
@ -16,11 +16,20 @@ export default class ShowDataLayer {
|
||||||
|
|
||||||
// Used to generate a fresh ID when needed
|
// Used to generate a fresh ID when needed
|
||||||
private _cleanCount = 0;
|
private _cleanCount = 0;
|
||||||
|
private geoLayer = undefined;
|
||||||
constructor(options: ShowDataLayerOptions & { layerToShow: LayerConfig}) {
|
|
||||||
|
/**
|
||||||
|
* If the selected element triggers, this is used to lookup the correct layer and to open the popup
|
||||||
|
* Used to avoid a lot of callbacks on the selected element
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
private readonly leafletLayersPerId = new Map<string, { feature: any, leafletlayer: any }>()
|
||||||
|
|
||||||
|
|
||||||
|
constructor(options: ShowDataLayerOptions & { layerToShow: LayerConfig }) {
|
||||||
this._leafletMap = options.leafletMap;
|
this._leafletMap = options.leafletMap;
|
||||||
this._enablePopups = options.enablePopups ?? true;
|
this._enablePopups = options.enablePopups ?? true;
|
||||||
if(options.features === undefined){
|
if (options.features === undefined) {
|
||||||
throw "Invalid ShowDataLayer invocation"
|
throw "Invalid ShowDataLayer invocation"
|
||||||
}
|
}
|
||||||
const features = options.features.features.map(featFreshes => featFreshes.map(ff => ff.feature));
|
const features = options.features.features.map(featFreshes => featFreshes.map(ff => ff.feature));
|
||||||
|
@ -28,58 +37,77 @@ export default class ShowDataLayer {
|
||||||
this._layerToShow = options.layerToShow;
|
this._layerToShow = options.layerToShow;
|
||||||
const self = this;
|
const self = this;
|
||||||
|
|
||||||
let geoLayer = undefined;
|
features.addCallback(() => self.update(options));
|
||||||
|
options.leafletMap.addCallback(() => self.update(options));
|
||||||
|
this.update(options);
|
||||||
|
|
||||||
function update() {
|
|
||||||
if (features.data === undefined) {
|
State.state.selectedElement.addCallbackAndRunD(selected => {
|
||||||
|
if (self._leafletMap.data === undefined) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const mp =options. leafletMap.data;
|
const v = self.leafletLayersPerId.get(selected.properties.id)
|
||||||
|
if(v === undefined){return;}
|
||||||
if (mp === undefined) {
|
const leafletLayer = v.leafletlayer
|
||||||
|
const feature = v.feature
|
||||||
|
if (leafletLayer.getPopup().isOpen()) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
if (selected.properties.id === feature.properties.id) {
|
||||||
self._cleanCount++
|
// A small sanity check to prevent infinite loops:
|
||||||
// clean all the old stuff away, if any
|
if (selected.geometry.type === feature.geometry.type // If a feature is rendered both as way and as point, opening one popup might trigger the other to open, which might trigger the one to open again
|
||||||
if (geoLayer !== undefined) {
|
&& feature.id === feature.properties.id // the feature might have as id 'node/-1' and as 'feature.properties.id' = 'the newly assigned id'. That is no good too
|
||||||
mp.removeLayer(geoLayer);
|
) {
|
||||||
}
|
leafletLayer.openPopup()
|
||||||
|
|
||||||
const allFeats = features.data;
|
|
||||||
geoLayer = self.CreateGeojsonLayer();
|
|
||||||
for (const feat of allFeats) {
|
|
||||||
if (feat === undefined) {
|
|
||||||
continue
|
|
||||||
}
|
}
|
||||||
// @ts-ignore
|
if (feature.id !== feature.properties.id) {
|
||||||
geoLayer.addData(feat);
|
console.trace("Not opening the popup for", feature)
|
||||||
}
|
|
||||||
|
|
||||||
mp.addLayer(geoLayer)
|
|
||||||
|
|
||||||
if (options.zoomToFeatures ?? false) {
|
|
||||||
try {
|
|
||||||
mp.fitBounds(geoLayer.getBounds(), {animate: false})
|
|
||||||
} catch (e) {
|
|
||||||
console.error(e)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
if (self._enablePopups) {
|
})
|
||||||
State.state.selectedElement.ping()
|
}
|
||||||
}
|
|
||||||
|
private update(options) {
|
||||||
|
if (this._features.data === undefined) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const mp = options.leafletMap.data;
|
||||||
|
|
||||||
|
if (mp === undefined) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
this._cleanCount++
|
||||||
|
// clean all the old stuff away, if any
|
||||||
|
if (this.geoLayer !== undefined) {
|
||||||
|
mp.removeLayer(this.geoLayer);
|
||||||
}
|
}
|
||||||
|
|
||||||
features.addCallback(() => update());
|
const allFeats = this._features.data;
|
||||||
options.leafletMap.addCallback(() => update());
|
this.geoLayer = this.CreateGeojsonLayer();
|
||||||
update();
|
for (const feat of allFeats) {
|
||||||
|
if (feat === undefined) {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
this.geoLayer.addData(feat);
|
||||||
|
}
|
||||||
|
|
||||||
|
mp.addLayer(this.geoLayer)
|
||||||
|
|
||||||
|
if (options.zoomToFeatures ?? false) {
|
||||||
|
try {
|
||||||
|
mp.fitBounds(this.geoLayer.getBounds(), {animate: false})
|
||||||
|
} catch (e) {
|
||||||
|
console.error(e)
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
private createStyleFor(feature) {
|
private createStyleFor(feature) {
|
||||||
const tagsSource = State.state.allElements.addOrGetElement(feature);
|
const tagsSource = State.state.allElements.addOrGetElement(feature);
|
||||||
// Every object is tied to exactly one layer
|
// Every object is tied to exactly one layer
|
||||||
const layer = this._layerToShow
|
const layer = this._layerToShow
|
||||||
return layer?.GenerateLeafletStyle(tagsSource, true);
|
return layer?.GenerateLeafletStyle(tagsSource, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -88,7 +116,7 @@ export default class ShowDataLayer {
|
||||||
// We have to convert them to the appropriate icon
|
// We have to convert them to the appropriate icon
|
||||||
// Click handling is done in the next step
|
// Click handling is done in the next step
|
||||||
|
|
||||||
const layer: LayerConfig = this._layerToShow
|
const layer: LayerConfig = this._layerToShow
|
||||||
if (layer === undefined) {
|
if (layer === undefined) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -159,28 +187,9 @@ export default class ShowDataLayer {
|
||||||
infobox.AttachTo(id)
|
infobox.AttachTo(id)
|
||||||
infobox.Activate();
|
infobox.Activate();
|
||||||
});
|
});
|
||||||
const self = this;
|
|
||||||
State.state.selectedElement.addCallbackAndRunD(selected => {
|
|
||||||
if (self._leafletMap.data === undefined) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (leafletLayer.getPopup().isOpen()) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (selected.properties.id === feature.properties.id) {
|
|
||||||
// A small sanity check to prevent infinite loops:
|
|
||||||
if (selected.geometry.type === feature.geometry.type // If a feature is rendered both as way and as point, opening one popup might trigger the other to open, which might trigger the one to open again
|
|
||||||
&& feature.id === feature.properties.id // the feature might have as id 'node/-1' and as 'feature.properties.id' = 'the newly assigned id'. That is no good too
|
|
||||||
) {
|
|
||||||
leafletLayer.openPopup()
|
|
||||||
}
|
|
||||||
if (feature.id !== feature.properties.id) {
|
|
||||||
console.trace("Not opening the popup for", feature)
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
|
// Add the feature to the index to open the popup when needed
|
||||||
|
this.leafletLayersPerId.set(feature.properties.id, {feature: feature, leafletlayer: leafletLayer})
|
||||||
}
|
}
|
||||||
|
|
||||||
private CreateGeojsonLayer(): L.Layer {
|
private CreateGeojsonLayer(): L.Layer {
|
||||||
|
|
Loading…
Reference in a new issue