forked from MapComplete/MapComplete
		
	Huge refactoring of the feature pipeline, WIP
This commit is contained in:
		
							parent
							
								
									7793297348
								
							
						
					
					
						commit
						973b5d8bbe
					
				
					 25 changed files with 522 additions and 591 deletions
				
			
		|  | @ -1,111 +0,0 @@ | |||
| import FeatureSource from "./FeatureSource"; | ||||
| import {UIEventSource} from "../UIEventSource"; | ||||
| import {OsmObject} from "../Osm/OsmObject"; | ||||
| import {Utils} from "../../Utils"; | ||||
| import Loc from "../../Models/Loc"; | ||||
| import FilteredLayer from "../../Models/FilteredLayer"; | ||||
| import Constants from "../../Models/Constants"; | ||||
| 
 | ||||
| 
 | ||||
| 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(minZoom = undefined, state: {locationControl: UIEventSource<Loc>, filteredLayers: UIEventSource<FilteredLayer[]>, leafletMap: UIEventSource<any>}) { | ||||
|         this._state = state; | ||||
|         if(minZoom !== undefined){ | ||||
|             if(minZoom < 14){ | ||||
|                 throw "MinZoom should be at least 14 or higher, OSM-api won't work otherwise" | ||||
|             } | ||||
|             const self = this; | ||||
|             state.locationControl.addCallbackAndRunD(location => { | ||||
|                 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; | ||||
|         } | ||||
|         const loc = this._state.locationControl.data; | ||||
|         if (loc.zoom < Constants.useOsmApiAt) { | ||||
|             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; | ||||
| 
 | ||||
|     } | ||||
| 
 | ||||
| } | ||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue