@jmarcelof/leaflet-ng2
Version:
Angular2 module for Leaflet
431 lines (410 loc) • 16.5 kB
text/typescript
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/
*/
export class GeoJSONDirective<T> extends GeoJSON implements OnDestroy, AfterContentInit {
/* tslint:disable:max-line-length */
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
*/
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
*/
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
*/
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
*/
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
*/
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
*/
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
*/
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
*/
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
*/
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
*/
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
*/
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
*/
public contextmenuEvent: EventEmitter<LeafletMouseEvent> = new EventEmitter();
/* tslint:disable:max-line-length */
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(
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
*/
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
*/
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
*/
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
*/
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!*
*/
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;
}
}
}