UNPKG

@jmarcelof/leaflet-ng2

Version:
431 lines (410 loc) 16.5 kB
import { AfterContentInit, Directive, EventEmitter, Input, OnDestroy, Output, SkipSelf, } from "@angular/core"; import { Feature as GeoJSONFeature, FeatureCollection as GeoJSONFeatureCollection, GeoJsonObject, GeometryObject, Point, } from "geojson"; import { GeoJSON, GeoJSONOptions, LatLng, Layer, LeafletEvent, LeafletMouseEvent, Map, Marker, PathOptions, PopupEvent, TooltipEvent, } from "leaflet"; import { DEFAULT_STYLE } from "./consts"; import { LayerGroupProvider } from "./layer-group.provider"; import { LayerProvider } from "./layer.provider"; /** * Interface for the styler function of the GeoJSON directive. * * You can return an individual style (PathOption) for each feature. As basis you get the feature itself and the default * style. * * *Note: This functions is enhanced against the original style function with a default style* * @link http://leafletjs.com/reference-1.2.0.html#geojson-style Original Leaflet documentation */ export type IGeoJSONStylerFn<T> = ( geoJSON: GeoJSONFeature<GeometryObject, T>, defaultStyle: PathOptions, ) => PathOptions; /** * Interface for the filter function of the GeoJSON directive. * * You can return a boolean value on each feature according if you want to add the feature or not. * @link http://leafletjs.com/reference-1.2.0.html#geojson-filter Original Leaflet documentation */ export type IGeoJSONFilterFn<T> = (feature: GeoJSONFeature<GeometryObject, T>) => boolean; /** * Interface for the point to layer function of the GeoJSON directive. * * You can return any type of Layer that should represent the feature of type point. * @link http://leafletjs.com/reference-1.2.0.html#geojson-pointtolayer Original Leaflet documentation */ export type IGeoJSONPointToLayerFn<T> = (geoJSON: GeoJSONFeature<Point, T>, latLng: LatLng) => Layer; /** * Interface for the protected middleware property of the GeoJSON directive. */ export interface IGeoJSONDirectiveMiddlewareDictionary<T> { styler?: IGeoJSONStylerFn<T>; filter?: IGeoJSONFilterFn<T>; pointToLayer?: IGeoJSONPointToLayerFn<T>; defaultStyle?: PathOptions; } /** * Angular2 directive for GeoJSON of Leaflet. * * *You can use this directive in an Angular2 template after importing `YagaModule`.* * * How to use in a template: * ```html * <yaga-map> * <yaga-geojson * [(data)]="..." * [(stroke)]="..." * [(color)]="..." * [(weight)]="..." * [(opacity)]="..." * [(lineCap)]="..." * [(lineJoin)]="..." * [(dashArray)]="..." * [(dashOffset)]="..." * [(fill)]="..." * [(fillColor)]="..." * [(fillOpacity)]="..." * [(fillRule)]="..." * [(className)]="..." * [(lat)]="..." * [(lng)]="..." * [(radius)]="..." * * (add)="..." * (remove)="..." * (popupopen)="..." * (popupclose)="..." * (tooltipopen)="..." * (tooltipclose)="..." * (click)="..." * (dblclick)="..." * (mousedown)="..." * (mouseover)="..." * (mouseout)="..." * (contextmenu)="..." * (onEachFeature)="..." * * [data]="..." * [filter]="..." * [pointToLayer]="..." * [styler]="..." * [defaultStyle]="..." * > * </yaga-geojson> * </yaga-map> * ``` * * @link http://leafletjs.com/reference-1.2.0.html#geojson Original Leaflet documentation * @link https://leaflet-ng2.yagajs.org/latest/browser-test?grep=GeoJSON%20Directive Unit-Test * @link https://leaflet-ng2.yagajs.org/latest/coverage/lcov-report/lib/geojson.directive.js.html * Test coverage * @link https://leaflet-ng2.yagajs.org/latest/typedoc/classes/geojson.directive.js.html API documentation * @example https://leaflet-ng2.yagajs.org/latest/examples/geojson-directive/ */ @Directive({ providers: [ LayerGroupProvider, LayerProvider ], selector: "yaga-geojson", }) export class GeoJSONDirective<T> extends GeoJSON implements OnDestroy, AfterContentInit { /* tslint:disable:max-line-length */ @Output() public dataChange: EventEmitter<GeoJSONFeatureCollection<GeometryObject, T>> = new EventEmitter(); /* tslint:enable */ /** * From leaflet fired add event. * Use it with `<yaga-geojson (add)="processEvent($event)">` * @link http://leafletjs.com/reference-1.2.0.html#geojson-add Original Leaflet documentation */ @Output("add") public addEvent: EventEmitter<LeafletEvent> = new EventEmitter(); /** * From leaflet fired remove event. * Use it with `<yaga-geojson (remove)="processEvent($event)">` * @link http://leafletjs.com/reference-1.2.0.html#geojson-remove Original Leaflet documentation */ @Output("remove") public removeEvent: EventEmitter<LeafletEvent> = new EventEmitter(); /** * From leaflet fired popupopen event. * Use it with `<yaga-geojson (popupopen)="processEvent($event)">` * @link http://leafletjs.com/reference-1.2.0.html#geojson-popupopen Original Leaflet documentation */ @Output("popupopen") public popupopenEvent: EventEmitter<PopupEvent> = new EventEmitter(); /** * From leaflet fired popupclose event. * Use it with `<yaga-geojson (popupclose)="processEvent($event)">` * @link http://leafletjs.com/reference-1.2.0.html#geojson-popupclose Original Leaflet documentation */ @Output("popupclose") public popupcloseEvent: EventEmitter<PopupEvent> = new EventEmitter(); /** * From leaflet fired tooltipopen event. * Use it with `<yaga-geojson (tooltipopen)="processEvent($event)">` * @link http://leafletjs.com/reference-1.2.0.html#geojson-tooltipopen Original Leaflet documentation */ @Output("tooltipopen") public tooltipopenEvent: EventEmitter<TooltipEvent> = new EventEmitter(); /** * From leaflet fired tooltipclose event. * Use it with `<yaga-geojson (tooltipclose)="processEvent($event)">` * @link http://leafletjs.com/reference-1.2.0.html#geojson-tooltipclose Original Leaflet documentation */ @Output("tooltipclose") public tooltipcloseEvent: EventEmitter<TooltipEvent> = new EventEmitter(); /** * From leaflet fired click event. * Use it with `<yaga-geojson (click)="processEvent($event)">` * @link http://leafletjs.com/reference-1.2.0.html#geojson-click Original Leaflet documentation */ @Output("click") public clickEvent: EventEmitter<LeafletMouseEvent> = new EventEmitter(); /** * From leaflet fired dblclick event. * Use it with `<yaga-geojson (dblclick)="processEvent($event)">` * @link http://leafletjs.com/reference-1.2.0.html#geojson-dblclick Original Leaflet documentation */ @Output("dblclick") public dblclickEvent: EventEmitter<LeafletMouseEvent> = new EventEmitter(); /** * From leaflet fired mousedown event. * Use it with `<yaga-geojson (mousedown)="processEvent($event)">` * @link http://leafletjs.com/reference-1.2.0.html#geojson-mousedown Original Leaflet documentation */ @Output("mousedown") public mousedownEvent: EventEmitter<LeafletMouseEvent> = new EventEmitter(); /** * From leaflet fired mouseover event. * Use it with `<yaga-geojson (mouseover)="processEvent($event)">` * @link http://leafletjs.com/reference-1.2.0.html#geojson-mouseover Original Leaflet documentation */ @Output("mouseover") public mouseoverEvent: EventEmitter<LeafletMouseEvent> = new EventEmitter(); /** * From leaflet fired mouseout event. * Use it with `<yaga-geojson (mouseout)="processEvent($event)">` * @link http://leafletjs.com/reference-1.2.0.html#geojson-mouseout Original Leaflet documentation */ @Output("mouseout") public mouseoutEvent: EventEmitter<LeafletMouseEvent> = new EventEmitter(); /** * From leaflet fired contextmenu event. * Use it with `<yaga-geojson (contextmenu)="processEvent($event)">` * @link http://leafletjs.com/reference-1.2.0.html#geojson-contextmenu Original Leaflet documentation */ @Output("contextmenu") public contextmenuEvent: EventEmitter<LeafletMouseEvent> = new EventEmitter(); /* tslint:disable:max-line-length */ @Output("onEachFeature") public onEachFeatureEvent: EventEmitter<{feature: GeoJSONFeature<GeometryObject, T>, layer: Layer}> = new EventEmitter(); /* tslint:enable */ /** * Property to prevent changes before directive is initialized */ protected initialized: boolean = false; /** * Object that stores the middleware functions and the default style */ protected middleware: IGeoJSONDirectiveMiddlewareDictionary<T> = { defaultStyle: DEFAULT_STYLE, }; constructor( @SkipSelf() protected parentLayerGroupProvider: LayerGroupProvider, layerGroupProvider: LayerGroupProvider, layerProvider: LayerProvider, ) { super(({features: [], type: "FeatureCollection"} as GeoJsonObject), { filter: (feature: GeoJSONFeature<GeometryObject, T>) => { if (this.middleware.filter) { return this.middleware.filter(feature); } return true; }, onEachFeature: (feature: GeoJSONFeature<GeometryObject, T>, layer: Layer) => { this.onEachFeatureEvent.emit({feature, layer}); }, pointToLayer: (geoJSON: GeoJSONFeature<Point, T>, latLng: LatLng): Layer => { if (this.middleware.pointToLayer) { return this.middleware.pointToLayer(geoJSON, latLng); } return new Marker(latLng); }, style: (geoJSON: GeoJSONFeature<GeometryObject, T>): PathOptions => { const defaultStyle = this.middleware.defaultStyle; if (this.middleware.styler) { return this.middleware.styler(geoJSON, defaultStyle as PathOptions); } return defaultStyle as PathOptions; }, } as GeoJSONOptions); layerProvider.ref = this; layerGroupProvider.ref = this; this.parentLayerGroupProvider.ref!.addLayer(this); // Events this.on("add", (event: LeafletEvent) => { this.addEvent.emit(event); }); this.on("remove", (event: LeafletEvent) => { this.removeEvent.emit(event); }); this.on("popupopen", (event: LeafletEvent) => { this.popupopenEvent.emit(event as PopupEvent); }); this.on("popupclose", (event: LeafletEvent) => { this.popupcloseEvent.emit(event as PopupEvent); }); this.on("tooltipopen", (event: LeafletEvent) => { this.tooltipopenEvent.emit(event as TooltipEvent); }); this.on("tooltipclose", (event: LeafletEvent) => { this.tooltipcloseEvent.emit(event as TooltipEvent); }); this.on("click", (event: LeafletEvent) => { this.clickEvent.emit(event as LeafletMouseEvent); }); this.on("dblclick", (event: LeafletEvent) => { this.dblclickEvent.emit(event as LeafletMouseEvent); }); this.on("mousedown", (event: LeafletEvent) => { this.mousedownEvent.emit(event as LeafletMouseEvent); }); this.on("mouseover", (event: LeafletEvent) => { this.mouseoverEvent.emit(event as LeafletMouseEvent); }); this.on("mouseout", (event: LeafletEvent) => { this.mouseoutEvent.emit(event as LeafletMouseEvent); }); this.on("contextmenu", (event: LeafletEvent) => { this.contextmenuEvent.emit(event as LeafletMouseEvent); }); } /** * Internal method that provides the initialization of the child popup or tooltip */ public ngAfterContentInit(): void { this.initialized = true; this.redraw(); } /** * Internal method to provide the removal of the layer in Leaflet, when removing it from the Angular template */ public ngOnDestroy(): void { this.removeFrom(this.parentLayerGroupProvider.ref as Map); } /** * Derived method of the original addData. * @link http://leafletjs.com/reference-1.2.0.html#geojson-adddata Original Leaflet documentation */ public addData(data: GeoJSONFeature<GeometryObject, T>): Layer { const returnValue: Layer = super.addData(data); if (!this.initialized) { return returnValue; } this.dataChange.emit((this.toGeoJSON() as GeoJSONFeatureCollection<GeometryObject, T>)); return returnValue; } /** * Derived method of the original clearLayers. * @link http://leafletjs.com/reference-1.2.0.html#geojson-clearlayers Original Leaflet documentation */ public clearLayers(): this { super.clearLayers(); this.dataChange.emit((this.toGeoJSON() as GeoJSONFeatureCollection<GeometryObject, T>)); return this; } /** * Method to remove all existing data and add the new data in one step. * * *Note: this is a combination of `clearLayers` and `addData`* */ public setData(val: GeoJSONFeatureCollection<GeometryObject, T>): this { super.clearLayers(); super.addData(val); return this; } /** * Two-Way bound property for the data geoJSON data. * Use it with `<yaga-geojson [(data)]="someValue">` or `<yaga-geojson [data]="someValue">` * @link http://leafletjs.com/reference-1.2.0.html#geojson-l-geojson Original Leaflet documentation */ @Input() public set data(val: GeoJSONFeatureCollection<GeometryObject, T>) { super.clearLayers(); super.addData(val); } public get data(): GeoJSONFeatureCollection<GeometryObject, T> { return (this.toGeoJSON() as GeoJSONFeatureCollection<GeometryObject, T>); } /** * Input for the filter function. * Use it with `<yaga-geojson [filter]="someValue">` * @link http://leafletjs.com/reference-1.2.0.html#geojson-filter Original Leaflet documentation */ @Input() public set filter(filterFn: IGeoJSONFilterFn<T> | undefined) { this.middleware.filter = filterFn; this.redraw(); } public get filter(): IGeoJSONFilterFn<T> | undefined { return this.middleware.filter; } /** * Input for the pointToLayer function. * Use it with `<yaga-geojson [pointToLayer]="someValue">` * @link http://leafletjs.com/reference-1.2.0.html#geojson-pointtolayer Original Leaflet documentation */ @Input() public set pointToLayer(pointToLayerFn: IGeoJSONPointToLayerFn<T> | undefined) { this.middleware.pointToLayer = pointToLayerFn; this.redraw(); } public get pointToLayer(): IGeoJSONPointToLayerFn<T> | undefined { return this.middleware.pointToLayer; } /** * Input for the styler function. * Use it with `<yaga-geojson [styler]="someValue">` * * *Note: The function can follow the `IGeoJSONStylerFn` interface. It enhances the leaflet ones with the default * style as second parameter* * @link http://leafletjs.com/reference-1.2.0.html#geojson-style Original Leaflet documentation */ @Input() public set styler(stylerFn: IGeoJSONStylerFn<T> | undefined) { this.middleware.styler = stylerFn; this.redraw(); } public get styler(): IGeoJSONStylerFn<T> | undefined { return this.middleware.styler; } /** * Input for the default style. * Use it with `<yaga-geojson [defaultStyle]="someValue">` * * *Note: Leaflet does not provide a default style, it just provides a style function!* */ @Input() public set defaultStyle(style: PathOptions) { this.middleware.defaultStyle = style; this.redraw(); } public get defaultStyle(): PathOptions { return this.middleware.defaultStyle!; // There is a fallback default style } /** * Method to apply changes to the geometries */ protected redraw() { if (this.initialized) { this.initialized = false; const data = this.data; super.clearLayers(); super.addData(data); this.initialized = true; } } }