/**! * The MIT License * * Copyright (c) 2013 the angular-leaflet-directive Team, http://tombatossals.github.io/angular-leaflet-directive * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN * THE SOFTWARE. * * angular-leaflet-directive * https://github.com/tombatossals/angular-leaflet-directive * * @authors https://github.com/tombatossals/angular-leaflet-directive/graphs/contributors */ /*! * angular-leaflet-directive 2015-11-06 * angular-leaflet-directive - An AngularJS directive to easily interact with Leaflet maps * git: https://github.com/tombatossals/angular-leaflet-directive */ (function(angular){ 'use strict'; angular.module('leaflet-directive', []).directive('leaflet', ["$q", "leafletData", "leafletMapDefaults", "leafletHelpers", "leafletMapEvents", function($q, leafletData, leafletMapDefaults, leafletHelpers, leafletMapEvents) { return { restrict: 'EA', replace: true, scope: { center: '=', lfCenter: '=', defaults: '=', maxbounds: '=', bounds: '=', markers: '=', legend: '=', geojson: '=', paths: '=', tiles: '=', layers: '=', controls: '=', decorations: '=', eventBroadcast: '=', markersWatchOptions: '=', geojsonWatchOptions: '=', }, transclude: true, template: '<div class="angular-leaflet-map"><div ng-transclude></div></div>', controller: ["$scope", function($scope) { this._leafletMap = $q.defer(); this.getMap = function() { return this._leafletMap.promise; }; this.getLeafletScope = function() { return $scope; }; }], link: function(scope, element, attrs, ctrl) { var isDefined = leafletHelpers.isDefined; var defaults = leafletMapDefaults.setDefaults(scope.defaults, attrs.id); var mapEvents = leafletMapEvents.getAvailableMapEvents(); var addEvents = leafletMapEvents.addEvents; scope.mapId = attrs.id; leafletData.setDirectiveControls({}, attrs.id); // Set width and height utility functions function updateWidth() { if (isNaN(attrs.width)) { element.css('width', attrs.width); } else { element.css('width', attrs.width + 'px'); } } function updateHeight() { if (isNaN(attrs.height)) { element.css('height', attrs.height); } else { element.css('height', attrs.height + 'px'); } } // If the width attribute defined update css // Then watch if bound property changes and update css if (isDefined(attrs.width)) { updateWidth(); scope.$watch( function() { return element[0].getAttribute('width'); }, function() { updateWidth(); map.invalidateSize(); }); } // If the height attribute defined update css // Then watch if bound property changes and update css if (isDefined(attrs.height)) { updateHeight(); scope.$watch( function() { return element[0].getAttribute('height'); }, function() { updateHeight(); map.invalidateSize(); }); } // Create the Leaflet Map Object with the options var map = new L.Map(element[0], leafletMapDefaults.getMapCreationDefaults(attrs.id)); ctrl._leafletMap.resolve(map); if (!isDefined(attrs.center) && !isDefined(attrs.lfCenter)) { map.setView([defaults.center.lat, defaults.center.lng], defaults.center.zoom); } // If no layers nor tiles defined, set the default tileLayer if (!isDefined(attrs.tiles) && (!isDefined(attrs.layers))) { var tileLayerObj = L.tileLayer(defaults.tileLayer, defaults.tileLayerOptions); tileLayerObj.addTo(map); leafletData.setTiles(tileLayerObj, attrs.id); } // Set zoom control configuration if (isDefined(map.zoomControl) && isDefined(defaults.zoomControlPosition)) { map.zoomControl.setPosition(defaults.zoomControlPosition); } if (isDefined(map.zoomControl) && defaults.zoomControl === false) { map.zoomControl.removeFrom(map); } if (isDefined(map.zoomsliderControl) && isDefined(defaults.zoomsliderControl) && defaults.zoomsliderControl === false) { map.zoomsliderControl.removeFrom(map); } // if no event-broadcast attribute, all events are broadcasted if (!isDefined(attrs.eventBroadcast)) { var logic = 'broadcast'; addEvents(map, mapEvents, 'eventName', scope, logic); } // Resolve the map object to the promises map.whenReady(function() { leafletData.setMap(map, attrs.id); }); scope.$on('$destroy', function() { leafletMapDefaults.reset(); map.remove(); leafletData.unresolveMap(attrs.id); }); //Handle request to invalidate the map size //Up scope using $scope.$emit('invalidateSize') //Down scope using $scope.$broadcast('invalidateSize') scope.$on('invalidateSize', function() { map.invalidateSize(); }); }, }; }]); angular.module('leaflet-directive').factory('leafletBoundsHelpers', ["$log", "leafletHelpers", function($log, leafletHelpers) { var isArray = leafletHelpers.isArray; var isNumber = leafletHelpers.isNumber; var isFunction = leafletHelpers.isFunction; var isDefined = leafletHelpers.isDefined; function _isValidBounds(bounds) { return angular.isDefined(bounds) && angular.isDefined(bounds.southWest) && angular.isDefined(bounds.northEast) && angular.isNumber(bounds.southWest.lat) && angular.isNumber(bounds.southWest.lng) && angular.isNumber(bounds.northEast.lat) && angular.isNumber(bounds.northEast.lng); } return { createLeafletBounds: function(bounds) { if (_isValidBounds(bounds)) { return L.latLngBounds([bounds.southWest.lat, bounds.southWest.lng], [bounds.northEast.lat, bounds.northEast.lng]); } }, isValidBounds: _isValidBounds, createBoundsFromArray: function(boundsArray) { if (!(isArray(boundsArray) && boundsArray.length === 2 && isArray(boundsArray[0]) && isArray(boundsArray[1]) && boundsArray[0].length === 2 && boundsArray[1].length === 2 && isNumber(boundsArray[0][0]) && isNumber(boundsArray[0][1]) && isNumber(boundsArray[1][0]) && isNumber(boundsArray[1][1]))) { $log.error('[AngularJS - Leaflet] The bounds array is not valid.'); return; } return { northEast: { lat: boundsArray[0][0], lng: boundsArray[0][1], }, southWest: { lat: boundsArray[1][0], lng: boundsArray[1][1], }, }; }, createBoundsFromLeaflet: function(lfBounds) { if (!(isDefined(lfBounds) && isFunction(lfBounds.getNorthEast) && isFunction(lfBounds.getSouthWest))) { $log.error('[AngularJS - Leaflet] The leaflet bounds is not valid object.'); return; } var northEast = lfBounds.getNorthEast(); var southWest = lfBounds.getSouthWest(); return { northEast: { lat: northEast.lat, lng: northEast.lng, }, southWest: { lat: southWest.lat, lng: southWest.lng, }, }; }, }; }]); angular.module('leaflet-directive').factory('leafletControlHelpers', ["$rootScope", "$log", "leafletHelpers", "leafletLayerHelpers", "leafletMapDefaults", function($rootScope, $log, leafletHelpers, leafletLayerHelpers, leafletMapDefaults) { var isDefined = leafletHelpers.isDefined; var isObject = leafletHelpers.isObject; var createLayer = leafletLayerHelpers.createLayer; var _controls = {}; var errorHeader = leafletHelpers.errorHeader + ' [Controls] '; var _controlLayersMustBeVisible = function(baselayers, overlays, mapId) { var defaults = leafletMapDefaults.getDefaults(mapId); if (!defaults.controls.layers.visible) { return false; } var atLeastOneControlItemMustBeShown = false; if (isObject(baselayers)) { Object.keys(baselayers).forEach(function(key) { var layer = baselayers[key]; if (!isDefined(layer.layerOptions) || layer.layerOptions.showOnSelector !== false) { atLeastOneControlItemMustBeShown = true; } }); } if (isObject(overlays)) { Object.keys(overlays).forEach(function(key) { var layer = overlays[key]; if (!isDefined(layer.layerParams) || layer.layerParams.showOnSelector !== false) { atLeastOneControlItemMustBeShown = true; } }); } return atLeastOneControlItemMustBeShown; }; var _createLayersControl = function(mapId) { var defaults = leafletMapDefaults.getDefaults(mapId); var controlOptions = { collapsed: defaults.controls.layers.collapsed, position: defaults.controls.layers.position, autoZIndex: false, }; angular.extend(controlOptions, defaults.controls.layers.options); var control; if (defaults.controls.layers && isDefined(defaults.controls.layers.control)) { control = defaults.controls.layers.control.apply(this, [[], [], controlOptions]); } else { control = new L.control.layers([], [], controlOptions); } return control; }; var controlTypes = { draw: { isPluginLoaded: function() { if (!angular.isDefined(L.Control.Draw)) { $log.error(errorHeader + ' Draw plugin is not loaded.'); return false; } return true; }, checkValidParams: function(/* params */) { return true; }, createControl: function(params) { return new L.Control.Draw(params); }, }, scale: { isPluginLoaded: function() { return true; }, checkValidParams: function(/* params */) { return true; }, createControl: function(params) { return new L.control.scale(params); }, }, fullscreen: { isPluginLoaded: function() { if (!angular.isDefined(L.Control.Fullscreen)) { $log.error(errorHeader + ' Fullscreen plugin is not loaded.'); return false; } return true; }, checkValidParams: function(/* params */) { return true; }, createControl: function(params) { return new L.Control.Fullscreen(params); }, }, search: { isPluginLoaded: function() { if (!angular.isDefined(L.Control.Search)) { $log.error(errorHeader + ' Search plugin is not loaded.'); return false; } return true; }, checkValidParams: function(/* params */) { return true; }, createControl: function(params) { return new L.Control.Search(params); }, }, custom: {}, minimap: { isPluginLoaded: function() { if (!angular.isDefined(L.Control.MiniMap)) { $log.error(errorHeader + ' Minimap plugin is not loaded.'); return false; } return true; }, checkValidParams: function(params) { if (!isDefined(params.layer)) { $log.warn(errorHeader + ' minimap "layer" option should be defined.'); return false; } return true; }, createControl: function(params) { var layer = createLayer(params.layer); if (!isDefined(layer)) { $log.warn(errorHeader + ' minimap control "layer" could not be created.'); return; } return new L.Control.MiniMap(layer, params); }, }, }; return { layersControlMustBeVisible: _controlLayersMustBeVisible, isValidControlType: function(type) { return Object.keys(controlTypes).indexOf(type) !== -1; }, createControl: function(type, params) { if (!controlTypes[type].checkValidParams(params)) { return; } return controlTypes[type].createControl(params); }, updateLayersControl: function(map, mapId, loaded, baselayers, overlays, leafletLayers) { var i; var _layersControl = _controls[mapId]; var mustBeLoaded = _controlLayersMustBeVisible(baselayers, overlays, mapId); if (isDefined(_layersControl) && loaded) { for (i in leafletLayers.baselayers) { _layersControl.removeLayer(leafletLayers.baselayers[i]); } for (i in leafletLayers.overlays) { _layersControl.removeLayer(leafletLayers.overlays[i]); } map.removeControl(_layersControl); delete _controls[mapId]; } if (mustBeLoaded) { _layersControl = _createLayersControl(mapId); _controls[mapId] = _layersControl; for (i in baselayers) { var hideOnSelector = isDefined(baselayers[i].layerOptions) && baselayers[i].layerOptions.showOnSelector === false; if (!hideOnSelector && isDefined(leafletLayers.baselayers[i])) { _layersControl.addBaseLayer(leafletLayers.baselayers[i], baselayers[i].name); } } for (i in overlays) { var hideOverlayOnSelector = isDefined(overlays[i].layerParams) && overlays[i].layerParams.showOnSelector === false; if (!hideOverlayOnSelector && isDefined(leafletLayers.overlays[i])) { _layersControl.addOverlay(leafletLayers.overlays[i], overlays[i].name); } } map.addControl(_layersControl); } return mustBeLoaded; }, }; }]); angular.module('leaflet-directive').service('leafletData', ["$log", "$q", "leafletHelpers", function($log, $q, leafletHelpers) { var getDefer = leafletHelpers.getDefer, getUnresolvedDefer = leafletHelpers.getUnresolvedDefer, setResolvedDefer = leafletHelpers.setResolvedDefer; var _private = {}; var self = this; var upperFirst = function(string) { return string.charAt(0).toUpperCase() + string.slice(1); }; var _privateItems = [ 'map', 'tiles', 'layers', 'paths', 'markers', 'geoJSON', 'UTFGrid', //odd ball on naming convention keeping to not break 'decorations', 'directiveControls',]; //init _privateItems.forEach(function(itemName) { _private[itemName] = {}; }); this.unresolveMap = function(scopeId) { var id = leafletHelpers.obtainEffectiveMapId(_private.map, scopeId); _privateItems.forEach(function(itemName) { _private[itemName][id] = undefined; }); }; //int repetitive stuff (get and sets) _privateItems.forEach(function(itemName) { var name = upperFirst(itemName); self['set' + name] = function(lObject, scopeId) { var defer = getUnresolvedDefer(_private[itemName], scopeId); defer.resolve(lObject); setResolvedDefer(_private[itemName], scopeId); }; self['get' + name] = function(scopeId) { var defer = getDefer(_private[itemName], scopeId); return defer.promise; }; }); }]); angular.module('leaflet-directive') .service('leafletDirectiveControlsHelpers', ["$log", "leafletData", "leafletHelpers", function($log, leafletData, leafletHelpers) { var _isDefined = leafletHelpers.isDefined; var _isString = leafletHelpers.isString; var _isObject = leafletHelpers.isObject; var _mainErrorHeader = leafletHelpers.errorHeader; var _errorHeader = _mainErrorHeader + '[leafletDirectiveControlsHelpers'; var _extend = function(id, thingToAddName, createFn, cleanFn) { var _fnHeader = _errorHeader + '.extend] '; var extender = {}; if (!_isDefined(thingToAddName)) { $log.error(_fnHeader + 'thingToAddName cannot be undefined'); return; } if (_isString(thingToAddName) && _isDefined(createFn) && _isDefined(cleanFn)) { extender[thingToAddName] = { create: createFn, clean: cleanFn, }; } else if (_isObject(thingToAddName) && !_isDefined(createFn) && !_isDefined(cleanFn)) { extender = thingToAddName; } else { $log.error(_fnHeader + 'incorrect arguments'); return; } //add external control to create / destroy markers without a watch leafletData.getDirectiveControls().then(function(controls) { angular.extend(controls, extender); leafletData.setDirectiveControls(controls, id); }); }; return { extend: _extend, }; }]); angular.module('leaflet-directive') .service('leafletGeoJsonHelpers', ["leafletHelpers", "leafletIterators", function(leafletHelpers, leafletIterators) { var lHlp = leafletHelpers; var lIt = leafletIterators; var Point = function(lat, lng) { this.lat = lat; this.lng = lng; return this; }; var _getLat = function(value) { if (Array.isArray(value) && value.length === 2) { return value[1]; } else if (lHlp.isDefined(value.type) && value.type === 'Point') { return +value.coordinates[1]; } else { return +value.lat; } }; var _getLng = function(value) { if (Array.isArray(value) && value.length === 2) { return value[0]; } else if (lHlp.isDefined(value.type) && value.type === 'Point') { return +value.coordinates[0]; } else { return +value.lng; } }; var _validateCoords = function(coords) { if (lHlp.isUndefined(coords)) { return false; } if (lHlp.isArray(coords)) { if (coords.length === 2 && lHlp.isNumber(coords[0]) && lHlp.isNumber(coords[1])) { return true; } } else if (lHlp.isDefined(coords.type)) { if ( coords.type === 'Point' && lHlp.isArray(coords.coordinates) && coords.coordinates.length === 2 && lHlp.isNumber(coords.coordinates[0]) && lHlp.isNumber(coords.coordinates[1])) { return true; } } var ret = lIt.all(['lat', 'lng'], function(pos) { return lHlp.isDefined(coords[pos]) && lHlp.isNumber(coords[pos]); }); return ret; }; var _getCoords = function(value) { if (!value || !_validateCoords(value)) { return; } var p = null; if (Array.isArray(value) && value.length === 2) { p = new Point(value[1], value[0]); } else if (lHlp.isDefined(value.type) && value.type === 'Point') { p = new Point(value.coordinates[1], value.coordinates[0]); } else { return value; } //note angular.merge is avail in angular 1.4.X we might want to fill it here return angular.extend(value, p);//tap on lat, lng if it doesnt exist }; return { getLat: _getLat, getLng: _getLng, validateCoords: _validateCoords, getCoords: _getCoords, }; }]); angular.module('leaflet-directive').service('leafletHelpers', ["$q", "$log", function($q, $log) { var _errorHeader = '[AngularJS - Leaflet] '; var _copy = angular.copy; var _clone = _copy; /* For parsing paths to a field in an object Example: var obj = { bike:{ 1: 'hi' 2: 'foo' } }; _getObjectValue(obj,"bike.1") returns 'hi' this is getPath in ui-gmap */ var _getObjectValue = function(object, pathStr) { var obj; if (!object || !angular.isObject(object)) return; //if the key is not a sting then we already have the value if ((pathStr === null) || !angular.isString(pathStr)) { return pathStr; } obj = object; pathStr.split('.').forEach(function(value) { if (obj) { obj = obj[value]; } }); return obj; }; /* Object Array Notation _getObjectArrayPath("bike.one.two") returns: 'bike["one"]["two"]' */ var _getObjectArrayPath = function(pathStr) { return pathStr.split('.').reduce(function(previous, current) { return previous + '["' + current + '"]'; }); }; /* Object Dot Notation _getObjectPath(["bike","one","two"]) returns: "bike.one.two" */ var _getObjectDotPath = function(arrayOfStrings) { return arrayOfStrings.reduce(function(previous, current) { return previous + '.' + current; }); }; function _obtainEffectiveMapId(d, mapId) { var id; var i; if (!angular.isDefined(mapId)) { if (Object.keys(d).length === 0) { id = 'main'; } else if (Object.keys(d).length >= 1) { for (i in d) { if (d.hasOwnProperty(i)) { id = i; } } } else { $log.error(_errorHeader + '- You have more than 1 map on the DOM, you must provide the map ID to the leafletData.getXXX call'); } } else { id = mapId; } return id; } function _getUnresolvedDefer(d, mapId) { var id = _obtainEffectiveMapId(d, mapId); var defer; if (!angular.isDefined(d[id]) || d[id].resolvedDefer === true) { defer = $q.defer(); d[id] = { defer: defer, resolvedDefer: false, }; } else { defer = d[id].defer; } return defer; } var _isDefined = function(value) { return angular.isDefined(value) && value !== null; }; var _isUndefined = function(value) { return !_isDefined(value); }; // BEGIN DIRECT PORT FROM AngularJS code base var SPECIAL_CHARS_REGEXP = /([\:\-\_]+(.))/g; var MOZ_HACK_REGEXP = /^moz([A-Z])/; var PREFIX_REGEXP = /^((?:x|data)[\:\-_])/i; /** Converts snake_case to camelCase. Also there is special case for Moz prefix starting with upper case letter. @param name Name to normalize */ var camelCase = function(name) { return name.replace(SPECIAL_CHARS_REGEXP, function(_, separator, letter, offset) { if (offset) { return letter.toUpperCase(); } else { return letter; } }).replace(MOZ_HACK_REGEXP, 'Moz$1'); }; /** Converts all accepted directives format into proper directive name. @param name Name to normalize */ var directiveNormalize = function(name) { return camelCase(name.replace(PREFIX_REGEXP, '')); }; // END AngularJS port return { camelCase: camelCase, directiveNormalize: directiveNormalize, copy:_copy, clone:_clone, errorHeader: _errorHeader, getObjectValue: _getObjectValue, getObjectArrayPath:_getObjectArrayPath, getObjectDotPath: _getObjectDotPath, defaultTo: function(val, _default) { return _isDefined(val) ? val : _default; }, //mainly for checking attributes of directives lets keep this minimal (on what we accept) isTruthy: function(val) { return val === 'true' || val === true; }, //Determine if a reference is {} isEmpty: function(value) { return Object.keys(value).length === 0; }, //Determine if a reference is undefined or {} isUndefinedOrEmpty: function(value) { return (angular.isUndefined(value) || value === null) || Object.keys(value).length === 0; }, // Determine if a reference is defined isDefined: _isDefined, isUndefined:_isUndefined, isNumber: angular.isNumber, isString: angular.isString, isArray: angular.isArray, isObject: angular.isObject, isFunction: angular.isFunction, equals: angular.equals, isValidCenter: function(center) { return angular.isDefined(center) && angular.isNumber(center.lat) && angular.isNumber(center.lng) && angular.isNumber(center.zoom); }, isValidPoint: function(point) { if (!angular.isDefined(point)) { return false; } if (angular.isArray(point)) { return point.length === 2 && angular.isNumber(point[0]) && angular.isNumber(point[1]); } return angular.isNumber(point.lat) && angular.isNumber(point.lng); }, isSameCenterOnMap: function(centerModel, map) { var mapCenter = map.getCenter(); var zoom = map.getZoom(); if (centerModel.lat && centerModel.lng && mapCenter.lat.toFixed(4) === centerModel.lat.toFixed(4) && mapCenter.lng.toFixed(4) === centerModel.lng.toFixed(4) && zoom === centerModel.zoom) { return true; } return false; }, safeApply: function($scope, fn) { var phase = $scope.$root.$$phase; if (phase === '$apply' || phase === '$digest') { $scope.$eval(fn); } else { $scope.$evalAsync(fn); } }, obtainEffectiveMapId: _obtainEffectiveMapId, getDefer: function(d, mapId) { var id = _obtainEffectiveMapId(d, mapId); var defer; if (!angular.isDefined(d[id]) || d[id].resolvedDefer === false) { defer = _getUnresolvedDefer(d, mapId); } else { defer = d[id].defer; } return defer; }, getUnresolvedDefer: _getUnresolvedDefer, setResolvedDefer: function(d, mapId) { var id = _obtainEffectiveMapId(d, mapId); d[id].resolvedDefer = true; }, rangeIsSupported: function() { var testrange = document.createElement('input'); testrange.setAttribute('type', 'range'); return testrange.type === 'range'; }, FullScreenControlPlugin: { isLoaded: function() { return angular.isDefined(L.Control.Fullscreen); }, }, MiniMapControlPlugin: { isLoaded: function() { return angular.isDefined(L.Control.MiniMap); }, }, AwesomeMarkersPlugin: { isLoaded: function() { return angular.isDefined(L.AwesomeMarkers) && angular.isDefined(L.AwesomeMarkers.Icon); }, is: function(icon) { if (this.isLoaded()) { return icon instanceof L.AwesomeMarkers.Icon; } else { return false; } }, equal: function(iconA, iconB) { if (!this.isLoaded()) { return false; } if (this.is(iconA)) { return angular.equals(iconA, iconB); } else { return false; } }, }, VectorMarkersPlugin: { isLoaded: function() { return angular.isDefined(L.VectorMarkers) && angular.isDefined(L.VectorMarkers.Icon); }, is: function(icon) { if (this.isLoaded()) { return icon instanceof L.VectorMarkers.Icon; } else { return false; } }, equal: function(iconA, iconB) { if (!this.isLoaded()) { return false; } if (this.is(iconA)) { return angular.equals(iconA, iconB); } else { return false; } }, }, DomMarkersPlugin: { isLoaded: function() { if (angular.isDefined(L.DomMarkers) && angular.isDefined(L.DomMarkers.Icon)) { return true; } else { return false; } }, is: function(icon) { if (this.isLoaded()) { return icon instanceof L.DomMarkers.Icon; } else { return false; } }, equal: function(iconA, iconB) { if (!this.isLoaded()) { return false; } if (this.is(iconA)) { return angular.equals(iconA, iconB); } else { return false; } }, }, PolylineDecoratorPlugin: { isLoaded: function() { if (angular.isDefined(L.PolylineDecorator)) { return true; } else { return false; } }, is: function(decoration) { if (this.isLoaded()) { return decoration instanceof L.PolylineDecorator; } else { return false; } }, equal: function(decorationA, decorationB) { if (!this.isLoaded()) { return false; } if (this.is(decorationA)) { return angular.equals(decorationA, decorationB); } else { return false; } }, }, MakiMarkersPlugin: { isLoaded: function() { if (angular.isDefined(L.MakiMarkers) && angular.isDefined(L.MakiMarkers.Icon)) { return true; } else { return false; } }, is: function(icon) { if (this.isLoaded()) { return icon instanceof L.MakiMarkers.Icon; } else { return false; } }, equal: function(iconA, iconB) { if (!this.isLoaded()) { return false; } if (this.is(iconA)) { return angular.equals(iconA, iconB); } else { return false; } }, }, ExtraMarkersPlugin: { isLoaded: function() { if (angular.isDefined(L.ExtraMarkers) && angular.isDefined(L.ExtraMarkers.Icon)) { return true; } else { return false; } }, is: function(icon) { if (this.isLoaded()) { return icon instanceof L.ExtraMarkers.Icon; } else { return false; } }, equal: function(iconA, iconB) { if (!this.isLoaded()) { return false; } if (this.is(iconA)) { return angular.equals(iconA, iconB); } else { return false; } }, }, LabelPlugin: { isLoaded: function() { return angular.isDefined(L.Label); }, is: function(layer) { if (this.isLoaded()) { return layer instanceof L.MarkerClusterGroup; } else { return false; } }, }, MarkerClusterPlugin: { isLoaded: function() { return angular.isDefined(L.MarkerClusterGroup); }, is: function(layer) { if (this.isLoaded()) { return layer instanceof L.MarkerClusterGroup; } else { return false; } }, }, GoogleLayerPlugin: { isLoaded: function() { return angular.isDefined(L.Google); }, is: function(layer) { if (this.isLoaded()) { return layer instanceof L.Google; } else { return false; } }, }, LeafletProviderPlugin: { isLoaded: function() { return angular.isDefined(L.TileLayer.Provider); }, is: function(layer) { if (this.isLoaded()) { return layer instanceof L.TileLayer.Provider; } else { return false; } }, }, ChinaLayerPlugin: { isLoaded: function() { return angular.isDefined(L.tileLayer.chinaProvider); }, }, HeatLayerPlugin: { isLoaded: function() { return angular.isDefined(L.heatLayer); }, }, WebGLHeatMapLayerPlugin: { isLoaded: function() { return angular.isDefined(L.TileLayer.WebGLHeatMap); }, }, BingLayerPlugin: { isLoaded: function() { return angular.isDefined(L.BingLayer); }, is: function(layer) { if (this.isLoaded()) { return layer instanceof L.BingLayer; } else { return false; } }, }, WFSLayerPlugin: { isLoaded: function() { return L.GeoJSON.WFS !== undefined; }, is: function(layer) { if (this.isLoaded()) { return layer instanceof L.GeoJSON.WFS; } else { return false; } }, }, AGSBaseLayerPlugin: { isLoaded: function() { return L.esri !== undefined && L.esri.basemapLayer !== undefined; }, is: function(layer) { if (this.isLoaded()) { return layer instanceof L.esri.basemapLayer; } else { return false; } }, }, AGSLayerPlugin: { isLoaded: function() { return lvector !== undefined && lvector.AGS !== undefined; }, is: function(layer) { if (this.isLoaded()) { return layer instanceof lvector.AGS; } else { return false; } }, }, AGSFeatureLayerPlugin: { isLoaded: function() { return L.esri !== undefined && L.esri.featureLayer !== undefined; }, is: function(layer) { if (this.isLoaded()) { return layer instanceof L.esri.featureLayer; } else { return false; } }, }, AGSTiledMapLayerPlugin: { isLoaded: function() { return L.esri !== undefined && L.esri.tiledMapLayer !== undefined; }, is: function(layer) { if (this.isLoaded()) { return layer instanceof L.esri.tiledMapLayer; } else { return false; } }, }, AGSDynamicMapLayerPlugin: { isLoaded: function() { return L.esri !== undefined && L.esri.dynamicMapLayer !== undefined; }, is: function(layer) { if (this.isLoaded()) { return layer instanceof L.esri.dynamicMapLayer; } else { return false; } }, }, AGSImageMapLayerPlugin: { isLoaded: function() { return L.esri !== undefined && L.esri.imageMapLayer !== undefined; }, is: function(layer) { if (this.isLoaded()) { return layer instanceof L.esri.imageMapLayer; } else { return false; } }, }, AGSClusteredLayerPlugin: { isLoaded: function() { return L.esri !== undefined && L.esri.clusteredFeatureLayer !== undefined; }, is: function(layer) { if (this.isLoaded()) { return layer instanceof L.esri.clusteredFeatureLayer; } else { return false; } }, }, AGSHeatmapLayerPlugin: { isLoaded: function() { return L.esri !== undefined && L.esri.heatmapFeatureLayer !== undefined; }, is: function(layer) { if (this.isLoaded()) { return layer instanceof L.esri.heatmapFeatureLayer; } else { return false; } }, }, YandexLayerPlugin: { isLoaded: function() { return angular.isDefined(L.Yandex); }, is: function(layer) { if (this.isLoaded()) { return layer instanceof L.Yandex; } else { return false; } }, }, GeoJSONPlugin: { isLoaded: function() { return angular.isDefined(L.TileLayer.GeoJSON); }, is: function(layer) { if (this.isLoaded()) { return layer instanceof L.TileLayer.GeoJSON; } else { return false; } }, }, UTFGridPlugin: { isLoaded: function() { return angular.isDefined(L.UtfGrid); }, is: function(layer) { if (this.isLoaded()) { return layer instanceof L.UtfGrid; } else { $log.error('[AngularJS - Leaflet] No UtfGrid plugin found.'); return false; } }, }, CartoDB: { isLoaded: function() { return cartodb; }, is: function(/*layer*/) { return true; /* if (this.isLoaded()) { return layer instanceof L.TileLayer.GeoJSON; } else { return false; }*/ }, }, Leaflet: { DivIcon: { is: function(icon) { return icon instanceof L.DivIcon; }, equal: function(iconA, iconB) { if (this.is(iconA)) { return angular.equals(iconA, iconB); } else { return false; } }, }, Icon: { is: function(icon) { return icon instanceof L.Icon; }, equal: function(iconA, iconB) { if (this.is(iconA)) { return angular.equals(iconA, iconB); } else { return false; } }, }, }, /* watchOptions - object to set deep nested watches and turn off watches all together (rely on control / functional updates) watchOptions - Object doWatch:boolean isDeep:boolean (sets $watch(function,isDeep)) individual doWatch:boolean isDeep:boolean */ //legacy defaults watchOptions: { doWatch:true, isDeep: true, individual:{ doWatch:true, isDeep: true, }, }, }; }]); angular.module('leaflet-directive').service('leafletIterators', ["$log", "leafletHelpers", function($log, leafletHelpers) { var lHlp = leafletHelpers; var errorHeader = leafletHelpers.errorHeader + 'leafletIterators: '; //BEGIN COPY from underscore var _keys = Object.keys; var _isFunction = lHlp.isFunction; var _isObject = lHlp.isObject; // Helper for collection methods to determine whether a collection // should be iterated as an array or as an object // Related: http://people.mozilla.org/~jorendorff/es6-draft.html#sec-tolength var MAX_ARRAY_INDEX = Math.pow(2, 53) - 1; var _isArrayLike = function(collection) { var length = collection !== null && collection.length; return lHlp.isNumber(length) && length >= 0 && length <= MAX_ARRAY_INDEX; }; // Keep the identity function around for default iteratees. var _identity = function(value) { return value; }; var _property = function(key) { return function(obj) { return obj === null ? void 0 : obj[key]; }; }; // Internal function that returns an efficient (for current engines) version // of the passed-in callback, to be repeatedly applied in other Underscore // functions. var optimizeCb = function(func, context, argCount) { if (context === void 0) return func; switch (argCount === null ? 3 : argCount) { case 1: return function(value) { return func.call(context, value); }; case 2: return function(value, other) { return func.call(context, value, other); }; case 3: return function(value, index, collection) { return func.call(context, value, index, collection); }; case 4: return function(accumulator, value, index, collection) { return func.call(context, accumulator, value, index, collection); }; } return function() { return func.apply(context, arguments); }; }; // An internal function for creating assigner functions. var createAssigner = function(keysFunc, undefinedOnly) { return function(obj) { var length = arguments.length; if (length < 2 || obj === null) return obj; for (var index = 1; index < length; index++) { var source = arguments[index]; var keys = keysFunc(source); var l = keys.length; for (var i = 0; i < l; i++) { var key = keys[i]; if (!undefinedOnly || obj[key] === void 0) obj[key] = source[key]; } } return obj; }; }; // Assigns a given object with all the own properties in the passed-in object(s) // (https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Object/assign) var _extendOwn; var _assign = null; _extendOwn = _assign = createAssigner(_keys); // Returns whether an object has a given set of `key:value` pairs. var _isMatch = function(object, attrs) { var keys = _keys(attrs); var length = keys.length; if (object === null) return !length; var obj = Object(object); for (var i = 0; i < length; i++) { var key = keys[i]; if (attrs[key] !== obj[key] || !(key in obj)) return false; } return true; }; // Returns a predicate for checking whether an object has a given set of // `key:value` pairs. var _matcher; var _matches = null; _matcher = _matches = function(attrs) { attrs = _extendOwn({}, attrs); return function(obj) { return _isMatch(obj, attrs); }; }; // A mostly-internal function to generate callbacks that can be applied // to each element in a collection, returning the desired result — either // identity, an arbitrary callback, a property matcher, or a property accessor. var cb = function(value, context, argCount) { if (value === null) return _identity; if (_isFunction(value)) return optimizeCb(value, context, argCount); if (_isObject(value)) return _matcher(value); return _property(value); }; var _every; var _all = null; _every = _all = function(obj, predicate, context) { predicate = cb(predicate, context); var keys = !_isArrayLike(obj) && _keys(obj); var length = (keys || obj).length; for (var index = 0; index < length; index++) { var currentKey = keys ? keys[index] : index; if (!predicate(obj[currentKey], currentKey, obj)) return false; } return true; }; //END COPY fron underscore var _hasErrors = function(collection, cb, ignoreCollection, cbName) { if (!ignoreCollection) { if (!lHlp.isDefined(collection) || !lHlp.isDefined(cb)) { return true; } } if (!lHlp.isFunction(cb)) { cbName = lHlp.defaultTo(cb, 'cb'); $log.error(errorHeader + cbName + ' is not a function'); return true; } return false; }; var _iterate = function(collection, externalCb, internalCb) { if (_hasErrors(undefined, internalCb, true, 'internalCb')) { return; } if (!_hasErrors(collection, externalCb)) { for (var key in collection) { if (collection.hasOwnProperty(key)) { internalCb(collection[key], key); } } } }; //see http://jsperf.com/iterators/3 //utilizing for in is way faster var _each = function(collection, cb) { _iterate(collection, cb, function(val, key) { cb(val, key); }); }; return { each:_each, forEach: _each, every: _every, all: _all, }; }]); angular.module('leaflet-directive') .factory('leafletLayerHelpers', ["$rootScope", "$log", "$q", "leafletHelpers", "leafletIterators", function($rootScope, $log, $q, leafletHelpers, leafletIterators) { var Helpers = leafletHelpers; var isString = leafletHelpers.isString; var isObject = leafletHelpers.isObject; var isArray = leafletHelpers.isArray; var isDefined = leafletHelpers.isDefined; var errorHeader = leafletHelpers.errorHeader; var $it = leafletIterators; var utfGridCreateLayer = function(params) { if (!Helpers.UTFGridPlugin.isLoaded()) { $log.error('[AngularJS - Leaflet] The UTFGrid plugin is not loaded.'); return; } var utfgrid = new L.UtfGrid(params.url, params.pluginOptions); utfgrid.on('mouseover', function(e) { $rootScope.$broadcast('leafletDirectiveMap.utfgridMouseover', e); }); utfgrid.on('mouseout', function(e) { $rootScope.$broadcast('leafletDirectiveMap.utfgridMouseout', e); }); utfgrid.on('click', function(e) { $rootScope.$broadcast('leafletDirectiveMap.utfgridClick', e); }); utfgrid.on('mousemove', function(e) { $rootScope.$broadcast('leafletDirectiveMap.utfgridMousemove', e); }); return utfgrid; }; var layerTypes = { xyz: { mustHaveUrl: true, createLayer: function(params) { return L.tileLayer(params.url, params.options); }, }, mapbox: { mustHaveKey: true, createLayer: function(params) { var version = 3; if (isDefined(params.options.version) && params.options.version === 4) { version = params.options.version; } var url = version === 3 ? '//{s}.tiles.mapbox.com/v3/' + params.key + '/{z}/{x}/{y}.png' : '//api.tiles.mapbox.com/v4/' + params.key + '/{z}/{x}/{y}.png?access_token=' + params.apiKey; return L.tileLayer(url, params.options); }, }, geoJSON: { mustHaveUrl: true, createLayer: function(params) { if (!Helpers.GeoJSONPlugin.isLoaded()) { return; } return new L.TileLayer.GeoJSON(params.url, params.pluginOptions, params.options); }, }, geoJSONShape: { mustHaveUrl: false, createLayer: function(params) { return new L.GeoJSON(params.data, params.options); }, }, geoJSONAwesomeMarker: { mustHaveUrl: false, createLayer: function(params) { return new L.geoJson(params.data, { pointToLayer: function(feature, latlng) { return L.marker(latlng, {icon: L.AwesomeMarkers.icon(params.icon)}); }, }); }, }, geoJSONVectorMarker: { mustHaveUrl: false, createLayer: function(params) { return new L.geoJson(params.data, { pointToLayer: function(feature, latlng) { return L.marker(latlng, {icon: L.VectorMarkers.icon(params.icon)}); }, }); }, }, utfGrid: { mustHaveUrl: true, createLayer: utfGridCreateLayer, }, cartodbTiles: { mustHaveKey: true, createLayer: function(params) { var url = '//' + params.user + '.cartodb.com/api/v1/map/' + params.key + '/{z}/{x}/{y}.png'; return L.tileLayer(url, params.options); }, }, cartodbUTFGrid: { mustHaveKey: true, mustHaveLayer: true, createLayer: function(params) { params.url = '//' + params.user + '.cartodb.com/api/v1/map/' + params.key + '/' + params.layer + '/{z}/{x}/{y}.grid.json'; return utfGridCreateLayer(params); }, }, cartodbInteractive: { mustHaveKey: true, mustHaveLayer: true, createLayer: function(params) { var tilesURL = '//' + params.user + '.cartodb.com/api/v1/map/' + params.key + '/{z}/{x}/{y}.png'; var tileLayer = L.tileLayer(tilesURL, params.options); params.url = '//' + params.user + '.cartodb.com/api/v1/map/' + params.key + '/' + params.layer + '/{z}/{x}/{y}.grid.json'; var utfLayer = utfGridCreateLayer(params); return L.layerGroup([tileLayer, utfLayer]); }, }, wms: { mustHaveUrl: true, createLayer: function(params) { return L.tileLayer.wms(params.url, params.options); }, }, wmts: { mustHaveUrl: true, createLayer: function(params) { return L.tileLayer.wmts(params.url, params.options); }, }, wfs: { mustHaveUrl: true, mustHaveLayer: true, createLayer: function(params) { if (!Helpers.WFSLayerPlugin.isLoaded()) { return; } var options = angular.copy(params.options); if (options.crs && typeof options.crs === 'string') { /*jshint -W061 */ options.crs = eval(options.crs); } return new L.GeoJSON.WFS(params.url, params.layer, options); }, }, group: { mustHaveUrl: false, createLayer: function(params) { var lyrs = []; $it.each(params.options.layers, function(l) { lyrs.push(createLayer(l)); }); params.options.loadedDefer = function() { var defers = []; if (isDefined(params.options.layers)) { for (var i = 0; i < params.options.layers.length; i++) { var d = params.options.layers[i].layerOptions.loadedDefer; if (isDefined(d)) { defers.push(d); } } } return defers; }; return L.layerGroup(lyrs); }, }, featureGroup: { mustHaveUrl: false, createLayer: function() { return L.featureGroup(); }, }, google: { mustHaveUrl: false, createLayer: function(params) { var type = params.type || 'SATELLITE'; if (!Helpers.GoogleLayerPlugin.isLoaded()) { return; } return new L.Google(type, params.options); }, }, here: { mustHaveUrl: false, createLayer: function(params) { var provider = params.provider || 'HERE.terrainDay'; if (!Helpers.LeafletProviderPlugin.isLoaded()) { return; } return new L.TileLayer.Provider(provider, params.options); }, }, china:{ mustHaveUrl:false, createLayer:function(params) { var type = params.type || ''; if (!Helpers.ChinaLayerPlugin.isLoaded()) { return; } return L.tileLayer.chinaProvider(type, params.options); }, }, agsBase: { mustHaveLayer: true, createLayer: function(params) { if (!Helpers.AGSBaseLayerPlugin.isLoaded()) { return; } return L.esri.basemapLayer(params.layer, params.options); }, }, ags: { mustHaveUrl: true, createLayer: function(params) { if (!Helpers.AGSLayerPlugin.isLoaded()) { return; } var options = angular.copy(params.options); angular.extend(options, { url: params.url, }); var layer = new lvector.AGS(options); layer.onAdd = function(map) { this.setMap(map); }; layer.onRemove = function() { this.setMap(null); }; return layer; }, }, agsFeature: { mustHaveUrl: true, createLayer: function(params) { if (!Helpers.AGSFeatureLayerPlugin.isLoaded()) { $log.warn(errorHeader + ' The esri plugin is not loaded.'); return; } params.options.url = params.url; var layer = L.esri.featureLayer(params.options); var load = function() { if (isDefined(params.options.loadedDefer)) { params.options.loadedDefer.resolve(); } }; layer.on('loading', function() { params.options.loadedDefer = $q.defer(); layer.off('load', load); layer.on('load', load); }); return layer; }, }, agsTiled: { mustHaveUrl: true, createLayer: function(params) { if (!Helpers.AGSTiledMapLayerPlugin.isLoaded()) { $log.warn(errorHeader + ' The esri plugin is not loaded.'); return; } params.options.url = params.url; return L.esri.tiledMapLayer(params.options); }, }, agsDynamic: { mustHaveUrl: true, createLayer: function(params) { if (!Helpers.AGSDynamicMapLayerPlugin.isLoaded()) { $log.warn(errorHeader + ' The esri plugin is not loaded.'); return; } params.options.url = params.url; return L.esri.dynamicMapLayer(params.options); }, }, agsImage: { mustHaveUrl: true, createLayer: function(params) { if (!Helpers.AGSImageMapLayerPlugin.isLoaded()) { $log.warn(errorHeader + ' The esri plugin is not loaded.'); return; } params.options.url = params.url; return L.esri.imageMapLayer(params.options); }, }, agsClustered: { mustHaveUrl: true, createLayer: function(params) { if (!Helpers.AGSClusteredLayerPlugin.isLoaded()) { $log.warn(errorHeader + ' The esri clustered layer plugin is not loaded.'); return; } if (!Helpers.MarkerClusterPlugin.isLoaded()) { $log.warn(errorHeader + ' The markercluster plugin is not loaded.'); return; } return L.esri.clusteredFeatureLayer(params.url, params.options); }, }, agsHeatmap: { mustHaveUrl: true, createLayer: function(params) { if (!Helpers.AGSHeatmapLayerPlugin.isLoaded()) { $log.warn(errorHeader + ' The esri heatmap layer plugin is not loaded.'); return; } if (!Helpers.HeatLayerPlugin.isLoaded()) { $log.warn(errorHeader + ' The heatlayer plugin is not loaded.'); return; } return L.esri.heatmapFeatureLayer(params.url, params.options); }, }, markercluster: { mustHaveUrl: false, createLayer: function(params) { if (!Helpers.MarkerClusterPlugin.isLoaded()) { $log.warn(errorHeader + ' The markercluster plugin is not loaded.'); return; } return new L.MarkerClusterGroup(params.options); }, }, bing: { mustHaveUrl: false, createLayer: function(params) { if (!Helpers.BingLayerPlugin.isLoaded()) { return; } return new L.BingLayer(params.key, params.options); }, }, webGLHeatmap: { mustHaveUrl: false, mustHaveData: true, createLayer: function(params) { if (!Helpers.WebGLHeatMapLayerPlugin.isLoaded()) { return; } var layer = new L.TileLayer.WebGLHeatMap(params.options); if (isDefined(params.data)) { layer.setData(params.data); } return layer; }, }, heat: { mustHaveUrl: false, mustHaveData: true, createLayer: function(params) { if (!Helpers.HeatLayerPlugin.isLoaded()) { return; } var layer = new L.heatLayer(); if (isArray(params.data)) { layer.setLatLngs(params.data); } if (isObject(params.options)) { layer.setOptions(params.options); } return layer; }, }, yandex: { mustHaveUrl: false, createLayer: function(params) { var type = params.type || 'map'; if (!Helpers.YandexLayerPlugin.isLoaded()) { return; } return new L.Yandex(type, params.options); }, }, imageOverlay: { mustHaveUrl: true, mustHaveBounds: true, createLayer: function(params) { return L.imageOverlay(params.url, params.bounds, params.options); }, }, iip: { mustHaveUrl: true, createLayer: function(params) { return L.tileLayer.iip(params.url, params.options); }, }, // This "custom" type is used to accept every layer that user want to define himself. // We can wrap these custom layers like heatmap or yandex, but it means a lot of work/code to wrap the world, // so we let user to define their own layer outside the directive, // and pass it on "createLayer" result for next processes custom: { createLayer: function(params) { if (params.layer instanceof L.Class) { return angular.copy(params.layer); } else { $log.error('[AngularJS - Leaflet] A custom layer must be a leaflet Class'); } }, }, cartodb: { mustHaveUrl: true, createLayer: function(params) { return cartodb.createLayer(params.map, params.url); }, }, }; function isValidLayerType(layerDefinition) { // Check if the baselayer has a valid type if (!isString(layerDefinition.type)) { $log.error('[AngularJS - Leaflet] A layer must have a valid type defined.'); return false; } if (Object.keys(layerTypes).indexOf(layerDefinition.type) === -1) { $log.error('[AngularJS - Leaflet] A layer must have a valid type: ' + Object.keys(layerTypes)); return false; } // Check if the layer must have an URL if (layerTypes[layerDefinition.type].mustHaveUrl && !isString(layerDefinition.url)) { $log.error('[AngularJS - Leaflet] A base layer must have an url'); return false; } if (layerTypes[layerDefinition.type].mustHaveData && !isDefined(layerDefinition.data)) { $log.error('[AngularJS - Leaflet] The base layer must have a "data" array attribute'); return false; } if (layerTypes[layerDefinition.type].mustHaveLayer && !isDefined(layerDefinition.layer)) { $log.error('[AngularJS - Leaflet] The type of layer ' + layerDefinition.type + ' must have an layer defined'); return false; } if (layerTypes[layerDefinition.type].mustHaveBounds && !isDefined(layerDefinition.bounds)) { $log.error('[AngularJS - Leaflet] The type of layer ' + layerDefinition.type + ' must have bounds defined'); return false; } if (layerTypes[layerDefinition.type].mustHaveKey && !isDefined(layerDefinition.key)) { $log.error('[AngularJS - Leaflet] The type of layer ' + layerDefinition.type + ' must have key defined'); return false; } return true; } function createLayer(layerDefinition) { if (!isValidLayerType(layerDefinition)) { return; } if (!isString(layerDefinition.name)) { $log.error('[AngularJS - Leaflet] A base layer must have a name'); return; } if (!isObject(layerDefinition.layerParams)) { layerDefinition.layerParams = {}; } if (!isObject(layerDefinition.layerOptions)) { layerDefinition.layerOptions = {}; } // Mix the layer specific parameters with the general Leaflet options. Although this is an overhead // the definition of a base layers is more 'clean' if the two types of parameters are differentiated for (var attrname in layerDefinition.layerParams) { layerDefinition.layerOptions[attrname] = layerDefinition.layerParams[attrname]; } var params = { url: layerDefinition.url, data: layerDefinition.data, options: layerDefinition.layerOptions, layer: layerDefinition.layer, icon: layerDefinition.icon, type: layerDefinition.layerType, bounds: layerDefinition.bounds, key: layerDefinition.key, apiKey: layerDefinition.apiKey, pluginOptions: layerDefinition.pluginOptions, user: layerDefinition.user, }; //TODO Add $watch to the layer properties return layerTypes[layerDefinition.type].createLayer(params); } function safeAddLayer(map, layer) { if (layer && typeof layer.addTo === 'function') { layer.addTo(map); } else { map.addLayer(layer); } } function safeRemoveLayer(map, layer, layerOptions) { if (isDefined(layerOptions) && isDefined(layerOptions.loadedDefer)) { if (angular.isFunction(layerOptions.loadedDefer)) { var defers = layerOptions.loadedDefer(); $log.debug('Loaded Deferred', defers); var count = defers.length; if (count > 0) { var resolve = function() { count--; if (count === 0) { map.removeLayer(layer); } }; for (var i = 0; i < defers.length; i++) { defers[i].promise.then(resolve); } } else { map.removeLayer(layer); } } else { layerOptions.loadedDefer.promise.then(function() { map.removeLayer(layer); }); } } else { map.removeLayer(layer); } } return { createLayer: createLayer, safeAddLayer: safeAddLayer, safeRemoveLayer: safeRemoveLayer, }; }]); angular.module('leaflet-directive').factory('leafletLegendHelpers', function() { var _updateLegend = function(div, legendData, type, url) { div.innerHTML = ''; if (legendData.error) { div.innerHTML += '<div class="info-title alert alert-danger">' + legendData.error.message + '</div>'; } else { if (type === 'arcgis') { for (var i = 0; i < legendData.layers.length; i++) { var layer = legendData.layers[i]; div.innerHTML += '<div class="info-title" data-layerid="' + layer.layerId + '">' + layer.layerName + '</div>'; for (var j = 0; j < layer.legend.length; j++) { var leg = layer.legend[j]; div.innerHTML += '<div class="inline" data-layerid="' + layer.layerId + '"><img src="data:' + leg.contentType + ';base64,' + leg.imageData + '" /></div>' + '<div class="info-label" data-layerid="' + layer.layerId + '">' + leg.label + '</div>'; } } } else if (type === 'image') { div.innerHTML = '<img src="' + url + '"/>'; } } }; var _getOnAddLegend = function(legendData, legendClass, type, url) { return function(/*map*/) { var div = L.DomUtil.create('div', legendClass); if (!L.Browser.touch) { L.DomEvent.disableClickPropagation(div); L.DomEvent.on(div, 'mousewheel', L.DomEvent.stopPropagation); } else { L.DomEvent.on(div, 'click', L.DomEvent.stopPropagation); } _updateLegend(div, legendData, type, url); return div; }; }; var _getOnAddArrayLegend = function(legend, legendClass) { return function(/*map*/) { var div = L.DomUtil.create('div', legendClass); for (var i = 0; i < legend.colors.length; i++) { div.innerHTML += '<div class="outline"><i style="background:' + legend.colors[i] + '"></i></div>' + '<div class="info-label">' + legend.labels[i] + '</div>'; } if (!L.Browser.touch) { L.DomEvent.disableClickPropagation(div); L.DomEvent.on(div, 'mousewheel', L.DomEvent.stopPropagation); } else { L.DomEvent.on(div, 'click', L.DomEvent.stopPropagation); } return div; }; }; return { getOnAddLegend: _getOnAddLegend, getOnAddArrayLegend: _getOnAddArrayLegend, updateLegend: _updateLegend, }; }); angular.module('leaflet-directive').factory('leafletMapDefaults', ["$q", "leafletHelpers", function($q, leafletHelpers) { function _getDefaults() { return { keyboard: true, dragging: true, worldCopyJump: false, doubleClickZoom: true, scrollWheelZoom: true, tap: true, touchZoom: true, zoomControl: true, zoomsliderControl: false, zoomControlPosition: 'topleft', attributionControl: true, controls: { layers: { visible: true, position: 'topright', collapsed: true, }, }, nominatim: { server: ' http://nominatim.openstreetmap.org/search', }, crs: L.CRS.EPSG3857, tileLayer: '//{s}.tile.openstreetmap.org/{z}/{x}/{y}.png', tileLayerOptions: { attribution: '© <a href="http://www.openstreetmap.org/copyright">OpenStreetMap</a> contributors', }, path: { weight: 10, opacity: 1, color: '#0000ff', }, center: { lat: 0, lng: 0, zoom: 1, }, }; } var isDefined = leafletHelpers.isDefined; var isObject = leafletHelpers.isObject; var obtainEffectiveMapId = leafletHelpers.obtainEffectiveMapId; var defaults = {}; // Get the _defaults dictionary, and override the properties defined by the user return { reset: function() { defaults = {}; }, getDefaults: function(scopeId) { var mapId = obtainEffectiveMapId(defaults, scopeId); return defaults[mapId]; }, getMapCreationDefaults: function(scopeId) { var mapId = obtainEffectiveMapId(defaults, scopeId); var d = defaults[mapId]; var mapDefaults = { maxZoom: d.maxZoom, keyboard: d.keyboard, dragging: d.dragging, zoomControl: d.zoomControl, doubleClickZoom: d.doubleClickZoom, scrollWheelZoom: d.scrollWheelZoom, tap: d.tap, touchZoom: d.touchZoom, attributionControl: d.attributionControl, worldCopyJump: d.worldCopyJump, crs: d.crs, }; if (isDefined(d.minZoom)) { mapDefaults.minZoom = d.minZoom; } if (isDefined(d.zoomAnimation)) { mapDefaults.zoomAnimation = d.zoomAnimation; } if (isDefined(d.fadeAnimation)) { mapDefaults.fadeAnimation = d.fadeAnimation; } if (isDefined(d.markerZoomAnimation)) { mapDefaults.markerZoomAnimation = d.markerZoomAnimation; } if (d.map) { for (var option in d.map) { mapDefaults[option] = d.map[option]; } } return mapDefaults; }, setDefaults: function(userDefaults, scopeId) { var newDefaults = _getDefaults(); if (isDefined(userDefaults)) { newDefaults.doubleClickZoom = isDefined(userDefaults.doubleClickZoom) ? userDefaults.doubleClickZoom : newDefaults.doubleClickZoom; newDefaults.scrollWheelZoom = isDefined(userDefaults.scrollWheelZoom) ? userDefaults.scrollWheelZoom : newDefaults.doubleClickZoom; newDefaults.tap = isDefined(userDefaults.tap) ? userDefaults.tap : newDefaults.tap; newDefaults.touchZoom = isDefined(userDefaults.touchZoom) ? userDefaults.touchZoom : newDefaults.doubleClickZoom; newDefaults.zoomControl = isDefined(userDefaults.zoomControl) ? userDefaults.zoomControl : newDefaults.zoomControl; newDefaults.zoomsliderControl = isDefined(userDefaults.zoomsliderControl) ? userDefaults.zoomsliderControl : newDefaults.zoomsliderControl; newDefaults.attributionControl = isDefined(userDefaults.attributionControl) ? userDefaults.attributionControl : newDefaults.attributionControl; newDefaults.tileLayer = isDefined(userDefaults.tileLayer) ? userDefaults.tileLayer : newDefaults.tileLayer; newDefaults.zoomControlPosition = isDefined(userDefaults.zoomControlPosition) ? userDefaults.zoomControlPosition : newDefaults.zoomControlPosition; newDefaults.keyboard = isDefined(userDefaults.keyboard) ? userDefaults.keyboard : newDefaults.keyboard; newDefaults.dragging = isDefined(userDefaults.dragging) ? userDefaults.dragging : newDefaults.dragging; if (isDefined(userDefaults.controls)) { angular.extend(newDefaults.controls, userDefaults.controls); } if (isObject(userDefaults.crs)) { newDefaults.crs = userDefaults.crs; } else if (isDefined(L.CRS[userDefaults.crs])) { newDefaults.crs = L.CRS[userDefaults.crs]; } if (isDefined(userDefaults.center)) { angular.copy(userDefaults.center, newDefaults.center); } if (isDefined(userDefaults.tileLayerOptions)) { angular.copy(userDefaults.tileLayerOptions, newDefaults.tileLayerOptions); } if (isDefined(userDefaults.maxZoom)) { newDefaults.maxZoom = userDefaults.maxZoom; } if (isDefined(userDefaults.minZoom)) { newDefaults.minZoom = userDefaults.minZoom; } if (isDefined(userDefaults.zoomAnimation)) { newDefaults.zoomAnimation = userDefaults.zoomAnimation; } if (isDefined(userDefaults.fadeAnimation)) { newDefaults.fadeAnimation = userDefaults.fadeAnimation; } if (isDefined(userDefaults.markerZoomAnimation)) { newDefaults.markerZoomAnimation = userDefaults.markerZoomAnimation; } if (isDefined(userDefaults.worldCopyJump)) { newDefaults.worldCopyJump = userDefaults.worldCopyJump; } if (isDefined(userDefaults.map)) { newDefaults.map = userDefaults.map; } if (isDefined(userDefaults.path)) { newDefaults.path = userDefaults.path; } } var mapId = obtainEffectiveMapId(defaults, scopeId); defaults[mapId] = newDefaults; return newDefaults; }, }; }]); angular.module('leaflet-directive').service('leafletMarkersHelpers', ["$rootScope", "$timeout", "leafletHelpers", "$log", "$compile", "leafletGeoJsonHelpers", function($rootScope, $timeout, leafletHelpers, $log, $compile, leafletGeoJsonHelpers) { var isDefined = leafletHelpers.isDefined; var defaultTo = leafletHelpers.defaultTo; var MarkerClusterPlugin = leafletHelpers.MarkerClusterPlugin; var AwesomeMarkersPlugin = leafletHelpers.AwesomeMarkersPlugin; var VectorMarkersPlugin = leafletHelpers.VectorMarkersPlugin; var MakiMarkersPlugin = leafletHelpers.MakiMarkersPlugin; var ExtraMarkersPlugin = leafletHelpers.ExtraMarkersPlugin; var DomMarkersPlugin = leafletHelpers.DomMarkersPlugin; var safeApply = leafletHelpers.safeApply; var Helpers = leafletHelpers; var isString = leafletHelpers.isString; var isNumber = leafletHelpers.isNumber; var isObject = leafletHelpers.isObject; var groups = {}; var geoHlp = leafletGeoJsonHelpers; var errorHeader = leafletHelpers.errorHeader; var _string = function(marker) { //this exists since JSON.stringify barfs on cyclic var retStr = ''; ['_icon', '_latlng', '_leaflet_id', '_map', '_shadow'].forEach(function(prop) { retStr += prop + ': ' + defaultTo(marker[prop], 'undefined') + ' \n'; }); return '[leafletMarker] : \n' + retStr; }; var _log = function(marker, useConsole) { var logger = useConsole ? console : $log; logger.debug(_string(marker)); }; var createLeafletIcon = function(iconData) { if (isDefined(iconData) && isDefined(iconData.type) && iconData.type === 'awesomeMarker') { if (!AwesomeMarkersPlugin.isLoaded()) { $log.error(errorHeader + ' The AwesomeMarkers Plugin is not loaded.'); } return new L.AwesomeMarkers.icon(iconData); } if (isDefined(iconData) && isDefined(iconData.type) && iconData.type === 'vectorMarker') { if (!VectorMarkersPlugin.isLoaded()) { $log.error(errorHeader + ' The VectorMarkers Plugin is not loaded.'); } return new L.VectorMarkers.icon(iconData); } if (isDefined(iconData) && isDefined(iconData.type) && iconData.type === 'makiMarker') { if (!MakiMarkersPlugin.isLoaded()) { $log.error(errorHeader + 'The MakiMarkers Plugin is not loaded.'); } return new L.MakiMarkers.icon(iconData); } if (isDefined(iconData) && isDefined(iconData.type) && iconData.type === 'extraMarker') { if (!ExtraMarkersPlugin.isLoaded()) { $log.error(errorHeader + 'The ExtraMarkers Plugin is not loaded.'); } return new L.ExtraMarkers.icon(iconData); } if (isDefined(iconData) && isDefined(iconData.type) && iconData.type === 'div') { return new L.divIcon(iconData); } if (isDefined(iconData) && isDefined(iconData.type) && iconData.type === 'dom') { if (!DomMarkersPlugin.isLoaded()) { $log.error(errorHeader + 'The DomMarkers Plugin is not loaded.'); } var markerScope = angular.isFunction(iconData.getMarkerScope) ? iconData.getMarkerScope() : $rootScope; var template = $compile(iconData.template)(markerScope); var iconDataCopy = angular.copy(iconData); iconDataCopy.element = template[0]; return new L.DomMarkers.icon(iconDataCopy); } // allow for any custom icon to be used... assumes the icon has already been initialized if (isDefined(iconData) && isDefined(iconData.type) && iconData.type === 'icon') { return iconData.icon; } var base64icon = ''; var base64shadow = ''; if (!isDefined(iconData) || !isDefined(iconData.iconUrl)) { return new L.Icon.Default({ iconUrl: base64icon, shadowUrl: base64shadow, iconSize: [25, 41], iconAnchor: [12, 41], popupAnchor: [1, -34], shadowSize: [41, 41], }); } return new L.Icon(iconData); }; var _resetMarkerGroup = function(groupName) { if (isDefined(groups[groupName])) { groups.splice(groupName, 1); } }; var _resetMarkerGroups = function() { groups = {}; }; var _deleteMarker = function(marker, map, layers) { marker.closePopup(); // There is no easy way to know if a marker is added to a layer, so we search for it // if there are overlays if (isDefined(layers) && isDefined(layers.overlays)) { for (var key in layers.overlays) { if (layers.overlays[key] instanceof L.LayerGroup || layers.overlays[key] instanceof L.FeatureGroup) { if (layers.overlays[key].hasLayer(marker)) { layers.overlays[key].removeLayer(marker); return; } } } } if (isDefined(groups)) { for (var groupKey in groups) { if (groups[groupKey].hasLayer(marker)) { groups[groupKey].removeLayer(marker); } } } if (map.hasLayer(marker)) { map.removeLayer(marker); } }; var adjustPopupPan = function(marker, map) { var containerHeight = marker._popup._container.offsetHeight; var layerPos = new L.Point(marker._popup._containerLeft, -containerHeight - marker._popup._containerBottom); var containerPos = map.layerPointToContainerPoint(layerPos); if (containerPos !== null) { marker._popup._adjustPan(); } }; var compilePopup = function(marker, markerScope) { $compile(marker._popup._contentNode)(markerScope); }; var updatePopup = function(marker, markerScope, map) { //The innerText should be more than 1 once angular has compiled. //We need to keep trying until angular has compiled before we _updateLayout and _updatePosition //This should take care of any scenario , eg ngincludes, whatever. //Is there a better way to check for this? var innerText = marker._popup._contentNode.innerText || marker._popup._contentNode.textContent; if (innerText.length < 1) { $timeout(function() { updatePopup(marker, markerScope, map); }); } //cause a reflow - this is also very important - if we don't do this then the widths are from before $compile var reflow = marker._popup._contentNode.offsetWidth; marker._popup._updateLayout(); marker._popup._updatePosition(); if (marker._popup.options.autoPan) { adjustPopupPan(marker, map); } //using / returning reflow so jshint doesn't moan return reflow; }; var _manageOpenPopup = function(marker, markerData, map) { // The marker may provide a scope returning function used to compile the message // default to $rootScope otherwise var markerScope = angular.isFunction(markerData.getMessageScope) ? markerData.getMessageScope() : $rootScope; var compileMessage = isDefined(markerData.compileMessage) ? markerData.compileMessage : true; if (compileMessage) { if (!isDefined(marker._popup) || !isDefined(marker._popup._contentNode)) { $log.error(errorHeader + 'Popup is invalid or does not have any content.'); return false; } compilePopup(marker, markerScope); updatePopup(marker, markerData, map); } }; var _manageOpenLabel = function(marker, markerData) { var markerScope = angular.isFunction(markerData.getMessageScope) ? markerData.getMessageScope() : $rootScope; var labelScope = angular.isFunction(markerData.getLabelScope) ? markerData.getLabelScope() : markerScope; var compileMessage = isDefined(markerData.compileMessage) ? markerData.compileMessage : true; if (Helpers.LabelPlugin.isLoaded() && isDefined(markerData.label)) { if (isDefined(markerData.label.options) && markerData.label.options.noHide === true) { marker.showLabel(); } if (compileMessage && isDefined(marker.label)) { $compile(marker.label._container)(labelScope); } } }; var _updateMarker = function(markerData, oldMarkerData, marker, name, leafletScope, layers, map) { if (!isDefined(oldMarkerData)) { return; } // Update the lat-lng property (always present in marker properties) if (!geoHlp.validateCoords(markerData)) { $log.warn('There are problems with lat-lng data, please verify your marker model'); _deleteMarker(marker, map, layers); return; } // watch is being initialized if old and new object is the same var isInitializing = markerData === oldMarkerData; // Update marker rotation if (isDefined(markerData.iconAngle) && oldMarkerData.iconAngle !== markerData.iconAngle) { marker.setIconAngle(markerData.iconAngle); } // It is possible that the layer has been removed or the layer marker does not exist // Update the layer group if present or move it to the map if not if (!isString(markerData.layer)) { // There is no layer information, we move the marker to the map if it was in a layer group if (isString(oldMarkerData.layer)) { // Remove from the layer group that is supposed to be if (isDefined(layers.overlays[oldMarkerData.layer]) && layers.overlays[oldMarkerData.layer].hasLayer(marker)) { layers.overlays[oldMarkerData.layer].removeLayer(marker); marker.closePopup(); } // Test if it is not on the map and add it if (!map.hasLayer(marker)) { map.addLayer(marker); } } } if ((isNumber(markerData.opacity) || isNumber(parseFloat(markerData.opacity))) && markerData.opacity !== oldMarkerData.opacity) { // There was a different opacity so we update it marker.setOpacity(markerData.opacity); } if (isString(markerData.layer) && oldMarkerData.layer !== markerData.layer) { // If it was on a layer group we have to remove it if (isString(oldMarkerData.layer) && isDefined(layers.overlays[oldMarkerData.layer]) && layers.overlays[oldMarkerData.layer].hasLayer(marker)) { layers.overlays[oldMarkerData.layer].removeLayer(marker); } marker.closePopup(); // Remove it from the map in case the new layer is hidden or there is an error in the new layer if (map.hasLayer(marker)) { map.removeLayer(marker); } // The markerData.layer is defined so we add the marker to the layer if it is different from the old data if (!isDefined(layers.overlays[markerData.layer])) { $log.error(errorHeader + 'You must use a name of an existing layer'); return; } // Is a group layer? var layerGroup = layers.overlays[markerData.layer]; if (!(layerGroup instanceof L.LayerGroup || layerGroup instanceof L.FeatureGroup)) { $log.error(errorHeader + 'A marker can only be added to a layer of type "group" or "featureGroup"'); return; } // The marker goes to a correct layer group, so first of all we add it layerGroup.addLayer(marker); // The marker is automatically added to the map depending on the visibility // of the layer, so we only have to open the popup if the marker is in the map if (map.hasLayer(marker) && markerData.focus === true) { marker.openPopup(); } } // Update the draggable property if (markerData.draggable !== true && oldMarkerData.draggable === true && (isDefined(marker.dragging))) { marker.dragging.disable(); } if (markerData.draggable === true && oldMarkerData.draggable !== true) { // The markerData.draggable property must be true so we update if there wasn't a previous value or it wasn't true if (marker.dragging) { marker.dragging.enable(); } else { if (L.Handler.MarkerDrag) { marker.dragging = new L.Handler.MarkerDrag(marker); marker.options.draggable = true; marker.dragging.enable(); } } } // Update the icon property if (!isObject(markerData.icon)) { // If there is no icon property or it's not an object if (isObject(oldMarkerData.icon)) { // If there was an icon before restore to the default marker.setIcon(createLeafletIcon()); marker.closePopup(); marker.unbindPopup(); if (isString(markerData.message)) { marker.bindPopup(markerData.message, markerData.popupOptions); } } } if (isObject(markerData.icon) && isObject(oldMarkerData.icon) && !angular.equals(markerData.icon, oldMarkerData.icon)) { var dragG = false; if (marker.dragging) { dragG = marker.dragging.enabled(); } marker.setIcon(createLeafletIcon(markerData.icon)); if (dragG) { marker.dragging.enable(); } marker.closePopup(); marker.unbindPopup(); if (isString(markerData.message)) { marker.bindPopup(markerData.message, markerData.popupOptions); // if marker has been already focused, reopen popup if (map.hasLayer(marker) && markerData.focus === true) { marker.openPopup(); } } } // Update the Popup message property if (!isString(markerData.message) && isString(oldMarkerData.message)) { marker.closePopup(); marker.unbindPopup(); } // Update the label content or bind a new label if the old one has been removed. if (Helpers.LabelPlugin.isLoaded()) { if (isDefined(markerData.label) && isDefined(markerData.label.message)) { if ('label' in oldMarkerData && 'message' in oldMarkerData.label && !angular.equals(markerData.label.message, oldMarkerData.label.message)) { marker.updateLabelContent(markerData.label.message); } else if (!angular.isFunction(marker.getLabel) || angular.isFunction(marker.getLabel) && !isDefined(marker.getLabel())) { marker.bindLabel(markerData.label.message, markerData.label.options); _manageOpenLabel(marker, markerData); } else { _manageOpenLabel(marker, markerData); } } else if (!('label' in markerData && !('message' in markerData.label))) { if (angular.isFunction(marker.unbindLabel)) { marker.unbindLabel(); } } } // There is some text in the popup, so we must show the text or update existing if (isString(markerData.message) && !isString(oldMarkerData.message)) { // There was no message before so we create it marker.bindPopup(markerData.message, markerData.popupOptions); } if (isString(markerData.message) && isString(oldMarkerData.message) && markerData.message !== oldMarkerData.message) { // There was a different previous message so we update it marker.setPopupContent(markerData.message); } // Update the focus property var updatedFocus = false; if (markerData.focus !== true && oldMarkerData.focus === true) { // If there was a focus property and was true we turn it off marker.closePopup(); updatedFocus = true; } // The markerData.focus property must be true so we update if there wasn't a previous value or it wasn't true if (markerData.focus === true && (!isDefined(oldMarkerData.focus) || oldMarkerData.focus === false) || (isInitializing && markerData.focus === true)) { // Reopen the popup when focus is still true marker.openPopup(); updatedFocus = true; } // zIndexOffset adjustment if (oldMarkerData.zIndexOffset !== markerData.zIndexOffset) { marker.setZIndexOffset(markerData.zIndexOffset); } var markerLatLng = marker.getLatLng(); var isCluster = (isString(markerData.layer) && Helpers.MarkerClusterPlugin.is(layers.overlays[markerData.layer])); // If the marker is in a cluster it has to be removed and added to the layer when the location is changed if (isCluster) { // The focus has changed even by a user click or programatically if (updatedFocus) { // We only have to update the location if it was changed programatically, because it was // changed by a user drag the marker data has already been updated by the internal event // listened by the directive if ((markerData.lat !== oldMarkerData.lat) || (markerData.lng !== oldMarkerData.lng)) { layers.overlays[markerData.layer].removeLayer(marker); marker.setLatLng([markerData.lat, markerData.lng]); layers.overlays[markerData.layer].addLayer(marker); } } else { // The marker has possibly moved. It can be moved by a user drag (marker location and data are equal but old // data is diferent) or programatically (marker location and data are diferent) if ((markerLatLng.lat !== markerData.lat) || (markerLatLng.lng !== markerData.lng)) { // The marker was moved by a user drag layers.overlays[markerData.layer].removeLayer(marker); marker.setLatLng([markerData.lat, markerData.lng]); layers.overlays[markerData.layer].addLayer(marker); } else if ((markerData.lat !== oldMarkerData.lat) || (markerData.lng !== oldMarkerData.lng)) { // The marker was moved programatically layers.overlays[markerData.layer].removeLayer(marker); marker.setLatLng([markerData.lat, markerData.lng]); layers.overlays[markerData.layer].addLayer(marker); } else if (isObject(markerData.icon) && isObject(oldMarkerData.icon) && !angular.equals(markerData.icon, oldMarkerData.icon)) { layers.overlays[markerData.layer].removeLayer(marker); layers.overlays[markerData.layer].addLayer(marker); } } } else if (markerLatLng.lat !== markerData.lat || markerLatLng.lng !== markerData.lng) { marker.setLatLng([markerData.lat, markerData.lng]); } }; return { resetMarkerGroup: _resetMarkerGroup, resetMarkerGroups: _resetMarkerGroups, deleteMarker: _deleteMarker, manageOpenPopup: _manageOpenPopup, manageOpenLabel: _manageOpenLabel, createMarker: function(markerData) { if (!isDefined(markerData) || !geoHlp.validateCoords(markerData)) { $log.error(errorHeader + 'The marker definition is not valid.'); return; } var coords = geoHlp.getCoords(markerData); if (!isDefined(coords)) { $log.error(errorHeader + 'Unable to get coordinates from markerData.'); return; } var markerOptions = { icon: createLeafletIcon(markerData.icon), title: isDefined(markerData.title) ? markerData.title : '', draggable: isDefined(markerData.draggable) ? markerData.draggable : false, clickable: isDefined(markerData.clickable) ? markerData.clickable : true, riseOnHover: isDefined(markerData.riseOnHover) ? markerData.riseOnHover : false, zIndexOffset: isDefined(markerData.zIndexOffset) ? markerData.zIndexOffset : 0, iconAngle: isDefined(markerData.iconAngle) ? markerData.iconAngle : 0, }; // Add any other options not added above to markerOptions for (var markerDatum in markerData) { if (markerData.hasOwnProperty(markerDatum) && !markerOptions.hasOwnProperty(markerDatum)) { markerOptions[markerDatum] = markerData[markerDatum]; } } var marker = new L.marker(coords, markerOptions); if (!isString(markerData.message)) { marker.unbindPopup(); } return marker; }, addMarkerToGroup: function(marker, groupName, groupOptions, map) { if (!isString(groupName)) { $log.error(errorHeader + 'The marker group you have specified is invalid.'); return; } if (!MarkerClusterPlugin.isLoaded()) { $log.error(errorHeader + 'The MarkerCluster plugin is not loaded.'); return; } if (!isDefined(groups[groupName])) { groups[groupName] = new L.MarkerClusterGroup(groupOptions); map.addLayer(groups[groupName]); } groups[groupName].addLayer(marker); }, listenMarkerEvents: function(marker, markerData, leafletScope, doWatch, map) { marker.on('popupopen', function(/* event */) { safeApply(leafletScope, function() { if (isDefined(marker._popup) || isDefined(marker._popup._contentNode)) { markerData.focus = true; _manageOpenPopup(marker, markerData, map);//needed since markerData is now a copy } }); }); marker.on('popupclose', function(/* event */) { safeApply(leafletScope, function() { markerData.focus = false; }); }); marker.on('add', function(/* event */) { safeApply(leafletScope, function() { if ('label' in markerData) _manageOpenLabel(marker, markerData); }); }); }, updateMarker: _updateMarker, addMarkerWatcher: function(marker, name, leafletScope, layers, map, isDeepWatch) { var markerWatchPath = Helpers.getObjectArrayPath('markers.' + name); isDeepWatch = defaultTo(isDeepWatch, true); var clearWatch = leafletScope.$watch(markerWatchPath, function(markerData, oldMarkerData) { if (!isDefined(markerData)) { _deleteMarker(marker, map, layers); clearWatch(); return; } _updateMarker(markerData, oldMarkerData, marker, name, leafletScope, layers, map); }, isDeepWatch); }, string: _string, log: _log, }; }]); angular.module('leaflet-directive').factory('leafletPathsHelpers', ["$rootScope", "$log", "leafletHelpers", function($rootScope, $log, leafletHelpers) { var isDefined = leafletHelpers.isDefined; var isArray = leafletHelpers.isArray; var isNumber = leafletHelpers.isNumber; var isValidPoint = leafletHelpers.isValidPoint; var availableOptions = [ // Path options 'stroke', 'weight', 'color', 'opacity', 'fill', 'fillColor', 'fillOpacity', 'dashArray', 'lineCap', 'lineJoin', 'clickable', 'pointerEvents', 'className', // Polyline options 'smoothFactor', 'noClip', ]; function _convertToLeafletLatLngs(latlngs) { return latlngs.filter(function(latlng) { return isValidPoint(latlng); }).map(function(latlng) { return _convertToLeafletLatLng(latlng); }); } function _convertToLeafletLatLng(latlng) { if (isArray(latlng)) { return new L.LatLng(latlng[0], latlng[1]); } else { return new L.LatLng(latlng.lat, latlng.lng); } } function _convertToLeafletMultiLatLngs(paths) { return paths.map(function(latlngs) { return _convertToLeafletLatLngs(latlngs); }); } function _getOptions(path, defaults) { var options = {}; for (var i = 0; i < availableOptions.length; i++) { var optionName = availableOptions[i]; if (isDefined(path[optionName])) { options[optionName] = path[optionName]; } else if (isDefined(defaults.path[optionName])) { options[optionName] = defaults.path[optionName]; } } return options; } var _updatePathOptions = function(path, data) { var updatedStyle = {}; for (var i = 0; i < availableOptions.length; i++) { var optionName = availableOptions[i]; if (isDefined(data[optionName])) { updatedStyle[optionName] = data[optionName]; } } path.setStyle(data); }; var _isValidPolyline = function(latlngs) { if (!isArray(latlngs)) { return false; } for (var i = 0; i < latlngs.length; i++) { var point = latlngs[i]; if (!isValidPoint(point)) { return false; } } return true; }; var pathTypes = { polyline: { isValid: function(pathData) { var latlngs = pathData.latlngs; return _isValidPolyline(latlngs); }, createPath: function(options) { return new L.Polyline([], options); }, setPath: function(path, data) { path.setLatLngs(_convertToLeafletLatLngs(data.latlngs)); _updatePathOptions(path, data); return; }, }, multiPolyline: { isValid: function(pathData) { var latlngs = pathData.latlngs; if (!isArray(latlngs)) { return false; } for (var i in latlngs) { var polyline = latlngs[i]; if (!_isValidPolyline(polyline)) { return false; } } return true; }, createPath: function(options) { return new L.multiPolyline([[[0, 0], [1, 1]]], options); }, setPath: function(path, data) { path.setLatLngs(_convertToLeafletMultiLatLngs(data.latlngs)); _updatePathOptions(path, data); return; }, }, polygon: { isValid: function(pathData) { var latlngs = pathData.latlngs; return _isValidPolyline(latlngs); }, createPath: function(options) { return new L.Polygon([], options); }, setPath: function(path, data) { path.setLatLngs(_convertToLeafletLatLngs(data.latlngs)); _updatePathOptions(path, data); return; }, }, multiPolygon: { isValid: function(pathData) { var latlngs = pathData.latlngs; if (!isArray(latlngs)) { return false; } for (var i in latlngs) { var polyline = latlngs[i]; if (!_isValidPolyline(polyline)) { return false; } } return true; }, createPath: function(options) { return new L.MultiPolygon([[[0, 0], [1, 1], [0, 1]]], options); }, setPath: function(path, data) { path.setLatLngs(_convertToLeafletMultiLatLngs(data.latlngs)); _updatePathOptions(path, data); return; }, }, rectangle: { isValid: function(pathData) { var latlngs = pathData.latlngs; if (!isArray(latlngs) || latlngs.length !== 2) { return false; } for (var i in latlngs) { var point = latlngs[i]; if (!isValidPoint(point)) { return false; } } return true; }, createPath: function(options) { return new L.Rectangle([[0, 0], [1, 1]], options); }, setPath: function(path, data) { path.setBounds(new L.LatLngBounds(_convertToLeafletLatLngs(data.latlngs))); _updatePathOptions(path, data); }, }, circle: { isValid: function(pathData) { var point = pathData.latlngs; return isValidPoint(point) && isNumber(pathData.radius); }, createPath: function(options) { return new L.Circle([0, 0], 1, options); }, setPath: function(path, data) { path.setLatLng(_convertToLeafletLatLng(data.latlngs)); if (isDefined(data.radius)) { path.setRadius(data.radius); } _updatePathOptions(path, data); }, }, circleMarker: { isValid: function(pathData) { var point = pathData.latlngs; return isValidPoint(point) && isNumber(pathData.radius); }, createPath: function(options) { return new L.CircleMarker([0, 0], options); }, setPath: function(path, data) { path.setLatLng(_convertToLeafletLatLng(data.latlngs)); if (isDefined(data.radius)) { path.setRadius(data.radius); } _updatePathOptions(path, data); }, }, }; var _getPathData = function(path) { var pathData = {}; if (path.latlngs) { pathData.latlngs = path.latlngs; } if (path.radius) { pathData.radius = path.radius; } return pathData; }; return { setPathOptions: function(leafletPath, pathType, data) { if (!isDefined(pathType)) { pathType = 'polyline'; } pathTypes[pathType].setPath(leafletPath, data); }, createPath: function(name, path, defaults) { if (!isDefined(path.type)) { path.type = 'polyline'; } var options = _getOptions(path, defaults); var pathData = _getPathData(path); if (!pathTypes[path.type].isValid(pathData)) { $log.error('[AngularJS - Leaflet] Invalid data passed to the ' + path.type + ' path'); return; } return pathTypes[path.type].createPath(options); }, }; }]); angular.module('leaflet-directive') .service('leafletWatchHelpers', function() { var _maybe = function(scope, watchFunctionName, thingToWatchStr, watchOptions, initCb) { //watchOptions.isDeep is/should be ignored in $watchCollection var unWatch = scope[watchFunctionName](thingToWatchStr, function(newValue, oldValue) { initCb(newValue, oldValue); if (!watchOptions.doWatch) unWatch(); }, watchOptions.isDeep); return unWatch; }; /* @name: maybeWatch @description: Utility to watch something once or forever. @returns unWatch function @param watchOptions - see markersWatchOptions and or derrivatives. This object is used to set watching to once and its watch depth. */ var _maybeWatch = function(scope, thingToWatchStr, watchOptions, initCb) { return _maybe(scope, '$watch', thingToWatchStr, watchOptions, initCb); }; /* @name: _maybeWatchCollection @description: Utility to watch something once or forever. @returns unWatch function @param watchOptions - see markersWatchOptions and or derrivatives. This object is used to set watching to once and its watch depth. */ var _maybeWatchCollection = function(scope, thingToWatchStr, watchOptions, initCb) { return _maybe(scope, '$watchCollection', thingToWatchStr, watchOptions, initCb); }; return { maybeWatch: _maybeWatch, maybeWatchCollection: _maybeWatchCollection, }; }); angular.module('leaflet-directive').factory('nominatimService', ["$q", "$http", "leafletHelpers", "leafletMapDefaults", function($q, $http, leafletHelpers, leafletMapDefaults) { var isDefined = leafletHelpers.isDefined; return { query: function(address, mapId) { var defaults = leafletMapDefaults.getDefaults(mapId); var url = defaults.nominatim.server; var df = $q.defer(); $http.get(url, { params: { format: 'json', limit: 1, q: address } }).success(function(data) { if (data.length > 0 && isDefined(data[0].boundingbox)) { df.resolve(data[0]); } else { df.reject('[Nominatim] Invalid address'); } }); return df.promise; }, }; }]); angular.module('leaflet-directive').directive('bounds', ["$log", "$timeout", "$http", "leafletHelpers", "nominatimService", "leafletBoundsHelpers", function($log, $timeout, $http, leafletHelpers, nominatimService, leafletBoundsHelpers) { return { restrict: 'A', scope: false, replace: false, require: ['leaflet'], link: function(scope, element, attrs, controller) { var isDefined = leafletHelpers.isDefined; var createLeafletBounds = leafletBoundsHelpers.createLeafletBounds; var leafletScope = controller[0].getLeafletScope(); var mapController = controller[0]; var errorHeader = leafletHelpers.errorHeader + ' [Bounds] '; var emptyBounds = function(bounds) { return (bounds._southWest.lat === 0 && bounds._southWest.lng === 0 && bounds._northEast.lat === 0 && bounds._northEast.lng === 0); }; mapController.getMap().then(function(map) { leafletScope.$on('boundsChanged', function(event) { var scope = event.currentScope; var bounds = map.getBounds(); if (emptyBounds(bounds) || scope.settingBoundsFromScope) { return; } scope.settingBoundsFromLeaflet = true; var newScopeBounds = { northEast: { lat: bounds._northEast.lat, lng: bounds._northEast.lng, }, southWest: { lat: bounds._southWest.lat, lng: bounds._southWest.lng, }, options: bounds.options, }; if (!angular.equals(scope.bounds, newScopeBounds)) { scope.bounds = newScopeBounds; } $timeout(function() { scope.settingBoundsFromLeaflet = false; }); }); var lastNominatimQuery; leafletScope.$watch('bounds', function(bounds) { if (scope.settingBoundsFromLeaflet) return; if (isDefined(bounds.address) && bounds.address !== lastNominatimQuery) { scope.settingBoundsFromScope = true; nominatimService.query(bounds.address, attrs.id).then(function(data) { var b = data.boundingbox; var newBounds = [[b[0], b[2]], [b[1], b[3]]]; map.fitBounds(newBounds); }, function(errMsg) { $log.error(errorHeader + ' ' + errMsg + '.'); }); lastNominatimQuery = bounds.address; $timeout(function() { scope.settingBoundsFromScope = false; }); return; } var leafletBounds = createLeafletBounds(bounds); if (leafletBounds && !map.getBounds().equals(leafletBounds)) { scope.settingBoundsFromScope = true; map.fitBounds(leafletBounds, bounds.options); $timeout(function() { scope.settingBoundsFromScope = false; }); } }, true); }); }, }; }]); var centerDirectiveTypes = ['center', 'lfCenter']; var centerDirectives = {}; centerDirectiveTypes.forEach(function(directiveName) { centerDirectives[directiveName] = ['$log', '$q', '$location', '$timeout', 'leafletMapDefaults', 'leafletHelpers', 'leafletBoundsHelpers', 'leafletMapEvents', function($log, $q, $location, $timeout, leafletMapDefaults, leafletHelpers, leafletBoundsHelpers, leafletMapEvents) { var isDefined = leafletHelpers.isDefined; var isNumber = leafletHelpers.isNumber; var isSameCenterOnMap = leafletHelpers.isSameCenterOnMap; var safeApply = leafletHelpers.safeApply; var isValidCenter = leafletHelpers.isValidCenter; var isValidBounds = leafletBoundsHelpers.isValidBounds; var isUndefinedOrEmpty = leafletHelpers.isUndefinedOrEmpty; var errorHeader = leafletHelpers.errorHeader; var shouldInitializeMapWithBounds = function(bounds, center) { return isDefined(bounds) && isValidBounds(bounds) && isUndefinedOrEmpty(center); }; var _leafletCenter; return { restrict: 'A', scope: false, replace: false, require: 'leaflet', controller: function() { _leafletCenter = $q.defer(); this.getCenter = function() { return _leafletCenter.promise; }; }, link: function(scope, element, attrs, controller) { var leafletScope = controller.getLeafletScope(); var centerModel = leafletScope[directiveName]; controller.getMap().then(function(map) { var defaults = leafletMapDefaults.getDefaults(attrs.id); if (attrs[directiveName].search('-') !== -1) { $log.error(errorHeader + ' The "center" variable can\'t use a "-" on its key name: "' + attrs[directiveName] + '".'); map.setView([defaults.center.lat, defaults.center.lng], defaults.center.zoom); return; } else if (shouldInitializeMapWithBounds(leafletScope.bounds, centerModel)) { map.fitBounds(leafletBoundsHelpers.createLeafletBounds(leafletScope.bounds), leafletScope.bounds.options); centerModel = map.getCenter(); safeApply(leafletScope, function(scope) { angular.extend(scope[directiveName], { lat: map.getCenter().lat, lng: map.getCenter().lng, zoom: map.getZoom(), autoDiscover: false, }); }); safeApply(leafletScope, function(scope) { var mapBounds = map.getBounds(); scope.bounds = { northEast: { lat: mapBounds._northEast.lat, lng: mapBounds._northEast.lng, }, southWest: { lat: mapBounds._southWest.lat, lng: mapBounds._southWest.lng, }, }; }); } else if (!isDefined(centerModel)) { $log.error(errorHeader + ' The "center" property is not defined in the main scope'); map.setView([defaults.center.lat, defaults.center.lng], defaults.center.zoom); return; } else if (!(isDefined(centerModel.lat) && isDefined(centerModel.lng)) && !isDefined(centerModel.autoDiscover)) { angular.copy(defaults.center, centerModel); } var urlCenterHash; var mapReady; if (attrs.urlHashCenter === 'yes') { var extractCenterFromUrl = function() { var search = $location.search(); var centerParam; if (isDefined(search.c)) { var cParam = search.c.split(':'); if (cParam.length === 3) { centerParam = { lat: parseFloat(cParam[0]), lng: parseFloat(cParam[1]), zoom: parseInt(cParam[2], 10), }; } } return centerParam; }; urlCenterHash = extractCenterFromUrl(); leafletScope.$on('$locationChangeSuccess', function(event) { var scope = event.currentScope; //$log.debug("updated location..."); var urlCenter = extractCenterFromUrl(); if (isDefined(urlCenter) && !isSameCenterOnMap(urlCenter, map)) { //$log.debug("updating center model...", urlCenter); angular.extend(scope[directiveName], { lat: urlCenter.lat, lng: urlCenter.lng, zoom: urlCenter.zoom, }); } }); } leafletScope.$watch(directiveName, function(center) { if (leafletScope.settingCenterFromLeaflet) return; //$log.debug("updated center model..."); // The center from the URL has priority if (isDefined(urlCenterHash)) { angular.copy(urlCenterHash, center); urlCenterHash = undefined; } if (!isValidCenter(center) && center.autoDiscover !== true) { $log.warn(errorHeader + ' invalid \'center\''); //map.setView([defaults.center.lat, defaults.center.lng], defaults.center.zoom); return; } if (center.autoDiscover === true) { if (!isNumber(center.zoom)) { map.setView([defaults.center.lat, defaults.center.lng], defaults.center.zoom); } if (isNumber(center.zoom) && center.zoom > defaults.center.zoom) { map.locate({ setView: true, maxZoom: center.zoom, }); } else if (isDefined(defaults.maxZoom)) { map.locate({ setView: true, maxZoom: defaults.maxZoom, }); } else { map.locate({ setView: true, }); } return; } if (mapReady && isSameCenterOnMap(center, map)) { //$log.debug("no need to update map again."); return; } //$log.debug("updating map center...", center); leafletScope.settingCenterFromScope = true; map.setView([center.lat, center.lng], center.zoom); leafletMapEvents.notifyCenterChangedToBounds(leafletScope, map); $timeout(function() { leafletScope.settingCenterFromScope = false; //$log.debug("allow center scope updates"); }); }, true); map.whenReady(function() { mapReady = true; }); map.on('moveend', function(/* event */) { // Resolve the center after the first map position _leafletCenter.resolve(); leafletMapEvents.notifyCenterUrlHashChanged(leafletScope, map, attrs, $location.search()); //$log.debug("updated center on map..."); if (isSameCenterOnMap(centerModel, map) || leafletScope.settingCenterFromScope) { //$log.debug("same center in model, no need to update again."); return; } leafletScope.settingCenterFromLeaflet = true; safeApply(leafletScope, function(scope) { if (!leafletScope.settingCenterFromScope) { //$log.debug("updating center model...", map.getCenter(), map.getZoom()); angular.extend(scope[directiveName], { lat: map.getCenter().lat, lng: map.getCenter().lng, zoom: map.getZoom(), autoDiscover: false, }); } leafletMapEvents.notifyCenterChangedToBounds(leafletScope, map); $timeout(function() { leafletScope.settingCenterFromLeaflet = false; }); }); }); if (centerModel.autoDiscover === true) { map.on('locationerror', function() { $log.warn(errorHeader + ' The Geolocation API is unauthorized on this page.'); if (isValidCenter(centerModel)) { map.setView([centerModel.lat, centerModel.lng], centerModel.zoom); leafletMapEvents.notifyCenterChangedToBounds(leafletScope, map); } else { map.setView([defaults.center.lat, defaults.center.lng], defaults.center.zoom); leafletMapEvents.notifyCenterChangedToBounds(leafletScope, map); } }); } }); }, }; }, ]; }); centerDirectiveTypes.forEach(function(dirType) { angular.module('leaflet-directive').directive(dirType, centerDirectives[dirType]); }); angular.module('leaflet-directive').directive('controls', ["$log", "leafletHelpers", "leafletControlHelpers", function($log, leafletHelpers, leafletControlHelpers) { return { restrict: 'A', scope: false, replace: false, require: '?^leaflet', link: function(scope, element, attrs, controller) { if (!controller) { return; } var createControl = leafletControlHelpers.createControl; var isValidControlType = leafletControlHelpers.isValidControlType; var leafletScope = controller.getLeafletScope(); var isDefined = leafletHelpers.isDefined; var isArray = leafletHelpers.isArray; var leafletControls = {}; var errorHeader = leafletHelpers.errorHeader + ' [Controls] '; controller.getMap().then(function(map) { leafletScope.$watchCollection('controls', function(newControls) { // Delete controls from the array for (var name in leafletControls) { if (!isDefined(newControls[name])) { if (map.hasControl(leafletControls[name])) { map.removeControl(leafletControls[name]); } delete leafletControls[name]; } } for (var newName in newControls) { var control; var controlType = isDefined(newControls[newName].type) ? newControls[newName].type : newName; if (!isValidControlType(controlType)) { $log.error(errorHeader + ' Invalid control type: ' + controlType + '.'); return; } if (controlType !== 'custom') { control = createControl(controlType, newControls[newName]); map.addControl(control); leafletControls[newName] = control; } else { var customControlValue = newControls[newName]; if (isArray(customControlValue)) { for (var i in customControlValue) { var customControl = customControlValue[i]; map.addControl(customControl); leafletControls[newName] = !isDefined(leafletControls[newName]) ? [customControl] : leafletControls[newName].concat([customControl]); } } else { map.addControl(customControlValue); leafletControls[newName] = customControlValue; } } } }); }); }, }; }]); angular.module('leaflet-directive').directive('decorations', ["$log", "leafletHelpers", function($log, leafletHelpers) { return { restrict: 'A', scope: false, replace: false, require: 'leaflet', link: function(scope, element, attrs, controller) { var leafletScope = controller.getLeafletScope(); var PolylineDecoratorPlugin = leafletHelpers.PolylineDecoratorPlugin; var isDefined = leafletHelpers.isDefined; var leafletDecorations = {}; /* Creates an "empty" decoration with a set of coordinates, but no pattern. */ function createDecoration(options) { if (isDefined(options) && isDefined(options.coordinates)) { if (!PolylineDecoratorPlugin.isLoaded()) { $log.error('[AngularJS - Leaflet] The PolylineDecorator Plugin is not loaded.'); } } return L.polylineDecorator(options.coordinates); } /* Updates the path and the patterns for the provided decoration, and returns the decoration. */ function setDecorationOptions(decoration, options) { if (isDefined(decoration) && isDefined(options)) { if (isDefined(options.coordinates) && isDefined(options.patterns)) { decoration.setPaths(options.coordinates); decoration.setPatterns(options.patterns); return decoration; } } } controller.getMap().then(function(map) { leafletScope.$watch('decorations', function(newDecorations) { for (var name in leafletDecorations) { if (!isDefined(newDecorations[name]) || !angular.equals(newDecorations[name], leafletDecorations)) { map.removeLayer(leafletDecorations[name]); delete leafletDecorations[name]; } } for (var newName in newDecorations) { var decorationData = newDecorations[newName]; var newDecoration = createDecoration(decorationData); if (isDefined(newDecoration)) { leafletDecorations[newName] = newDecoration; map.addLayer(newDecoration); setDecorationOptions(newDecoration, decorationData); } } }, true); }); }, }; }]); angular.module('leaflet-directive').directive('eventBroadcast', ["$log", "$rootScope", "leafletHelpers", "leafletMapEvents", "leafletIterators", function($log, $rootScope, leafletHelpers, leafletMapEvents, leafletIterators) { return { restrict: 'A', scope: false, replace: false, require: 'leaflet', link: function(scope, element, attrs, controller) { var isObject = leafletHelpers.isObject; var isDefined = leafletHelpers.isDefined; var leafletScope = controller.getLeafletScope(); var eventBroadcast = leafletScope.eventBroadcast; var availableMapEvents = leafletMapEvents.getAvailableMapEvents(); var addEvents = leafletMapEvents.addEvents; controller.getMap().then(function(map) { var mapEvents = []; var logic = 'broadcast'; // We have a possible valid object if (!isDefined(eventBroadcast.map)) { // We do not have events enable/disable do we do nothing (all enabled by default) mapEvents = availableMapEvents; } else if (!isObject(eventBroadcast.map)) { // Not a valid object $log.warn('[AngularJS - Leaflet] event-broadcast.map must be an object check your model.'); } else { // We have a possible valid map object // Event propadation logic if (eventBroadcast.map.logic !== 'emit' && eventBroadcast.map.logic !== 'broadcast') { // This is an error $log.warn('[AngularJS - Leaflet] Available event propagation logic are: \'emit\' or \'broadcast\'.'); } else { logic = eventBroadcast.map.logic; } if (!(isObject(eventBroadcast.map.enable) && eventBroadcast.map.enable.length >= 0)) { $log.warn('[AngularJS - Leaflet] event-broadcast.map.enable must be an object check your model.'); } else { // Enable events leafletIterators.each(eventBroadcast.map.enable, function(eventName) { // Do we have already the event enabled? if (mapEvents.indexOf(eventName) === -1 && availableMapEvents.indexOf(eventName) !== -1) { mapEvents.push(eventName); } }); } } // as long as the map is removed in the root leaflet directive we // do not need ot clean up the events as leaflet does it itself addEvents(map, mapEvents, 'eventName', leafletScope, logic); }); }, }; }]); angular.module('leaflet-directive') .directive('geojson', ["$log", "$rootScope", "leafletData", "leafletHelpers", "leafletWatchHelpers", "leafletDirectiveControlsHelpers", "leafletIterators", "leafletGeoJsonEvents", function($log, $rootScope, leafletData, leafletHelpers, leafletWatchHelpers, leafletDirectiveControlsHelpers, leafletIterators, leafletGeoJsonEvents) { var _maybeWatch = leafletWatchHelpers.maybeWatch; var _watchOptions = leafletHelpers.watchOptions; var _extendDirectiveControls = leafletDirectiveControlsHelpers.extend; var hlp = leafletHelpers; var $it = leafletIterators; return { restrict: 'A', scope: false, replace: false, require: 'leaflet', link: function(scope, element, attrs, controller) { var isDefined = leafletHelpers.isDefined; var leafletScope = controller.getLeafletScope(); var leafletGeoJSON = {}; var _hasSetLeafletData = false; controller.getMap().then(function(map) { var watchOptions = leafletScope.geojsonWatchOptions || _watchOptions; var _hookUpEvents = function(geojson, maybeName) { var onEachFeature; if (angular.isFunction(geojson.onEachFeature)) { onEachFeature = geojson.onEachFeature; } else { onEachFeature = function(feature, layer) { if (leafletHelpers.LabelPlugin.isLoaded() && isDefined(feature.properties.description)) { layer.bindLabel(feature.properties.description); } leafletGeoJsonEvents.bindEvents(attrs.id, layer, null, feature, leafletScope, maybeName, {resetStyleOnMouseout: geojson.resetStyleOnMouseout, mapId: attrs.id, }); }; } return onEachFeature; }; var isNested = (hlp.isDefined(attrs.geojsonNested) && hlp.isTruthy(attrs.geojsonNested)); var _clean = function() { if (!leafletGeoJSON) return; var _remove = function(lObject) { if (isDefined(lObject) && map.hasLayer(lObject)) { map.removeLayer(lObject); } }; if (isNested) { $it.each(leafletGeoJSON, function(lObject) { _remove(lObject); }); return; } _remove(leafletGeoJSON); }; var _addGeojson = function(model, maybeName) { var geojson = angular.copy(model); if (!(isDefined(geojson) && isDefined(geojson.data))) { return; } var onEachFeature = _hookUpEvents(geojson, maybeName); if (!isDefined(geojson.options)) { //right here is why we use a clone / copy (we modify and thus) //would kick of a watcher.. we need to be more careful everywhere //for stuff like this geojson.options = { style: geojson.style, filter: geojson.filter, onEachFeature: onEachFeature, pointToLayer: geojson.pointToLayer, }; } var lObject = L.geoJson(geojson.data, geojson.options); if (maybeName && hlp.isString(maybeName)) { leafletGeoJSON[maybeName] = lObject; } else { leafletGeoJSON = lObject; } lObject.addTo(map); if (!_hasSetLeafletData) {//only do this once and play with the same ref forever _hasSetLeafletData = true; leafletData.setGeoJSON(leafletGeoJSON, attrs.id); } }; var _create = function(model) { _clean(); if (isNested) { if (!model || !Object.keys(model).length) return; $it.each(model, function(m, name) { //name could be layerName and or groupName //for now it is not tied to a layer _addGeojson(m, name); }); return; } _addGeojson(model); }; _extendDirectiveControls(attrs.id, 'geojson', _create, _clean); _maybeWatch(leafletScope, 'geojson', watchOptions, function(geojson) { _create(geojson); }); }); }, }; }]); angular.module('leaflet-directive').directive('layercontrol', ["$filter", "$log", "leafletData", "leafletHelpers", function($filter, $log, leafletData, leafletHelpers) { return { restrict: 'E', scope: { icons: '=?', autoHideOpacity: '=?', // Hide other opacity controls when one is activated. showGroups: '=?', // Hide other opacity controls when one is activated. title: '@', baseTitle: '@', overlaysTitle: '@', }, replace: true, transclude: false, require: '^leaflet', controller: ["$scope", "$element", "$sce", function($scope, $element, $sce) { $log.debug('[Angular Directive - Layers] layers', $scope, $element); var safeApply = leafletHelpers.safeApply; var isDefined = leafletHelpers.isDefined; angular.extend($scope, { baselayer: '', oldGroup: '', layerProperties: {}, groupProperties: {}, rangeIsSupported: leafletHelpers.rangeIsSupported(), changeBaseLayer: function(key, e) { leafletHelpers.safeApply($scope, function(scp) { scp.baselayer = key; leafletData.getMap().then(function(map) { leafletData.getLayers().then(function(leafletLayers) { if (map.hasLayer(leafletLayers.baselayers[key])) { return; } for (var i in scp.layers.baselayers) { scp.layers.baselayers[i].icon = scp.icons.unradio; if (map.hasLayer(leafletLayers.baselayers[i])) { map.removeLayer(leafletLayers.baselayers[i]); } } map.addLayer(leafletLayers.baselayers[key]); scp.layers.baselayers[key].icon = $scope.icons.radio; }); }); }); e.preventDefault(); }, moveLayer: function(ly, newIndex, e) { var delta = Object.keys($scope.layers.baselayers).length; if (newIndex >= (1 + delta) && newIndex <= ($scope.overlaysArray.length + delta)) { var oldLy; for (var key in $scope.layers.overlays) { if ($scope.layers.overlays[key].index === newIndex) { oldLy = $scope.layers.overlays[key]; break; } } if (oldLy) { safeApply($scope, function() { oldLy.index = ly.index; ly.index = newIndex; }); } } e.stopPropagation(); e.preventDefault(); }, initIndex: function(layer, idx) { var delta = Object.keys($scope.layers.baselayers).length; layer.index = isDefined(layer.index) ? layer.index : idx + delta + 1; }, initGroup: function(groupName) { $scope.groupProperties[groupName] = $scope.groupProperties[groupName] ? $scope.groupProperties[groupName] : {}; }, toggleOpacity: function(e, layer) { if (layer.visible) { if ($scope.autoHideOpacity && !$scope.layerProperties[layer.name].opacityControl) { for (var k in $scope.layerProperties) { $scope.layerProperties[k].opacityControl = false; } } $scope.layerProperties[layer.name].opacityControl = !$scope.layerProperties[layer.name].opacityControl; } e.stopPropagation(); e.preventDefault(); }, toggleLegend: function(layer) { $scope.layerProperties[layer.name].showLegend = !$scope.layerProperties[layer.name].showLegend; }, showLegend: function(layer) { return layer.legend && $scope.layerProperties[layer.name].showLegend; }, unsafeHTML: function(html) { return $sce.trustAsHtml(html); }, getOpacityIcon: function(layer) { return layer.visible && $scope.layerProperties[layer.name].opacityControl ? $scope.icons.close : $scope.icons.open; }, getGroupIcon: function(group) { return group.visible ? $scope.icons.check : $scope.icons.uncheck; }, changeOpacity: function(layer) { var op = $scope.layerProperties[layer.name].opacity; leafletData.getMap().then(function(map) { leafletData.getLayers().then(function(leafletLayers) { var ly; for (var k in $scope.layers.overlays) { if ($scope.layers.overlays[k] === layer) { ly = leafletLayers.overlays[k]; break; } } if (map.hasLayer(ly)) { if (ly.setOpacity) { ly.setOpacity(op / 100); } if (ly.getLayers && ly.eachLayer) { ly.eachLayer(function(lay) { if (lay.setOpacity) { lay.setOpacity(op / 100); } }); } } }); }); }, changeGroupVisibility: function(groupName) { if (!isDefined($scope.groupProperties[groupName])) { return; } var visible = $scope.groupProperties[groupName].visible; for (var k in $scope.layers.overlays) { var layer = $scope.layers.overlays[k]; if (layer.group === groupName) { layer.visible = visible; } } }, }); var div = $element.get(0); if (!L.Browser.touch) { L.DomEvent.disableClickPropagation(div); L.DomEvent.on(div, 'mousewheel', L.DomEvent.stopPropagation); } else { L.DomEvent.on(div, 'click', L.DomEvent.stopPropagation); } }], template: '<div class="angular-leaflet-control-layers" ng-show="overlaysArray.length">' + '<h4 ng-if="title">{{ title }}</h4>' + '<div class="lf-baselayers">' + '<h5 class="lf-title" ng-if="baseTitle">{{ baseTitle }}</h5>' + '<div class="lf-row" ng-repeat="(key, layer) in baselayersArray">' + '<label class="lf-icon-bl" ng-click="changeBaseLayer(key, $event)">' + '<input class="leaflet-control-layers-selector" type="radio" name="lf-radio" ' + 'ng-show="false" ng-checked="baselayer === key" ng-value="key" /> ' + '<i class="lf-icon lf-icon-radio" ng-class="layer.icon"></i>' + '<div class="lf-text">{{layer.name}}</div>' + '</label>' + '</div>' + '</div>' + '<div class="lf-overlays">' + '<h5 class="lf-title" ng-if="overlaysTitle">{{ overlaysTitle }}</h5>' + '<div class="lf-container">' + '<div class="lf-row" ng-repeat="layer in (o = (overlaysArray | orderBy:\'index\':order))" ng-init="initIndex(layer, $index)">' + '<label class="lf-icon-ol-group" ng-if="showGroups && layer.group && layer.group != o[$index-1].group">' + '<input class="lf-control-layers-selector" type="checkbox" ng-show="false" ' + 'ng-change="changeGroupVisibility(layer.group)" ng-model="groupProperties[layer.group].visible"/> ' + '<i class="lf-icon lf-icon-check" ng-class="getGroupIcon(groupProperties[layer.group])"></i>' + '<div class="lf-text">{{ layer.group }}</div>' + '</label>' + '<label class="lf-icon-ol">' + '<input class="lf-control-layers-selector" type="checkbox" ng-show="false" ng-model="layer.visible"/> ' + '<i class="lf-icon lf-icon-check" ng-class="layer.icon"></i>' + '<div class="lf-text">{{layer.name}}</div>' + '</label>' + '<div class="lf-icons">' + '<i class="lf-icon lf-up" ng-class="icons.up" ng-click="moveLayer(layer, layer.index - orderNumber, $event)"></i> ' + '<i class="lf-icon lf-down" ng-class="icons.down" ng-click="moveLayer(layer, layer.index + orderNumber, $event)"></i> ' + '<i class="lf-icon lf-toggle-legend" ng-class="icons.toggleLegend" ng-if="layer.legend" ng-click="toggleLegend(layer)"></i> ' + '<i class="lf-icon lf-open" ng-class="getOpacityIcon(layer)" ng-click="toggleOpacity($event, layer)"></i>' + '</div>' + '<div class="lf-legend" ng-if="showLegend(layer)" ng-bind-html="unsafeHTML(layer.legend)"></div>' + '<div class="lf-opacity clearfix" ng-if="layer.visible && layerProperties[layer.name].opacityControl">' + '<label ng-if="rangeIsSupported" class="pull-left" style="width: 50%">0</label>' + '<label ng-if="rangeIsSupported" class="pull-left text-right" style="width: 50%">100</label>' + '<input ng-if="rangeIsSupported" class="clearfix" type="range" min="0" max="100" class="lf-opacity-control" ' + 'ng-model="layerProperties[layer.name].opacity" ng-change="changeOpacity(layer)"/>' + '<h6 ng-if="!rangeIsSupported">Range is not supported in this browser</h6>' + '</div>' + '</div>' + '</div>' + '</div>' + '</div>', link: function(scope, element, attrs, controller) { var isDefined = leafletHelpers.isDefined; var leafletScope = controller.getLeafletScope(); var layers = leafletScope.layers; scope.$watch('icons', function() { var defaultIcons = { uncheck: 'fa fa-square-o', check: 'fa fa-check-square-o', radio: 'fa fa-dot-circle-o', unradio: 'fa fa-circle-o', up: 'fa fa-angle-up', down: 'fa fa-angle-down', open: 'fa fa-angle-double-down', close: 'fa fa-angle-double-up', toggleLegend: 'fa fa-pencil-square-o', }; if (isDefined(scope.icons)) { angular.extend(defaultIcons, scope.icons); angular.extend(scope.icons, defaultIcons); } else { scope.icons = defaultIcons; } }); // Setting layer stack order. attrs.order = (isDefined(attrs.order) && (attrs.order === 'normal' || attrs.order === 'reverse')) ? attrs.order : 'normal'; scope.order = attrs.order === 'normal'; scope.orderNumber = attrs.order === 'normal' ? -1 : 1; scope.layers = layers; controller.getMap().then(function(map) { leafletScope.$watch('layers.baselayers', function(newBaseLayers) { var baselayersArray = {}; leafletData.getLayers().then(function(leafletLayers) { var key; for (key in newBaseLayers) { var layer = newBaseLayers[key]; layer.icon = scope.icons[map.hasLayer(leafletLayers.baselayers[key]) ? 'radio' : 'unradio']; baselayersArray[key] = layer; } scope.baselayersArray = baselayersArray; }); }); leafletScope.$watch('layers.overlays', function(newOverlayLayers) { var overlaysArray = []; var groupVisibleCount = {}; leafletData.getLayers().then(function(leafletLayers) { var key; for (key in newOverlayLayers) { var layer = newOverlayLayers[key]; layer.icon = scope.icons[(layer.visible ? 'check' : 'uncheck')]; overlaysArray.push(layer); if (!isDefined(scope.layerProperties[layer.name])) { scope.layerProperties[layer.name] = { opacity: isDefined(layer.layerOptions.opacity) ? layer.layerOptions.opacity * 100 : 100, opacityControl: false, showLegend: true, }; } if (isDefined(layer.group)) { if (!isDefined(scope.groupProperties[layer.group])) { scope.groupProperties[layer.group] = { visible: false, }; } groupVisibleCount[layer.group] = isDefined(groupVisibleCount[layer.group]) ? groupVisibleCount[layer.group] : { count: 0, visibles: 0, }; groupVisibleCount[layer.group].count++; if (layer.visible) { groupVisibleCount[layer.group].visibles++; } } if (isDefined(layer.index) && leafletLayers.overlays[key].setZIndex) { leafletLayers.overlays[key].setZIndex(newOverlayLayers[key].index); } } for (key in groupVisibleCount) { scope.groupProperties[key].visible = groupVisibleCount[key].visibles === groupVisibleCount[key].count; } scope.overlaysArray = overlaysArray; }); }, true); }); }, }; }]); angular.module('leaflet-directive').directive('layers', ["$log", "$q", "leafletData", "leafletHelpers", "leafletLayerHelpers", "leafletControlHelpers", function($log, $q, leafletData, leafletHelpers, leafletLayerHelpers, leafletControlHelpers) { return { restrict: 'A', scope: false, replace: false, require: 'leaflet', controller: ["$scope", function($scope) { $scope._leafletLayers = $q.defer(); this.getLayers = function() { return $scope._leafletLayers.promise; }; }], link: function(scope, element, attrs, controller) { var isDefined = leafletHelpers.isDefined; var leafletLayers = {}; var leafletScope = controller.getLeafletScope(); var layers = leafletScope.layers; var createLayer = leafletLayerHelpers.createLayer; var safeAddLayer = leafletLayerHelpers.safeAddLayer; var safeRemoveLayer = leafletLayerHelpers.safeRemoveLayer; var updateLayersControl = leafletControlHelpers.updateLayersControl; var isLayersControlVisible = false; controller.getMap().then(function(map) { // We have baselayers to add to the map scope._leafletLayers.resolve(leafletLayers); leafletData.setLayers(leafletLayers, attrs.id); leafletLayers.baselayers = {}; leafletLayers.overlays = {}; var mapId = attrs.id; // Setup all baselayers definitions var oneVisibleLayer = false; for (var layerName in layers.baselayers) { var newBaseLayer = createLayer(layers.baselayers[layerName]); if (!isDefined(newBaseLayer)) { delete layers.baselayers[layerName]; continue; } leafletLayers.baselayers[layerName] = newBaseLayer; // Only add the visible layer to the map, layer control manages the addition to the map // of layers in its control if (layers.baselayers[layerName].top === true) { safeAddLayer(map, leafletLayers.baselayers[layerName]); oneVisibleLayer = true; } } // If there is no visible layer add first to the map if (!oneVisibleLayer && Object.keys(leafletLayers.baselayers).length > 0) { safeAddLayer(map, leafletLayers.baselayers[Object.keys(layers.baselayers)[0]]); } // Setup the Overlays for (layerName in layers.overlays) { //if (layers.overlays[layerName].type === 'cartodb') { // //} var newOverlayLayer = createLayer(layers.overlays[layerName]); if (!isDefined(newOverlayLayer)) { delete layers.overlays[layerName]; continue; } leafletLayers.overlays[layerName] = newOverlayLayer; // Only add the visible overlays to the map if (layers.overlays[layerName].visible === true) { safeAddLayer(map, leafletLayers.overlays[layerName]); } } // Watch for the base layers leafletScope.$watch('layers.baselayers', function(newBaseLayers, oldBaseLayers) { if (angular.equals(newBaseLayers, oldBaseLayers)) { isLayersControlVisible = updateLayersControl(map, mapId, isLayersControlVisible, newBaseLayers, layers.overlays, leafletLayers); return true; } // Delete layers from the array for (var name in leafletLayers.baselayers) { if (!isDefined(newBaseLayers[name]) || newBaseLayers[name].doRefresh) { // Remove from the map if it's on it if (map.hasLayer(leafletLayers.baselayers[name])) { map.removeLayer(leafletLayers.baselayers[name]); } delete leafletLayers.baselayers[name]; if (newBaseLayers[name] && newBaseLayers[name].doRefresh) { newBaseLayers[name].doRefresh = false; } } } // add new layers for (var newName in newBaseLayers) { if (!isDefined(leafletLayers.baselayers[newName])) { var testBaseLayer = createLayer(newBaseLayers[newName]); if (isDefined(testBaseLayer)) { leafletLayers.baselayers[newName] = testBaseLayer; // Only add the visible layer to the map if (newBaseLayers[newName].top === true) { safeAddLayer(map, leafletLayers.baselayers[newName]); } } } else { if (newBaseLayers[newName].top === true && !map.hasLayer(leafletLayers.baselayers[newName])) { safeAddLayer(map, leafletLayers.baselayers[newName]); } else if (newBaseLayers[newName].top === false && map.hasLayer(leafletLayers.baselayers[newName])) { map.removeLayer(leafletLayers.baselayers[newName]); } } } //we have layers, so we need to make, at least, one active var found = false; // search for an active layer for (var key in leafletLayers.baselayers) { if (map.hasLayer(leafletLayers.baselayers[key])) { found = true; break; } } // If there is no active layer make one active if (!found && Object.keys(leafletLayers.baselayers).length > 0) { safeAddLayer(map, leafletLayers.baselayers[Object.keys(leafletLayers.baselayers)[0]]); } // Only show the layers switch selector control if we have more than one baselayer + overlay isLayersControlVisible = updateLayersControl(map, mapId, isLayersControlVisible, newBaseLayers, layers.overlays, leafletLayers); }, true); // Watch for the overlay layers leafletScope.$watch('layers.overlays', function(newOverlayLayers, oldOverlayLayers) { if (angular.equals(newOverlayLayers, oldOverlayLayers)) { isLayersControlVisible = updateLayersControl(map, mapId, isLayersControlVisible, layers.baselayers, newOverlayLayers, leafletLayers); return true; } // Delete layers from the array for (var name in leafletLayers.overlays) { if (!isDefined(newOverlayLayers[name]) || newOverlayLayers[name].doRefresh) { // Remove from the map if it's on it if (map.hasLayer(leafletLayers.overlays[name])) { // Safe remove when ArcGIS layers is loading. var options = isDefined(newOverlayLayers[name]) ? newOverlayLayers[name].layerOptions : null; safeRemoveLayer(map, leafletLayers.overlays[name], options); } // TODO: Depending on the layer type we will have to delete what's included on it delete leafletLayers.overlays[name]; if (newOverlayLayers[name] && newOverlayLayers[name].doRefresh) { newOverlayLayers[name].doRefresh = false; } } } // add new overlays for (var newName in newOverlayLayers) { if (!isDefined(leafletLayers.overlays[newName])) { var testOverlayLayer = createLayer(newOverlayLayers[newName]); if (!isDefined(testOverlayLayer)) { // If the layer creation fails, continue to the next overlay continue; } leafletLayers.overlays[newName] = testOverlayLayer; if (newOverlayLayers[newName].visible === true) { safeAddLayer(map, leafletLayers.overlays[newName]); } } else { // check for the .visible property to hide/show overLayers if (newOverlayLayers[newName].visible && !map.hasLayer(leafletLayers.overlays[newName])) { safeAddLayer(map, leafletLayers.overlays[newName]); } else if (newOverlayLayers[newName].visible === false && map.hasLayer(leafletLayers.overlays[newName])) { // Safe remove when ArcGIS layers is loading. safeRemoveLayer(map, leafletLayers.overlays[newName], newOverlayLayers[newName].layerOptions); } } //refresh heatmap data if present if (newOverlayLayers[newName].visible && map._loaded && newOverlayLayers[newName].data && newOverlayLayers[newName].type === 'heatmap') { leafletLayers.overlays[newName].setData(newOverlayLayers[newName].data); leafletLayers.overlays[newName].update(); } } // Only add the layers switch selector control if we have more than one baselayer + overlay isLayersControlVisible = updateLayersControl(map, mapId, isLayersControlVisible, layers.baselayers, newOverlayLayers, leafletLayers); }, true); }); }, }; }]); angular.module('leaflet-directive').directive('legend', ["$log", "$http", "leafletHelpers", "leafletLegendHelpers", function($log, $http, leafletHelpers, leafletLegendHelpers) { return { restrict: 'A', scope: false, replace: false, require: 'leaflet', link: function(scope, element, attrs, controller) { var isArray = leafletHelpers.isArray; var isDefined = leafletHelpers.isDefined; var isFunction = leafletHelpers.isFunction; var leafletScope = controller.getLeafletScope(); var legend = leafletScope.legend; var legendClass; var position; var leafletLegend; var type; leafletScope.$watch('legend', function(newLegend) { if (isDefined(newLegend)) { legendClass = newLegend.legendClass ? newLegend.legendClass : 'legend'; position = newLegend.position || 'bottomright'; // default to arcgis type = newLegend.type || 'arcgis'; } }, true); controller.getMap().then(function(map) { leafletScope.$watch('legend', function(newLegend) { if (!isDefined(newLegend)) { if (isDefined(leafletLegend)) { leafletLegend.removeFrom(map); leafletLegend = null; } return; } if (!isDefined(newLegend.url) && (type === 'arcgis') && (!isArray(newLegend.colors) || !isArray(newLegend.labels) || newLegend.colors.length !== newLegend.labels.length)) { $log.warn('[AngularJS - Leaflet] legend.colors and legend.labels must be set.'); return; } if (isDefined(newLegend.url)) { $log.info('[AngularJS - Leaflet] loading legend service.'); return; } if (isDefined(leafletLegend)) { leafletLegend.removeFrom(map); leafletLegend = null; } leafletLegend = L.control({ position: position, }); if (type === 'arcgis') { leafletLegend.onAdd = leafletLegendHelpers.getOnAddArrayLegend(newLegend, legendClass); } leafletLegend.addTo(map); }); leafletScope.$watch('legend.url', function(newURL) { if (!isDefined(newURL)) { return; } $http.get(newURL) .success(function(legendData) { if (isDefined(leafletLegend)) { leafletLegendHelpers.updateLegend(leafletLegend.getContainer(), legendData, type, newURL); } else { leafletLegend = L.control({ position: position, }); leafletLegend.onAdd = leafletLegendHelpers.getOnAddLegend(legendData, legendClass, type, newURL); leafletLegend.addTo(map); } if (isDefined(legend.loadedData) && isFunction(legend.loadedData)) { legend.loadedData(); } }) .error(function() { $log.warn('[AngularJS - Leaflet] legend.url not loaded.'); }); }); }); }, }; }]); angular.module('leaflet-directive').directive('markers', ["$log", "$rootScope", "$q", "leafletData", "leafletHelpers", "leafletMapDefaults", "leafletMarkersHelpers", "leafletMarkerEvents", "leafletIterators", "leafletWatchHelpers", "leafletDirectiveControlsHelpers", function($log, $rootScope, $q, leafletData, leafletHelpers, leafletMapDefaults, leafletMarkersHelpers, leafletMarkerEvents, leafletIterators, leafletWatchHelpers, leafletDirectiveControlsHelpers) { //less terse vars to helpers var isDefined = leafletHelpers.isDefined; var errorHeader = leafletHelpers.errorHeader; var Helpers = leafletHelpers; var isString = leafletHelpers.isString; var addMarkerWatcher = leafletMarkersHelpers.addMarkerWatcher; var updateMarker = leafletMarkersHelpers.updateMarker; var listenMarkerEvents = leafletMarkersHelpers.listenMarkerEvents; var addMarkerToGroup = leafletMarkersHelpers.addMarkerToGroup; var createMarker = leafletMarkersHelpers.createMarker; var deleteMarker = leafletMarkersHelpers.deleteMarker; var $it = leafletIterators; var _markersWatchOptions = leafletHelpers.watchOptions; var maybeWatch = leafletWatchHelpers.maybeWatch; var extendDirectiveControls = leafletDirectiveControlsHelpers.extend; var _getLMarker = function(leafletMarkers, name, maybeLayerName) { if (!Object.keys(leafletMarkers).length) return; if (maybeLayerName && isString(maybeLayerName)) { if (!leafletMarkers[maybeLayerName] || !Object.keys(leafletMarkers[maybeLayerName]).length) return; return leafletMarkers[maybeLayerName][name]; } return leafletMarkers[name]; }; var _setLMarker = function(lObject, leafletMarkers, name, maybeLayerName) { if (maybeLayerName && isString(maybeLayerName)) { if (!isDefined(leafletMarkers[maybeLayerName])) leafletMarkers[maybeLayerName] = {}; leafletMarkers[maybeLayerName][name] = lObject; } else leafletMarkers[name] = lObject; return lObject; }; var _maybeAddMarkerToLayer = function(layerName, layers, model, marker, doIndividualWatch, map) { if (!isString(layerName)) { $log.error(errorHeader + ' A layername must be a string'); return false; } if (!isDefined(layers)) { $log.error(errorHeader + ' You must add layers to the directive if the markers are going to use this functionality.'); return false; } if (!isDefined(layers.overlays) || !isDefined(layers.overlays[layerName])) { $log.error(errorHeader + ' A marker can only be added to a layer of type "group"'); return false; } var layerGroup = layers.overlays[layerName]; if (!(layerGroup instanceof L.LayerGroup || layerGroup instanceof L.FeatureGroup)) { $log.error(errorHeader + ' Adding a marker to an overlay needs a overlay of the type "group" or "featureGroup"'); return false; } // The marker goes to a correct layer group, so first of all we add it layerGroup.addLayer(marker); // The marker is automatically added to the map depending on the visibility // of the layer, so we only have to open the popup if the marker is in the map if (!doIndividualWatch && map.hasLayer(marker) && model.focus === true) { marker.openPopup(); } return true; }; //TODO: move to leafletMarkersHelpers??? or make a new class/function file (leafletMarkersHelpers is large already) var _addMarkers = function(mapId, markersToRender, oldModels, map, layers, leafletMarkers, leafletScope, watchOptions, maybeLayerName, skips) { for (var newName in markersToRender) { if (skips[newName]) continue; if (newName.search('-') !== -1) { $log.error('The marker can\'t use a "-" on his key name: "' + newName + '".'); continue; } var model = Helpers.copy(markersToRender[newName]); var pathToMarker = Helpers.getObjectDotPath(maybeLayerName ? [maybeLayerName, newName] : [newName]); var maybeLMarker = _getLMarker(leafletMarkers, newName, maybeLayerName); if (!isDefined(maybeLMarker)) { //(nmccready) very important to not have model changes when lObject is changed //this might be desirable in some cases but it causes two-way binding to lObject which is not ideal //if it is left as the reference then all changes from oldModel vs newModel are ignored //see _destroy (where modelDiff becomes meaningless if we do not copy here) var marker = createMarker(model); var layerName = (model ? model.layer : undefined) || maybeLayerName; //original way takes pref if (!isDefined(marker)) { $log.error(errorHeader + ' Received invalid data on the marker ' + newName + '.'); continue; } _setLMarker(marker, leafletMarkers, newName, maybeLayerName); // Bind message if (isDefined(model.message)) { marker.bindPopup(model.message, model.popupOptions); } // Add the marker to a cluster group if needed if (isDefined(model.group)) { var groupOptions = isDefined(model.groupOption) ? model.groupOption : null; addMarkerToGroup(marker, model.group, groupOptions, map); } // Show label if defined if (Helpers.LabelPlugin.isLoaded() && isDefined(model.label) && isDefined(model.label.message)) { marker.bindLabel(model.label.message, model.label.options); } // Check if the marker should be added to a layer if (isDefined(model) && (isDefined(model.layer) || isDefined(maybeLayerName))) { var pass = _maybeAddMarkerToLayer(layerName, layers, model, marker, watchOptions.individual.doWatch, map); if (!pass) continue; //something went wrong move on in the loop } else if (!isDefined(model.group)) { // We do not have a layer attr, so the marker goes to the map layer map.addLayer(marker); if (!watchOptions.individual.doWatch && model.focus === true) { marker.openPopup(); } } if (watchOptions.individual.doWatch) { addMarkerWatcher(marker, pathToMarker, leafletScope, layers, map, watchOptions.individual.isDeep); } listenMarkerEvents(marker, model, leafletScope, watchOptions.individual.doWatch, map); leafletMarkerEvents.bindEvents(mapId, marker, pathToMarker, model, leafletScope, layerName); } else { var oldModel = isDefined(oldModel) ? oldModels[newName] : undefined; updateMarker(model, oldModel, maybeLMarker, pathToMarker, leafletScope, layers, map); } } }; var _seeWhatWeAlreadyHave = function(markerModels, oldMarkerModels, lMarkers, isEqual, cb) { var hasLogged = false; var equals = false; var oldMarker; var newMarker; var doCheckOldModel = isDefined(oldMarkerModels); for (var name in lMarkers) { if (!hasLogged) { $log.debug(errorHeader + '[markers] destroy: '); hasLogged = true; } if (doCheckOldModel) { //might want to make the option (in watch options) to disable deep checking //ie the options to only check !== (reference check) instead of angular.equals (slow) newMarker = markerModels[name]; oldMarker = oldMarkerModels[name]; equals = angular.equals(newMarker, oldMarker) && isEqual; } if (!isDefined(markerModels) || !Object.keys(markerModels).length || !isDefined(markerModels[name]) || !Object.keys(markerModels[name]).length || equals) { if (cb && Helpers.isFunction(cb)) cb(newMarker, oldMarker, name); } } }; var _destroy = function(markerModels, oldMarkerModels, lMarkers, map, layers) { _seeWhatWeAlreadyHave(markerModels, oldMarkerModels, lMarkers, false, function(newMarker, oldMarker, lMarkerName) { $log.debug(errorHeader + '[marker] is deleting marker: ' + lMarkerName); deleteMarker(lMarkers[lMarkerName], map, layers); delete lMarkers[lMarkerName]; }); }; var _getNewModelsToSkipp = function(newModels, oldModels, lMarkers) { var skips = {}; _seeWhatWeAlreadyHave(newModels, oldModels, lMarkers, true, function(newMarker, oldMarker, lMarkerName) { $log.debug(errorHeader + '[marker] is already rendered, marker: ' + lMarkerName); skips[lMarkerName] = newMarker; }); return skips; }; return { restrict: 'A', scope: false, replace: false, require: ['leaflet', '?layers'], link: function(scope, element, attrs, controller) { var mapController = controller[0]; var leafletScope = mapController.getLeafletScope(); mapController.getMap().then(function(map) { var leafletMarkers = {}; var getLayers; // If the layers attribute is used, we must wait until the layers are created if (isDefined(controller[1])) { getLayers = controller[1].getLayers; } else { getLayers = function() { var deferred = $q.defer(); deferred.resolve(); return deferred.promise; }; } var watchOptions = leafletScope.markersWatchOptions || _markersWatchOptions; // backwards compat if (isDefined(attrs.watchMarkers)) watchOptions.doWatch = watchOptions.individual.doWatch = (!isDefined(attrs.watchMarkers) || Helpers.isTruthy(attrs.watchMarkers)); var isNested = (isDefined(attrs.markersNested) && Helpers.isTruthy(attrs.markersNested)); getLayers().then(function(layers) { var _clean = function(models, oldModels) { if (isNested) { $it.each(models, function(markerToMaybeDel, layerName) { var oldModel = isDefined(oldModel) ? oldModels[layerName] : undefined; _destroy(markerToMaybeDel, oldModel, leafletMarkers[layerName], map, layers); }); return; } _destroy(models, oldModels, leafletMarkers, map, layers); }; var _create = function(models, oldModels) { _clean(models, oldModels); var skips = null; if (isNested) { $it.each(models, function(markersToAdd, layerName) { var oldModel = isDefined(oldModel) ? oldModels[layerName] : undefined; skips = _getNewModelsToSkipp(models[layerName], oldModel, leafletMarkers[layerName]); _addMarkers(attrs.id, markersToAdd, oldModels, map, layers, leafletMarkers, leafletScope, watchOptions, layerName, skips); }); return; } skips = _getNewModelsToSkipp(models, oldModels, leafletMarkers); _addMarkers(attrs.id, models, oldModels, map, layers, leafletMarkers, leafletScope, watchOptions, undefined, skips); }; extendDirectiveControls(attrs.id, 'markers', _create, _clean); leafletData.setMarkers(leafletMarkers, attrs.id); maybeWatch(leafletScope, 'markers', watchOptions, function(newMarkers, oldMarkers) { _create(newMarkers, oldMarkers); }); }); }); }, }; }]); angular.module('leaflet-directive').directive('maxbounds', ["$log", "leafletMapDefaults", "leafletBoundsHelpers", "leafletHelpers", function($log, leafletMapDefaults, leafletBoundsHelpers, leafletHelpers) { return { restrict: 'A', scope: false, replace: false, require: 'leaflet', link: function(scope, element, attrs, controller) { var leafletScope = controller.getLeafletScope(); var isValidBounds = leafletBoundsHelpers.isValidBounds; var isNumber = leafletHelpers.isNumber; controller.getMap().then(function(map) { leafletScope.$watch('maxbounds', function(maxbounds) { if (!isValidBounds(maxbounds)) { // Unset any previous maxbounds map.setMaxBounds(); return; } var leafletBounds = leafletBoundsHelpers.createLeafletBounds(maxbounds); if (isNumber(maxbounds.pad)) { leafletBounds = leafletBounds.pad(maxbounds.pad); } map.setMaxBounds(leafletBounds); if (!attrs.center && !attrs.lfCenter) { map.fitBounds(leafletBounds); } }); }); }, }; }]); angular.module('leaflet-directive').directive('paths', ["$log", "$q", "leafletData", "leafletMapDefaults", "leafletHelpers", "leafletPathsHelpers", "leafletPathEvents", function($log, $q, leafletData, leafletMapDefaults, leafletHelpers, leafletPathsHelpers, leafletPathEvents) { return { restrict: 'A', scope: false, replace: false, require: ['leaflet', '?layers'], link: function(scope, element, attrs, controller) { var mapController = controller[0]; var isDefined = leafletHelpers.isDefined; var isString = leafletHelpers.isString; var leafletScope = mapController.getLeafletScope(); var paths = leafletScope.paths; var createPath = leafletPathsHelpers.createPath; var bindPathEvents = leafletPathEvents.bindPathEvents; var setPathOptions = leafletPathsHelpers.setPathOptions; mapController.getMap().then(function(map) { var defaults = leafletMapDefaults.getDefaults(attrs.id); var getLayers; // If the layers attribute is used, we must wait until the layers are created if (isDefined(controller[1])) { getLayers = controller[1].getLayers; } else { getLayers = function() { var deferred = $q.defer(); deferred.resolve(); return deferred.promise; }; } if (!isDefined(paths)) { return; } getLayers().then(function(layers) { var leafletPaths = {}; leafletData.setPaths(leafletPaths, attrs.id); // Should we watch for every specific marker on the map? var shouldWatch = (!isDefined(attrs.watchPaths) || attrs.watchPaths === 'true'); // Function for listening every single path once created var watchPathFn = function(leafletPath, name) { var clearWatch = leafletScope.$watch('paths["' + name + '"]', function(pathData, old) { if (!isDefined(pathData)) { if (isDefined(old.layer)) { for (var i in layers.overlays) { var overlay = layers.overlays[i]; overlay.removeLayer(leafletPath); } } map.removeLayer(leafletPath); clearWatch(); return; } setPathOptions(leafletPath, pathData.type, pathData); }, true); }; leafletScope.$watchCollection('paths', function(newPaths) { // Delete paths (by name) from the array for (var name in leafletPaths) { if (!isDefined(newPaths[name])) { map.removeLayer(leafletPaths[name]); delete leafletPaths[name]; } } // Create the new paths for (var newName in newPaths) { if (newName.search('\\$') === 0) { continue; } if (newName.search('-') !== -1) { $log.error('[AngularJS - Leaflet] The path name "' + newName + '" is not valid. It must not include "-" and a number.'); continue; } if (!isDefined(leafletPaths[newName])) { var pathData = newPaths[newName]; var newPath = createPath(newName, newPaths[newName], defaults); // bind popup if defined if (isDefined(newPath) && isDefined(pathData.message)) { newPath.bindPopup(pathData.message, pathData.popupOptions); } // Show label if defined if (leafletHelpers.LabelPlugin.isLoaded() && isDefined(pathData.label) && isDefined(pathData.label.message)) { newPath.bindLabel(pathData.label.message, pathData.label.options); } // Check if the marker should be added to a layer if (isDefined(pathData) && isDefined(pathData.layer)) { if (!isString(pathData.layer)) { $log.error('[AngularJS - Leaflet] A layername must be a string'); continue; } if (!isDefined(layers)) { $log.error('[AngularJS - Leaflet] You must add layers to the directive if the markers are going to use this functionality.'); continue; } if (!isDefined(layers.overlays) || !isDefined(layers.overlays[pathData.layer])) { $log.error('[AngularJS - Leaflet] A path can only be added to a layer of type "group"'); continue; } var layerGroup = layers.overlays[pathData.layer]; if (!(layerGroup instanceof L.LayerGroup || layerGroup instanceof L.FeatureGroup)) { $log.error('[AngularJS - Leaflet] Adding a path to an overlay needs a overlay of the type "group" or "featureGroup"'); continue; } // Listen for changes on the new path leafletPaths[newName] = newPath; // The path goes to a correct layer group, so first of all we add it layerGroup.addLayer(newPath); if (shouldWatch) { watchPathFn(newPath, newName); } else { setPathOptions(newPath, pathData.type, pathData); } } else if (isDefined(newPath)) { // Listen for changes on the new path leafletPaths[newName] = newPath; map.addLayer(newPath); if (shouldWatch) { watchPathFn(newPath, newName); } else { setPathOptions(newPath, pathData.type, pathData); } } bindPathEvents(attrs.id, newPath, newName, pathData, leafletScope); } } }); }); }); }, }; }]); angular.module('leaflet-directive').directive('tiles', ["$log", "leafletData", "leafletMapDefaults", "leafletHelpers", function($log, leafletData, leafletMapDefaults, leafletHelpers) { return { restrict: 'A', scope: false, replace: false, require: 'leaflet', link: function(scope, element, attrs, controller) { var isDefined = leafletHelpers.isDefined; var leafletScope = controller.getLeafletScope(); var tiles = leafletScope.tiles; if (!isDefined(tiles) || !isDefined(tiles.url)) { $log.warn('[AngularJS - Leaflet] The \'tiles\' definition doesn\'t have the \'url\' property.'); return; } controller.getMap().then(function(map) { var defaults = leafletMapDefaults.getDefaults(attrs.id); var tileLayerObj; leafletScope.$watch('tiles', function(tiles, oldtiles) { var tileLayerOptions = defaults.tileLayerOptions; var tileLayerUrl = defaults.tileLayer; // If no valid tiles are in the scope, remove the last layer if (!isDefined(tiles.url) && isDefined(tileLayerObj)) { map.removeLayer(tileLayerObj); return; } // No leafletTiles object defined yet if (!isDefined(tileLayerObj)) { if (isDefined(tiles.options)) { angular.copy(tiles.options, tileLayerOptions); } if (isDefined(tiles.url)) { tileLayerUrl = tiles.url; } if (tiles.type === 'wms') { tileLayerObj = L.tileLayer.wms(tileLayerUrl, tileLayerOptions); } else { tileLayerObj = L.tileLayer(tileLayerUrl, tileLayerOptions); } tileLayerObj.addTo(map); leafletData.setTiles(tileLayerObj, attrs.id); return; } // If the options of the tilelayer is changed, we need to redraw the layer if (isDefined(tiles.url) && isDefined(tiles.options) && (tiles.type !== oldtiles.type || !angular.equals(tiles.options, tileLayerOptions))) { map.removeLayer(tileLayerObj); tileLayerOptions = defaults.tileLayerOptions; angular.copy(tiles.options, tileLayerOptions); tileLayerUrl = tiles.url; if (tiles.type === 'wms') { tileLayerObj = L.tileLayer.wms(tileLayerUrl, tileLayerOptions); } else { tileLayerObj = L.tileLayer(tileLayerUrl, tileLayerOptions); } tileLayerObj.addTo(map); leafletData.setTiles(tileLayerObj, attrs.id); return; } // Only the URL of the layer is changed, update the tiles object if (isDefined(tiles.url)) { tileLayerObj.setUrl(tiles.url); } }, true); }); }, }; }]); /* Create multiple similar directives for watchOptions to support directiveControl instead. (when watches are disabled) NgAnnotate does not work here due to the functional creation */ ['markers', 'geojson'].forEach(function(name) { angular.module('leaflet-directive').directive(name + 'WatchOptions', [ '$log', '$rootScope', '$q', 'leafletData', 'leafletHelpers', function($log, $rootScope, $q, leafletData, leafletHelpers) { var isDefined = leafletHelpers.isDefined, errorHeader = leafletHelpers.errorHeader, isObject = leafletHelpers.isObject, _watchOptions = leafletHelpers.watchOptions; return { restrict: 'A', scope: false, replace: false, require: ['leaflet'], link: function(scope, element, attrs, controller) { var mapController = controller[0], leafletScope = mapController.getLeafletScope(); mapController.getMap().then(function() { if (isDefined(scope[name + 'WatchOptions'])) { if (isObject(scope[name + 'WatchOptions'])) angular.extend(_watchOptions, scope[name + 'WatchOptions']); else $log.error(errorHeader + '[' + name + 'WatchOptions] is not an object'); leafletScope[name + 'WatchOptions'] = _watchOptions; } }); }, }; },]); }); angular.module('leaflet-directive') .factory('LeafletEventsHelpersFactory', ["$rootScope", "$q", "$log", "leafletHelpers", function($rootScope, $q, $log, leafletHelpers) { var safeApply = leafletHelpers.safeApply; var isDefined = leafletHelpers.isDefined; var isObject = leafletHelpers.isObject; var isArray = leafletHelpers.isArray; var errorHeader = leafletHelpers.errorHeader; var EventsHelper = function(rootBroadcastName, lObjectType) { this.rootBroadcastName = rootBroadcastName; $log.debug('LeafletEventsHelpersFactory: lObjectType: ' + lObjectType + 'rootBroadcastName: ' + rootBroadcastName); //used to path/key out certain properties based on the type , "markers", "geojson" this.lObjectType = lObjectType; }; EventsHelper.prototype.getAvailableEvents = function() {return [];}; /* argument: name: Note this can be a single string or dot notation Example: markerModel : { m1: { lat:_, lon: _} } //would yield name of name = "m1" If nested: markerModel : { cars: { m1: { lat:_, lon: _} } } //would yield name of name = "cars.m1" */ EventsHelper.prototype.genDispatchEvent = function(maybeMapId, eventName, logic, leafletScope, lObject, name, model, layerName, extra) { var _this = this; maybeMapId = maybeMapId || ''; if (maybeMapId) maybeMapId = '.' + maybeMapId; return function(e) { var broadcastName = _this.rootBroadcastName + maybeMapId + '.' + eventName; $log.debug(broadcastName); _this.fire(leafletScope, broadcastName, logic, e, e.target || lObject, model, name, layerName, extra); }; }; EventsHelper.prototype.fire = function(scope, broadcastName, logic, event, lObject, model, modelName, layerName) { // Safely broadcast the event safeApply(scope, function() { var toSend = { leafletEvent: event, leafletObject: lObject, modelName: modelName, model: model, }; if (isDefined(layerName)) angular.extend(toSend, {layerName: layerName}); if (logic === 'emit') { scope.$emit(broadcastName, toSend); } else { $rootScope.$broadcast(broadcastName, toSend); } }); }; EventsHelper.prototype.bindEvents = function(maybeMapId, lObject, name, model, leafletScope, layerName, extra) { var events = []; var logic = 'emit'; var _this = this; if (!isDefined(leafletScope.eventBroadcast)) { // Backward compatibility, if no event-broadcast attribute, all events are broadcasted events = this.getAvailableEvents(); } else if (!isObject(leafletScope.eventBroadcast)) { // Not a valid object $log.error(errorHeader + 'event-broadcast must be an object check your model.'); } else { // We have a possible valid object if (!isDefined(leafletScope.eventBroadcast[_this.lObjectType])) { // We do not have events enable/disable do we do nothing (all enabled by default) events = this.getAvailableEvents(); } else if (!isObject(leafletScope.eventBroadcast[_this.lObjectType])) { // Not a valid object $log.warn(errorHeader + 'event-broadcast.' + [_this.lObjectType] + ' must be an object check your model.'); } else { // We have a possible valid map object // Event propadation logic if (isDefined(leafletScope.eventBroadcast[this.lObjectType].logic)) { // We take care of possible propagation logic if (leafletScope.eventBroadcast[_this.lObjectType].logic !== 'emit' && leafletScope.eventBroadcast[_this.lObjectType].logic !== 'broadcast') $log.warn(errorHeader + 'Available event propagation logic are: \'emit\' or \'broadcast\'.'); } // Enable / Disable var eventsEnable = false; var eventsDisable = false; if (isDefined(leafletScope.eventBroadcast[_this.lObjectType].enable) && isArray(leafletScope.eventBroadcast[_this.lObjectType].enable)) eventsEnable = true; if (isDefined(leafletScope.eventBroadcast[_this.lObjectType].disable) && isArray(leafletScope.eventBroadcast[_this.lObjectType].disable)) eventsDisable = true; if (eventsEnable && eventsDisable) { // Both are active, this is an error $log.warn(errorHeader + 'can not enable and disable events at the same time'); } else if (!eventsEnable && !eventsDisable) { // Both are inactive, this is an error $log.warn(errorHeader + 'must enable or disable events'); } else { // At this point the object is OK, lets enable or disable events if (eventsEnable) { // Enable events leafletScope.eventBroadcast[this.lObjectType].enable.forEach(function(eventName) { // Do we have already the event enabled? if (events.indexOf(eventName) !== -1) { // Repeated event, this is an error $log.warn(errorHeader + 'This event ' + eventName + ' is already enabled'); } else { // Does the event exists? if (_this.getAvailableEvents().indexOf(eventName) === -1) { // The event does not exists, this is an error $log.warn(errorHeader + 'This event ' + eventName + ' does not exist'); } else { // All ok enable the event events.push(eventName); } } }); } else { // Disable events events = this.getAvailableEvents(); leafletScope.eventBroadcast[_this.lObjectType].disable.forEach(function(eventName) { var index = events.indexOf(eventName); if (index === -1) { // The event does not exist $log.warn(errorHeader + 'This event ' + eventName + ' does not exist or has been already disabled'); } else { events.splice(index, 1); } }); } } } } events.forEach(function(eventName) { lObject.on(eventName, _this.genDispatchEvent(maybeMapId, eventName, logic, leafletScope, lObject, name, model, layerName, extra)); }); return logic; }; return EventsHelper; }]) .service('leafletEventsHelpers', ["LeafletEventsHelpersFactory", function(LeafletEventsHelpersFactory) { return new LeafletEventsHelpersFactory(); }]); angular.module('leaflet-directive') .factory('leafletGeoJsonEvents', ["$rootScope", "$q", "$log", "leafletHelpers", "LeafletEventsHelpersFactory", "leafletData", function($rootScope, $q, $log, leafletHelpers, LeafletEventsHelpersFactory, leafletData) { var safeApply = leafletHelpers.safeApply; var EventsHelper = LeafletEventsHelpersFactory; var GeoJsonEvents = function() { EventsHelper.call(this, 'leafletDirectiveGeoJson', 'geojson'); }; GeoJsonEvents.prototype = new EventsHelper(); GeoJsonEvents.prototype.genDispatchEvent = function(maybeMapId, eventName, logic, leafletScope, lObject, name, model, layerName, extra) { var base = EventsHelper.prototype.genDispatchEvent.call(this, maybeMapId, eventName, logic, leafletScope, lObject, name, model, layerName); var _this = this; return function(e) { if (eventName === 'mouseout') { if (extra.resetStyleOnMouseout) { leafletData.getGeoJSON(extra.mapId) .then(function(leafletGeoJSON) { //this is broken on nested needs to traverse or user layerName (nested) var lobj = layerName ? leafletGeoJSON[layerName] : leafletGeoJSON; lobj.resetStyle(e.target); }); } safeApply(leafletScope, function() { $rootScope.$broadcast(_this.rootBroadcastName + '.mouseout', e); }); } base(e); //common }; }; GeoJsonEvents.prototype.getAvailableEvents = function() { return [ 'click', 'dblclick', 'mouseover', 'mouseout', ]; }; return new GeoJsonEvents(); }]); angular.module('leaflet-directive') .factory('leafletLabelEvents', ["$rootScope", "$q", "$log", "leafletHelpers", "LeafletEventsHelpersFactory", function($rootScope, $q, $log, leafletHelpers, LeafletEventsHelpersFactory) { var Helpers = leafletHelpers; var EventsHelper = LeafletEventsHelpersFactory; var LabelEvents = function() { EventsHelper.call(this, 'leafletDirectiveLabel', 'markers'); }; LabelEvents.prototype = new EventsHelper(); LabelEvents.prototype.genDispatchEvent = function(maybeMapId, eventName, logic, leafletScope, lObject, name, model, layerName) { var markerName = name.replace('markers.', ''); return EventsHelper.prototype .genDispatchEvent.call(this, maybeMapId, eventName, logic, leafletScope, lObject, markerName, model, layerName); }; LabelEvents.prototype.getAvailableEvents = function() { return [ 'click', 'dblclick', 'mousedown', 'mouseover', 'mouseout', 'contextmenu', ]; }; LabelEvents.prototype.genEvents = function(maybeMapId, eventName, logic, leafletScope, lObject, name, model, layerName) { var _this = this; var labelEvents = this.getAvailableEvents(); var scopeWatchName = Helpers.getObjectArrayPath('markers.' + name); labelEvents.forEach(function(eventName) { lObject.label.on(eventName, _this.genDispatchEvent( maybeMapId, eventName, logic, leafletScope, lObject.label, scopeWatchName, model, layerName)); }); }; LabelEvents.prototype.bindEvents = function() {}; return new LabelEvents(); }]); angular.module('leaflet-directive') .factory('leafletMapEvents', ["$rootScope", "$q", "$log", "leafletHelpers", "leafletEventsHelpers", "leafletIterators", function($rootScope, $q, $log, leafletHelpers, leafletEventsHelpers, leafletIterators) { var isDefined = leafletHelpers.isDefined; var fire = leafletEventsHelpers.fire; var _getAvailableMapEvents = function() { return [ 'click', 'dblclick', 'mousedown', 'mouseup', 'mouseover', 'mouseout', 'mousemove', 'contextmenu', 'focus', 'blur', 'preclick', 'load', 'unload', 'viewreset', 'movestart', 'move', 'moveend', 'dragstart', 'drag', 'dragend', 'zoomstart', 'zoomanim', 'zoomend', 'zoomlevelschange', 'resize', 'autopanstart', 'layeradd', 'layerremove', 'baselayerchange', 'overlayadd', 'overlayremove', 'locationfound', 'locationerror', 'popupopen', 'popupclose', 'draw:created', 'draw:edited', 'draw:deleted', 'draw:drawstart', 'draw:drawstop', 'draw:editstart', 'draw:editstop', 'draw:deletestart', 'draw:deletestop', ]; }; var _genDispatchMapEvent = function(scope, eventName, logic, maybeMapId) { if (maybeMapId) maybeMapId = maybeMapId + '.'; return function(e) { // Put together broadcast name var broadcastName = 'leafletDirectiveMap.' + maybeMapId + eventName; $log.debug(broadcastName); // Safely broadcast the event fire(scope, broadcastName, logic, e, e.target, scope); }; }; var _notifyCenterChangedToBounds = function(scope) { scope.$broadcast('boundsChanged'); }; var _notifyCenterUrlHashChanged = function(scope, map, attrs, search) { if (!isDefined(attrs.urlHashCenter)) { return; } var center = map.getCenter(); var centerUrlHash = (center.lat).toFixed(4) + ':' + (center.lng).toFixed(4) + ':' + map.getZoom(); if (!isDefined(search.c) || search.c !== centerUrlHash) { //$log.debug("notified new center..."); scope.$emit('centerUrlHash', centerUrlHash); } }; var _addEvents = function(map, mapEvents, contextName, scope, logic) { leafletIterators.each(mapEvents, function(eventName) { var context = {}; context[contextName] = eventName; map.on(eventName, _genDispatchMapEvent(scope, eventName, logic, map._container.id || ''), context); }); }; return { getAvailableMapEvents: _getAvailableMapEvents, genDispatchMapEvent: _genDispatchMapEvent, notifyCenterChangedToBounds: _notifyCenterChangedToBounds, notifyCenterUrlHashChanged: _notifyCenterUrlHashChanged, addEvents: _addEvents, }; }]); angular.module('leaflet-directive') .factory('leafletMarkerEvents', ["$rootScope", "$q", "$log", "leafletHelpers", "LeafletEventsHelpersFactory", "leafletLabelEvents", function($rootScope, $q, $log, leafletHelpers, LeafletEventsHelpersFactory, leafletLabelEvents) { var safeApply = leafletHelpers.safeApply; var isDefined = leafletHelpers.isDefined; var Helpers = leafletHelpers; var lblHelp = leafletLabelEvents; var EventsHelper = LeafletEventsHelpersFactory; var MarkerEvents = function() { EventsHelper.call(this, 'leafletDirectiveMarker', 'markers'); }; MarkerEvents.prototype = new EventsHelper(); MarkerEvents.prototype.genDispatchEvent = function(maybeMapId, eventName, logic, leafletScope, lObject, name, model, layerName) { var handle = EventsHelper.prototype .genDispatchEvent.call(this, maybeMapId, eventName, logic, leafletScope, lObject, name, model, layerName); return function(e) { // Broadcast old marker click name for backwards compatibility if (eventName === 'click') { safeApply(leafletScope, function() { $rootScope.$broadcast('leafletDirectiveMarkersClick', name); }); } else if (eventName === 'dragend') { safeApply(leafletScope, function() { model.lat = lObject.getLatLng().lat; model.lng = lObject.getLatLng().lng; }); if (model.message && model.focus === true) { lObject.openPopup(); } } handle(e); //common }; }; MarkerEvents.prototype.getAvailableEvents = function() { return [ 'click', 'dblclick', 'mousedown', 'mouseover', 'mouseout', 'contextmenu', 'dragstart', 'drag', 'dragend', 'move', 'remove', 'popupopen', 'popupclose', 'touchend', 'touchstart', 'touchmove', 'touchcancel', 'touchleave', ]; }; MarkerEvents.prototype.bindEvents = function(maybeMapId, lObject, name, model, leafletScope, layerName) { var logic = EventsHelper.prototype.bindEvents.call(this, maybeMapId, lObject, name, model, leafletScope, layerName); if (Helpers.LabelPlugin.isLoaded() && isDefined(lObject.label)) { lblHelp.genEvents(maybeMapId, name, logic, leafletScope, lObject, model, layerName); } }; return new MarkerEvents(); }]); angular.module('leaflet-directive') .factory('leafletPathEvents', ["$rootScope", "$q", "$log", "leafletHelpers", "leafletLabelEvents", "leafletEventsHelpers", function($rootScope, $q, $log, leafletHelpers, leafletLabelEvents, leafletEventsHelpers) { var isDefined = leafletHelpers.isDefined; var isObject = leafletHelpers.isObject; var Helpers = leafletHelpers; var errorHeader = leafletHelpers.errorHeader; var lblHelp = leafletLabelEvents; var fire = leafletEventsHelpers.fire; /* TODO (nmccready) This EventsHelper needs to be derrived from leafletEventsHelpers to elminate copy and paste code. */ var _genDispatchPathEvent = function(maybeMapId, eventName, logic, leafletScope, lObject, name, model, layerName) { maybeMapId = maybeMapId || ''; if (maybeMapId) maybeMapId = '.' + maybeMapId; return function(e) { var broadcastName = 'leafletDirectivePath' + maybeMapId + '.' + eventName; $log.debug(broadcastName); fire(leafletScope, broadcastName, logic, e, e.target || lObject, model, name, layerName); }; }; var _bindPathEvents = function(maybeMapId, lObject, name, model, leafletScope) { var pathEvents = []; var i; var eventName; var logic = 'broadcast'; if (!isDefined(leafletScope.eventBroadcast)) { // Backward compatibility, if no event-broadcast attribute, all events are broadcasted pathEvents = _getAvailablePathEvents(); } else if (!isObject(leafletScope.eventBroadcast)) { // Not a valid object $log.error(errorHeader + 'event-broadcast must be an object check your model.'); } else { // We have a possible valid object if (!isDefined(leafletScope.eventBroadcast.path)) { // We do not have events enable/disable do we do nothing (all enabled by default) pathEvents = _getAvailablePathEvents(); } else if (isObject(leafletScope.eventBroadcast.paths)) { // Not a valid object $log.warn(errorHeader + 'event-broadcast.path must be an object check your model.'); } else { // We have a possible valid map object // Event propadation logic if (leafletScope.eventBroadcast.path.logic !== undefined && leafletScope.eventBroadcast.path.logic !== null) { // We take care of possible propagation logic if (leafletScope.eventBroadcast.path.logic !== 'emit' && leafletScope.eventBroadcast.path.logic !== 'broadcast') { // This is an error $log.warn(errorHeader + 'Available event propagation logic are: \'emit\' or \'broadcast\'.'); } else if (leafletScope.eventBroadcast.path.logic === 'emit') { logic = 'emit'; } } // Enable / Disable var pathEventsEnable = false; var pathEventsDisable = false; if (leafletScope.eventBroadcast.path.enable !== undefined && leafletScope.eventBroadcast.path.enable !== null) { if (typeof leafletScope.eventBroadcast.path.enable === 'object') { pathEventsEnable = true; } } if (leafletScope.eventBroadcast.path.disable !== undefined && leafletScope.eventBroadcast.path.disable !== null) { if (typeof leafletScope.eventBroadcast.path.disable === 'object') { pathEventsDisable = true; } } if (pathEventsEnable && pathEventsDisable) { // Both are active, this is an error $log.warn(errorHeader + 'can not enable and disable events at the same time'); } else if (!pathEventsEnable && !pathEventsDisable) { // Both are inactive, this is an error $log.warn(errorHeader + 'must enable or disable events'); } else { // At this point the path object is OK, lets enable or disable events if (pathEventsEnable) { // Enable events for (i = 0; i < leafletScope.eventBroadcast.path.enable.length; i++) { eventName = leafletScope.eventBroadcast.path.enable[i]; // Do we have already the event enabled? if (pathEvents.indexOf(eventName) !== -1) { // Repeated event, this is an error $log.warn(errorHeader + 'This event ' + eventName + ' is already enabled'); } else { // Does the event exists? if (_getAvailablePathEvents().indexOf(eventName) === -1) { // The event does not exists, this is an error $log.warn(errorHeader + 'This event ' + eventName + ' does not exist'); } else { // All ok enable the event pathEvents.push(eventName); } } } } else { // Disable events pathEvents = _getAvailablePathEvents(); for (i = 0; i < leafletScope.eventBroadcast.path.disable.length; i++) { eventName = leafletScope.eventBroadcast.path.disable[i]; var index = pathEvents.indexOf(eventName); if (index === -1) { // The event does not exist $log.warn(errorHeader + 'This event ' + eventName + ' does not exist or has been already disabled'); } else { pathEvents.splice(index, 1); } } } } } } for (i = 0; i < pathEvents.length; i++) { eventName = pathEvents[i]; lObject.on(eventName, _genDispatchPathEvent(maybeMapId, eventName, logic, leafletScope, pathEvents, name)); } if (Helpers.LabelPlugin.isLoaded() && isDefined(lObject.label)) { lblHelp.genEvents(maybeMapId, name, logic, leafletScope, lObject, model); } }; var _getAvailablePathEvents = function() { return [ 'click', 'dblclick', 'mousedown', 'mouseover', 'mouseout', 'contextmenu', 'add', 'remove', 'popupopen', 'popupclose', ]; }; return { getAvailablePathEvents: _getAvailablePathEvents, bindPathEvents: _bindPathEvents, }; }]); }(angular));