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 LayerConfig from "./Models/ThemeConfig/LayerConfig"; | ||||
| import Minimap from "./UI/Base/Minimap"; | ||||
| import SelectedFeatureHandler from "./Logic/Actors/SelectedFeatureHandler"; | ||||
| 
 | ||||
| export class InitUiElements { | ||||
|     static InitAll( | ||||
|  | @ -194,6 +195,8 @@ export class InitUiElements { | |||
|             State.state.locationControl.ping(); | ||||
|         } | ||||
| 
 | ||||
|         new SelectedFeatureHandler(Hash.hash, State.state) | ||||
| 
 | ||||
|         // Reset the loading message once things are loaded
 | ||||
|         new CenterMessageBox().AttachTo("centermessage"); | ||||
|         document | ||||
|  | @ -404,15 +407,6 @@ export class InitUiElements { | |||
|             }, state | ||||
|         ); | ||||
| 
 | ||||
|         /*   const selectedFeatureHandler = new SelectedFeatureHandler( | ||||
|                Hash.hash, | ||||
|                State.state.selectedElement, | ||||
|                source, | ||||
|                State.state.osmApiFeatureSource | ||||
|            ); | ||||
|            selectedFeatureHandler.zoomToSelectedFeature( | ||||
|                State.state.locationControl | ||||
|            );*/ | ||||
|     } | ||||
| 
 | ||||
|     private static setupAllLayerElements() { | ||||
|  |  | |||
|  | @ -1,49 +1,45 @@ | |||
| import {UIEventSource} from "../UIEventSource"; | ||||
| import FeatureSource from "../FeatureSource/FeatureSource"; | ||||
| import {OsmObject} from "../Osm/OsmObject"; | ||||
| import Loc from "../../Models/Loc"; | ||||
| import FeaturePipeline from "../FeatureSource/FeaturePipeline"; | ||||
| import OsmApiFeatureSource from "../FeatureSource/Sources/OsmApiFeatureSource"; | ||||
| import {ElementStorage} from "../ElementStorage"; | ||||
| 
 | ||||
| /** | ||||
|  * Makes sure the hash shows the selected element and vice-versa. | ||||
|  */ | ||||
| export default class SelectedFeatureHandler { | ||||
|     private static readonly _no_trigger_on = ["welcome", "copyright", "layers", "new"] | ||||
|     private readonly _featureSource: FeatureSource; | ||||
|     private readonly _hash: UIEventSource<string>; | ||||
|     private readonly _selectedFeature: UIEventSource<any>; | ||||
|     private readonly _osmApiSource: OsmApiFeatureSource; | ||||
|     hash: UIEventSource<string>; | ||||
|     private readonly state: { | ||||
|         selectedElement: UIEventSource<any> | ||||
|     } | ||||
| 
 | ||||
|     constructor(hash: UIEventSource<string>, | ||||
|                 selectedFeature: UIEventSource<any>, | ||||
|                 featureSource: FeaturePipeline, | ||||
|                 osmApiSource: OsmApiFeatureSource) { | ||||
|         this._hash = hash; | ||||
|         this._selectedFeature = selectedFeature; | ||||
|         this._featureSource = featureSource; | ||||
|         this._osmApiSource = osmApiSource; | ||||
|         const self = this; | ||||
|     constructor( | ||||
|         hash: UIEventSource<string>, | ||||
|         state: { | ||||
|             selectedElement: UIEventSource<any>, | ||||
|             allElements: ElementStorage; | ||||
|         } | ||||
|     ) { | ||||
|         this.hash = hash; | ||||
| 
 | ||||
|         this.state = state | ||||
|          | ||||
|         // Getting a blank hash clears the selected element
 | ||||
|         hash.addCallback(h => { | ||||
|             if (h === undefined || h === "") { | ||||
|                 selectedFeature.setData(undefined); | ||||
|             } else { | ||||
|                 self.selectFeature(); | ||||
|                 state.selectedElement.setData(undefined); | ||||
|             }else{ | ||||
|                 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()); | ||||
| 
 | ||||
|         selectedFeature.addCallback(feature => { | ||||
|         state.selectedElement.addCallback(feature => { | ||||
|             if (feature === undefined) { | ||||
|                 console.trace("Resetting hash") | ||||
|                 if (SelectedFeatureHandler._no_trigger_on.indexOf(hash.data) < 0) { | ||||
|                     hash.setData("") | ||||
|                 } | ||||
|  | @ -55,13 +51,11 @@ export default class SelectedFeatureHandler { | |||
|             } | ||||
|         }) | ||||
| 
 | ||||
|         this.selectFeature(); | ||||
| 
 | ||||
|     } | ||||
| 
 | ||||
|     // If a feature is selected via the hash, zoom there
 | ||||
|     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) { | ||||
|             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
 | ||||
| import FeatureSource from "./FeatureSource/FeatureSource"; | ||||
| import {UIEventSource} from "./UIEventSource"; | ||||
| import FeaturePipeline from "./FeatureSource/FeaturePipeline"; | ||||
| import Loc from "../Models/Loc"; | ||||
| import State from "../State"; | ||||
| import {BBox} from "./GeoOperations"; | ||||
| 
 | ||||
| export default class ContributorCount { | ||||
|  | @ -33,7 +31,7 @@ export default class ContributorCount { | |||
|         if (this.lastUpdate !== undefined && ((now.getTime() - this.lastUpdate.getTime()) < 1000 * 60)) { | ||||
|             return; | ||||
|         } | ||||
|         console.log("Calculating contributors") | ||||
|         this.lastUpdate = now; | ||||
|         const featuresList = this.state.featurePipeline.GetAllFeaturesWithin(bbox) | ||||
|         const hist = new Map<string, number>(); | ||||
|         for (const list of featuresList) { | ||||
|  |  | |||
|  | @ -36,7 +36,6 @@ export default class FeaturePipeline implements FeatureSourceState { | |||
|     constructor( | ||||
|         handleFeatureSource: (source: FeatureSourceForLayer) => void, | ||||
|         state: { | ||||
|             osmApiFeatureSource: FeatureSource, | ||||
|             filteredLayers: UIEventSource<FilteredLayer[]>, | ||||
|             locationControl: UIEventSource<Loc>, | ||||
|             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 TitleHandler from "./Logic/Actors/TitleHandler"; | ||||
| import PendingChangesUploader from "./Logic/Actors/PendingChangesUploader"; | ||||
| import OsmApiFeatureSource from "./Logic/FeatureSource/Sources/OsmApiFeatureSource"; | ||||
| import FeaturePipeline from "./Logic/FeatureSource/FeaturePipeline"; | ||||
| import FilteredLayer from "./Models/FilteredLayer"; | ||||
| import ChangeToElementsActor from "./Logic/Actors/ChangeToElementsActor"; | ||||
|  | @ -55,8 +54,6 @@ export default class State { | |||
| 
 | ||||
|     public favouriteLayers: UIEventSource<string[]>; | ||||
| 
 | ||||
|     public osmApiFeatureSource: OsmApiFeatureSource; | ||||
| 
 | ||||
|     public filteredLayers: UIEventSource<FilteredLayer[]> = new UIEventSource<FilteredLayer[]>([], "filteredLayers"); | ||||
| 
 | ||||
|     /** | ||||
|  | @ -395,8 +392,6 @@ export default class State { | |||
| 
 | ||||
|         new ChangeToElementsActor(this.changes, this.allElements) | ||||
| 
 | ||||
|         this.osmApiFeatureSource = new OsmApiFeatureSource(this) | ||||
| 
 | ||||
|         new PendingChangesUploader(this.changes, this.selectedElement); | ||||
| 
 | ||||
|         this.mangroveIdentity = new MangroveIdentity( | ||||
|  |  | |||
|  | @ -16,11 +16,20 @@ export default class ShowDataLayer { | |||
| 
 | ||||
|     // Used to generate a fresh ID when needed
 | ||||
|     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._enablePopups = options.enablePopups ?? true; | ||||
|         if(options.features === undefined){ | ||||
|         if (options.features === undefined) { | ||||
|             throw "Invalid ShowDataLayer invocation" | ||||
|         } | ||||
|         const features = options.features.features.map(featFreshes => featFreshes.map(ff => ff.feature)); | ||||
|  | @ -28,51 +37,70 @@ export default class ShowDataLayer { | |||
|         this._layerToShow = options.layerToShow; | ||||
|         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; | ||||
|             } | ||||
|             const mp =options. leafletMap.data; | ||||
|             const v = self.leafletLayersPerId.get(selected.properties.id) | ||||
|             if(v === undefined){return;} | ||||
|             const leafletLayer = v.leafletlayer | ||||
|             const feature = v.feature | ||||
|             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) | ||||
|                 } | ||||
| 
 | ||||
|             } | ||||
|         }) | ||||
|     } | ||||
| 
 | ||||
|     private update(options) { | ||||
|         if (this._features.data === undefined) { | ||||
|             return; | ||||
|         } | ||||
|         const mp = options.leafletMap.data; | ||||
| 
 | ||||
|         if (mp === undefined) { | ||||
|             return; | ||||
|         } | ||||
| 
 | ||||
|             self._cleanCount++ | ||||
|         this._cleanCount++ | ||||
|         // clean all the old stuff away, if any
 | ||||
|             if (geoLayer !== undefined) { | ||||
|                 mp.removeLayer(geoLayer); | ||||
|         if (this.geoLayer !== undefined) { | ||||
|             mp.removeLayer(this.geoLayer); | ||||
|         } | ||||
| 
 | ||||
|             const allFeats = features.data; | ||||
|             geoLayer = self.CreateGeojsonLayer(); | ||||
|         const allFeats = this._features.data; | ||||
|         this.geoLayer = this.CreateGeojsonLayer(); | ||||
|         for (const feat of allFeats) { | ||||
|             if (feat === undefined) { | ||||
|                 continue | ||||
|             } | ||||
|                 // @ts-ignore
 | ||||
|                 geoLayer.addData(feat); | ||||
|             this.geoLayer.addData(feat); | ||||
|         } | ||||
| 
 | ||||
|                 mp.addLayer(geoLayer) | ||||
|         mp.addLayer(this.geoLayer) | ||||
| 
 | ||||
|         if (options.zoomToFeatures ?? false) { | ||||
|             try { | ||||
|                     mp.fitBounds(geoLayer.getBounds(), {animate: false}) | ||||
|                 mp.fitBounds(this.geoLayer.getBounds(), {animate: false}) | ||||
|             } catch (e) { | ||||
|                 console.error(e) | ||||
|             } | ||||
|         } | ||||
|             if (self._enablePopups) { | ||||
|                 State.state.selectedElement.ping() | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         features.addCallback(() => update()); | ||||
|         options.leafletMap.addCallback(() => update()); | ||||
|         update(); | ||||
|     } | ||||
| 
 | ||||
| 
 | ||||
|  | @ -159,28 +187,9 @@ export default class ShowDataLayer { | |||
|             infobox.AttachTo(id) | ||||
|             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 { | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue