2876 lines
No EOL
120 KiB
JavaScript
2876 lines
No EOL
120 KiB
JavaScript
(function () {
|
|
'use strict';
|
|
angular.module('leaflet-directive', []).directive('leaflet', [
|
|
'$q',
|
|
'leafletData',
|
|
'leafletMapDefaults',
|
|
'leafletHelpers',
|
|
'leafletEvents',
|
|
function ($q, leafletData, leafletMapDefaults, leafletHelpers, leafletEvents) {
|
|
var _leafletMap;
|
|
return {
|
|
restrict: 'EA',
|
|
replace: true,
|
|
scope: {
|
|
center: '=center',
|
|
defaults: '=defaults',
|
|
maxbounds: '=maxbounds',
|
|
bounds: '=bounds',
|
|
markers: '=markers',
|
|
legend: '=legend',
|
|
geojson: '=geojson',
|
|
paths: '=paths',
|
|
tiles: '=tiles',
|
|
layers: '=layers',
|
|
controls: '=controls',
|
|
eventBroadcast: '=eventBroadcast'
|
|
},
|
|
template: '<div class="angular-leaflet-map"></div>',
|
|
controller: [
|
|
'$scope',
|
|
function ($scope) {
|
|
_leafletMap = $q.defer();
|
|
this.getMap = function () {
|
|
return _leafletMap.promise;
|
|
};
|
|
this.getLeafletScope = function () {
|
|
return $scope;
|
|
};
|
|
}
|
|
],
|
|
link: function (scope, element, attrs) {
|
|
var isDefined = leafletHelpers.isDefined, defaults = leafletMapDefaults.setDefaults(scope.defaults, attrs.id), genDispatchMapEvent = leafletEvents.genDispatchMapEvent, mapEvents = leafletEvents.getAvailableMapEvents();
|
|
// Set width and height if they are defined
|
|
if (isDefined(attrs.width)) {
|
|
if (isNaN(attrs.width)) {
|
|
element.css('width', attrs.width);
|
|
} else {
|
|
element.css('width', attrs.width + 'px');
|
|
}
|
|
}
|
|
if (isDefined(attrs.height)) {
|
|
if (isNaN(attrs.height)) {
|
|
element.css('height', attrs.height);
|
|
} else {
|
|
element.css('height', attrs.height + 'px');
|
|
}
|
|
}
|
|
// Create the Leaflet Map Object with the options
|
|
var map = new L.Map(element[0], leafletMapDefaults.getMapCreationDefaults(attrs.id));
|
|
_leafletMap.resolve(map);
|
|
if (!isDefined(attrs.center)) {
|
|
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';
|
|
for (var i = 0; i < mapEvents.length; i++) {
|
|
var eventName = mapEvents[i];
|
|
map.on(eventName, genDispatchMapEvent(scope, eventName, logic), { eventName: eventName });
|
|
}
|
|
}
|
|
// Resolve the map object to the promises
|
|
map.whenReady(function () {
|
|
leafletData.setMap(map, attrs.id);
|
|
});
|
|
scope.$on('$destroy', function () {
|
|
leafletData.unresolveMap(attrs.id);
|
|
});
|
|
}
|
|
};
|
|
}
|
|
]);
|
|
angular.module('leaflet-directive').directive('center', [
|
|
'$log',
|
|
'$q',
|
|
'$location',
|
|
'leafletMapDefaults',
|
|
'leafletHelpers',
|
|
'leafletBoundsHelpers',
|
|
'leafletEvents',
|
|
function ($log, $q, $location, leafletMapDefaults, leafletHelpers, leafletBoundsHelpers, leafletEvents) {
|
|
var isDefined = leafletHelpers.isDefined, isNumber = leafletHelpers.isNumber, isSameCenterOnMap = leafletHelpers.isSameCenterOnMap, safeApply = leafletHelpers.safeApply, isValidCenter = leafletHelpers.isValidCenter, isEmpty = leafletHelpers.isEmpty, isUndefinedOrEmpty = leafletHelpers.isUndefinedOrEmpty;
|
|
var shouldInitializeMapWithBounds = function (bounds, center) {
|
|
return isDefined(bounds) && !isEmpty(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(), centerModel = leafletScope.center;
|
|
controller.getMap().then(function (map) {
|
|
var defaults = leafletMapDefaults.getDefaults(attrs.id);
|
|
if (attrs.center.search('-') !== -1) {
|
|
$log.error('The "center" variable can\'t use a "-" on his key name: "' + attrs.center + '".');
|
|
map.setView([
|
|
defaults.center.lat,
|
|
defaults.center.lng
|
|
], defaults.center.zoom);
|
|
return;
|
|
} else if (shouldInitializeMapWithBounds(leafletScope.bounds, centerModel)) {
|
|
map.fitBounds(leafletBoundsHelpers.createLeafletBounds(leafletScope.bounds));
|
|
centerModel = map.getCenter();
|
|
safeApply(leafletScope, function (scope) {
|
|
scope.center = {
|
|
lat: map.getCenter().lat,
|
|
lng: map.getCenter().lng,
|
|
zoom: map.getZoom(),
|
|
autoDiscover: false
|
|
};
|
|
});
|
|
safeApply(leafletScope, function (scope) {
|
|
var mapBounds = map.getBounds();
|
|
var newScopeBounds = {
|
|
northEast: {
|
|
lat: mapBounds._northEast.lat,
|
|
lng: mapBounds._northEast.lng
|
|
},
|
|
southWest: {
|
|
lat: mapBounds._southWest.lat,
|
|
lng: mapBounds._southWest.lng
|
|
}
|
|
};
|
|
scope.bounds = newScopeBounds;
|
|
});
|
|
} else if (!isDefined(centerModel)) {
|
|
$log.error('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, 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);
|
|
scope.center = {
|
|
lat: urlCenter.lat,
|
|
lng: urlCenter.lng,
|
|
zoom: urlCenter.zoom
|
|
};
|
|
}
|
|
});
|
|
}
|
|
leafletScope.$watch('center', function (center) {
|
|
//$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('[AngularJS - Leaflet] 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);
|
|
map.setView([
|
|
center.lat,
|
|
center.lng
|
|
], center.zoom);
|
|
leafletEvents.notifyCenterChangedToBounds(leafletScope, map);
|
|
}, true);
|
|
map.whenReady(function () {
|
|
mapReady = true;
|
|
});
|
|
map.on('moveend', function () {
|
|
// Resolve the center after the first map position
|
|
_leafletCenter.resolve();
|
|
leafletEvents.notifyCenterUrlHashChanged(leafletScope, map, attrs, $location.search());
|
|
//$log.debug("updated center on map...");
|
|
if (isSameCenterOnMap(centerModel, map)) {
|
|
//$log.debug("same center in model, no need to update again.");
|
|
return;
|
|
}
|
|
safeApply(leafletScope, function (scope) {
|
|
//$log.debug("updating center model...", map.getCenter(), map.getZoom());
|
|
scope.center = {
|
|
lat: map.getCenter().lat,
|
|
lng: map.getCenter().lng,
|
|
zoom: map.getZoom(),
|
|
autoDiscover: false
|
|
};
|
|
leafletEvents.notifyCenterChangedToBounds(leafletScope, map);
|
|
});
|
|
});
|
|
if (centerModel.autoDiscover === true) {
|
|
map.on('locationerror', function () {
|
|
$log.warn('[AngularJS - Leaflet] The Geolocation API is unauthorized on this page.');
|
|
if (isValidCenter(centerModel)) {
|
|
map.setView([
|
|
centerModel.lat,
|
|
centerModel.lng
|
|
], centerModel.zoom);
|
|
leafletEvents.notifyCenterChangedToBounds(leafletScope, map);
|
|
} else {
|
|
map.setView([
|
|
defaults.center.lat,
|
|
defaults.center.lng
|
|
], defaults.center.zoom);
|
|
leafletEvents.notifyCenterChangedToBounds(leafletScope, map);
|
|
}
|
|
});
|
|
}
|
|
});
|
|
}
|
|
};
|
|
}
|
|
]);
|
|
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, leafletScope = controller.getLeafletScope(), 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) {
|
|
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;
|
|
}
|
|
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) && !angular.equals(tiles.options, tileLayerOptions)) {
|
|
map.removeLayer(tileLayerObj);
|
|
tileLayerOptions = defaults.tileLayerOptions;
|
|
angular.copy(tiles.options, tileLayerOptions);
|
|
tileLayerUrl = tiles.url;
|
|
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);
|
|
});
|
|
}
|
|
};
|
|
}
|
|
]);
|
|
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, isDefined = leafletHelpers.isDefined, isFunction = leafletHelpers.isFunction, leafletScope = controller.getLeafletScope(), legend = leafletScope.legend;
|
|
var legendClass = legend.legendClass ? legend.legendClass : 'legend';
|
|
var position = legend.position || 'bottomright';
|
|
var leafletLegend;
|
|
controller.getMap().then(function (map) {
|
|
if (!isDefined(legend.url) && (!isArray(legend.colors) || !isArray(legend.labels) || legend.colors.length !== legend.labels.length)) {
|
|
$log.warn('[AngularJS - Leaflet] legend.colors and legend.labels must be set.');
|
|
} else if (isDefined(legend.url)) {
|
|
$log.info('[AngularJS - Leaflet] loading arcgis legend service.');
|
|
} else {
|
|
// TODO: Watch array legend.
|
|
leafletLegend = L.control({ position: position });
|
|
leafletLegend.onAdd = leafletLegendHelpers.getOnAddArrayLegend(legend, legendClass);
|
|
leafletLegend.addTo(map);
|
|
}
|
|
leafletScope.$watch('legend.url', function (newURL) {
|
|
if (!isDefined(newURL)) {
|
|
return;
|
|
}
|
|
$http.get(newURL).success(function (legendData) {
|
|
if (isDefined(leafletLegend)) {
|
|
leafletLegendHelpers.updateArcGISLegend(leafletLegend.getContainer(), legendData);
|
|
} else {
|
|
leafletLegend = L.control({ position: position });
|
|
leafletLegend.onAdd = leafletLegendHelpers.getOnAddArcGISLegend(legendData, legendClass);
|
|
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('geojson', [
|
|
'$log',
|
|
'$rootScope',
|
|
'leafletData',
|
|
'leafletHelpers',
|
|
function ($log, $rootScope, leafletData, leafletHelpers) {
|
|
return {
|
|
restrict: 'A',
|
|
scope: false,
|
|
replace: false,
|
|
require: 'leaflet',
|
|
link: function (scope, element, attrs, controller) {
|
|
var safeApply = leafletHelpers.safeApply, isDefined = leafletHelpers.isDefined, leafletScope = controller.getLeafletScope(), leafletGeoJSON = {};
|
|
controller.getMap().then(function (map) {
|
|
leafletScope.$watch('geojson', function (geojson) {
|
|
if (isDefined(leafletGeoJSON) && map.hasLayer(leafletGeoJSON)) {
|
|
map.removeLayer(leafletGeoJSON);
|
|
}
|
|
if (!(isDefined(geojson) && isDefined(geojson.data))) {
|
|
return;
|
|
}
|
|
var resetStyleOnMouseout = geojson.resetStyleOnMouseout, onEachFeature = geojson.onEachFeature;
|
|
if (!onEachFeature) {
|
|
onEachFeature = function (feature, layer) {
|
|
if (leafletHelpers.LabelPlugin.isLoaded() && isDefined(geojson.label)) {
|
|
layer.bindLabel(feature.properties.description);
|
|
}
|
|
layer.on({
|
|
mouseover: function (e) {
|
|
safeApply(leafletScope, function () {
|
|
geojson.selected = feature;
|
|
$rootScope.$broadcast('leafletDirectiveMap.geojsonMouseover', e);
|
|
});
|
|
},
|
|
mouseout: function (e) {
|
|
if (resetStyleOnMouseout) {
|
|
leafletGeoJSON.resetStyle(e.target);
|
|
}
|
|
safeApply(leafletScope, function () {
|
|
geojson.selected = undefined;
|
|
$rootScope.$broadcast('leafletDirectiveMap.geojsonMouseout', e);
|
|
});
|
|
},
|
|
click: function (e) {
|
|
safeApply(leafletScope, function () {
|
|
geojson.selected = feature;
|
|
$rootScope.$broadcast('leafletDirectiveMap.geojsonClick', geojson.selected, e);
|
|
});
|
|
}
|
|
});
|
|
};
|
|
}
|
|
geojson.options = {
|
|
style: geojson.style,
|
|
filter: geojson.filter,
|
|
onEachFeature: onEachFeature,
|
|
pointToLayer: geojson.pointToLayer
|
|
};
|
|
leafletGeoJSON = L.geoJson(geojson.data, geojson.options);
|
|
leafletData.setGeoJSON(leafletGeoJSON);
|
|
leafletGeoJSON.addTo(map);
|
|
});
|
|
});
|
|
}
|
|
};
|
|
}
|
|
]);
|
|
angular.module('leaflet-directive').directive('layers', [
|
|
'$log',
|
|
'$q',
|
|
'leafletData',
|
|
'leafletHelpers',
|
|
'leafletLayerHelpers',
|
|
'leafletControlHelpers',
|
|
function ($log, $q, leafletData, leafletHelpers, leafletLayerHelpers, leafletControlHelpers) {
|
|
var _leafletLayers;
|
|
return {
|
|
restrict: 'A',
|
|
scope: false,
|
|
replace: false,
|
|
require: 'leaflet',
|
|
controller: function () {
|
|
_leafletLayers = $q.defer();
|
|
this.getLayers = function () {
|
|
return _leafletLayers.promise;
|
|
};
|
|
},
|
|
link: function (scope, element, attrs, controller) {
|
|
var isDefined = leafletHelpers.isDefined, leafletLayers = {}, leafletScope = controller.getLeafletScope(), layers = leafletScope.layers, createLayer = leafletLayerHelpers.createLayer, updateLayersControl = leafletControlHelpers.updateLayersControl, isLayersControlVisible = false;
|
|
controller.getMap().then(function (map) {
|
|
// Do we have a baselayers property?
|
|
if (!isDefined(layers) || !isDefined(layers.baselayers) || Object.keys(layers.baselayers).length === 0) {
|
|
// No baselayers property
|
|
$log.error('[AngularJS - Leaflet] At least one baselayer has to be defined');
|
|
return;
|
|
}
|
|
// We have baselayers to add to the map
|
|
_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) {
|
|
map.addLayer(leafletLayers.baselayers[layerName]);
|
|
oneVisibleLayer = true;
|
|
}
|
|
}
|
|
// If there is no visible layer add first to the map
|
|
if (!oneVisibleLayer && Object.keys(leafletLayers.baselayers).length > 0) {
|
|
map.addLayer(leafletLayers.baselayers[Object.keys(layers.baselayers)[0]]);
|
|
}
|
|
// Setup the Overlays
|
|
for (layerName in layers.overlays) {
|
|
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) {
|
|
map.addLayer(leafletLayers.overlays[layerName]);
|
|
}
|
|
}
|
|
// Watch for the base layers
|
|
leafletScope.$watch('layers.baselayers', function (newBaseLayers) {
|
|
// Delete layers from the array
|
|
for (var name in leafletLayers.baselayers) {
|
|
if (!isDefined(newBaseLayers[name])) {
|
|
// Remove from the map if it's on it
|
|
if (map.hasLayer(leafletLayers.baselayers[name])) {
|
|
map.removeLayer(leafletLayers.baselayers[name]);
|
|
}
|
|
delete leafletLayers.baselayers[name];
|
|
}
|
|
}
|
|
// 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) {
|
|
map.addLayer(leafletLayers.baselayers[newName]);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if (Object.keys(leafletLayers.baselayers).length === 0) {
|
|
$log.error('[AngularJS - Leaflet] At least one baselayer has to be defined');
|
|
return;
|
|
}
|
|
//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) {
|
|
map.addLayer(leafletLayers.baselayers[Object.keys(layers.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) {
|
|
// Delete layers from the array
|
|
for (var name in leafletLayers.overlays) {
|
|
if (!isDefined(newOverlayLayers[name])) {
|
|
// Remove from the map if it's on it
|
|
if (map.hasLayer(leafletLayers.overlays[name])) {
|
|
map.removeLayer(leafletLayers.overlays[name]);
|
|
}
|
|
// TODO: Depending on the layer type we will have to delete what's included on it
|
|
delete leafletLayers.overlays[name];
|
|
}
|
|
}
|
|
// add new overlays
|
|
for (var newName in newOverlayLayers) {
|
|
if (!isDefined(leafletLayers.overlays[newName])) {
|
|
var testOverlayLayer = createLayer(newOverlayLayers[newName]);
|
|
if (isDefined(testOverlayLayer)) {
|
|
leafletLayers.overlays[newName] = testOverlayLayer;
|
|
if (newOverlayLayers[newName].visible === true) {
|
|
map.addLayer(leafletLayers.overlays[newName]);
|
|
}
|
|
}
|
|
}
|
|
// check for the .visible property to hide/show overLayers
|
|
if (newOverlayLayers[newName].visible && !map.hasLayer(leafletLayers.overlays[newName])) {
|
|
map.addLayer(leafletLayers.overlays[newName]);
|
|
} else if (newOverlayLayers[newName].visible === false && map.hasLayer(leafletLayers.overlays[newName])) {
|
|
map.removeLayer(leafletLayers.overlays[newName]);
|
|
}
|
|
}
|
|
// 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('bounds', [
|
|
'$log',
|
|
'$timeout',
|
|
'leafletHelpers',
|
|
'leafletBoundsHelpers',
|
|
function ($log, $timeout, leafletHelpers, leafletBoundsHelpers) {
|
|
return {
|
|
restrict: 'A',
|
|
scope: false,
|
|
replace: false,
|
|
require: [
|
|
'leaflet',
|
|
'center'
|
|
],
|
|
link: function (scope, element, attrs, controller) {
|
|
var isDefined = leafletHelpers.isDefined, createLeafletBounds = leafletBoundsHelpers.createLeafletBounds, leafletScope = controller[0].getLeafletScope(), mapController = controller[0];
|
|
var emptyBounds = function (bounds) {
|
|
if (bounds._southWest.lat === 0 && bounds._southWest.lng === 0 && bounds._northEast.lat === 0 && bounds._northEast.lng === 0) {
|
|
return true;
|
|
}
|
|
return false;
|
|
};
|
|
mapController.getMap().then(function (map) {
|
|
leafletScope.$on('boundsChanged', function (event) {
|
|
var scope = event.currentScope;
|
|
var bounds = map.getBounds();
|
|
//$log.debug('updated map bounds...', bounds);
|
|
if (emptyBounds(bounds)) {
|
|
return;
|
|
}
|
|
var newScopeBounds = {
|
|
northEast: {
|
|
lat: bounds._northEast.lat,
|
|
lng: bounds._northEast.lng
|
|
},
|
|
southWest: {
|
|
lat: bounds._southWest.lat,
|
|
lng: bounds._southWest.lng
|
|
}
|
|
};
|
|
if (!angular.equals(scope.bounds, newScopeBounds)) {
|
|
//$log.debug('Need to update scope bounds.');
|
|
scope.bounds = newScopeBounds;
|
|
}
|
|
});
|
|
leafletScope.$watch('bounds', function (bounds) {
|
|
//$log.debug('updated bounds...', bounds);
|
|
if (!isDefined(bounds)) {
|
|
$log.error('[AngularJS - Leaflet] Invalid bounds');
|
|
return;
|
|
}
|
|
var leafletBounds = createLeafletBounds(bounds);
|
|
if (leafletBounds && !map.getBounds().equals(leafletBounds)) {
|
|
//$log.debug('Need to update map bounds.');
|
|
map.fitBounds(leafletBounds);
|
|
}
|
|
}, true);
|
|
});
|
|
}
|
|
};
|
|
}
|
|
]);
|
|
angular.module('leaflet-directive').directive('markers', [
|
|
'$log',
|
|
'$rootScope',
|
|
'$q',
|
|
'leafletData',
|
|
'leafletHelpers',
|
|
'leafletMapDefaults',
|
|
'leafletMarkersHelpers',
|
|
'leafletEvents',
|
|
function ($log, $rootScope, $q, leafletData, leafletHelpers, leafletMapDefaults, leafletMarkersHelpers, leafletEvents) {
|
|
return {
|
|
restrict: 'A',
|
|
scope: false,
|
|
replace: false,
|
|
require: [
|
|
'leaflet',
|
|
'?layers'
|
|
],
|
|
link: function (scope, element, attrs, controller) {
|
|
var mapController = controller[0], Helpers = leafletHelpers, isDefined = leafletHelpers.isDefined, isString = leafletHelpers.isString, leafletScope = mapController.getLeafletScope(), markers = leafletScope.markers, deleteMarker = leafletMarkersHelpers.deleteMarker, addMarkerWatcher = leafletMarkersHelpers.addMarkerWatcher, listenMarkerEvents = leafletMarkersHelpers.listenMarkerEvents, addMarkerToGroup = leafletMarkersHelpers.addMarkerToGroup, bindMarkerEvents = leafletEvents.bindMarkerEvents, createMarker = leafletMarkersHelpers.createMarker;
|
|
mapController.getMap().then(function (map) {
|
|
var leafletMarkers = {}, 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(markers)) {
|
|
return;
|
|
}
|
|
getLayers().then(function (layers) {
|
|
leafletData.setMarkers(leafletMarkers, attrs.id);
|
|
leafletScope.$watch('markers', function (newMarkers) {
|
|
// Delete markers from the array
|
|
for (var name in leafletMarkers) {
|
|
if (!isDefined(newMarkers) || !isDefined(newMarkers[name])) {
|
|
deleteMarker(leafletMarkers[name], map, layers);
|
|
delete leafletMarkers[name];
|
|
}
|
|
}
|
|
// add new markers
|
|
for (var newName in newMarkers) {
|
|
if (newName.search('-') !== -1) {
|
|
$log.error('The marker can\'t use a "-" on his key name: "' + newName + '".');
|
|
continue;
|
|
}
|
|
if (!isDefined(leafletMarkers[newName])) {
|
|
var markerData = newMarkers[newName];
|
|
var marker = createMarker(markerData);
|
|
if (!isDefined(marker)) {
|
|
$log.error('[AngularJS - Leaflet] Received invalid data on the marker ' + newName + '.');
|
|
continue;
|
|
}
|
|
leafletMarkers[newName] = marker;
|
|
// Bind message
|
|
if (isDefined(markerData.message)) {
|
|
marker.bindPopup(markerData.message, markerData.popupOptions);
|
|
}
|
|
// Add the marker to a cluster group if needed
|
|
if (isDefined(markerData.group)) {
|
|
addMarkerToGroup(marker, markerData.group, map);
|
|
}
|
|
// Show label if defined
|
|
if (Helpers.LabelPlugin.isLoaded() && isDefined(markerData.label) && isDefined(markerData.label.message)) {
|
|
marker.bindLabel(markerData.label.message, markerData.label.options);
|
|
}
|
|
// Check if the marker should be added to a layer
|
|
if (isDefined(markerData) && isDefined(markerData.layer)) {
|
|
if (!isString(markerData.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[markerData.layer])) {
|
|
$log.error('[AngularJS - Leaflet] A marker can only be added to a layer of type "group"');
|
|
continue;
|
|
}
|
|
var layerGroup = layers.overlays[markerData.layer];
|
|
if (!(layerGroup instanceof L.LayerGroup)) {
|
|
$log.error('[AngularJS - Leaflet] Adding a marker to an overlay needs a overlay of the type "group"');
|
|
continue;
|
|
}
|
|
// 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();
|
|
} // Add the marker to the map if it hasn't been added to a layer or to a group
|
|
} else if (!isDefined(markerData.group)) {
|
|
// We do not have a layer attr, so the marker goes to the map layer
|
|
map.addLayer(marker);
|
|
if (markerData.focus === true) {
|
|
marker.openPopup();
|
|
}
|
|
if (Helpers.LabelPlugin.isLoaded() && isDefined(markerData.label) && isDefined(markerData.label.options) && markerData.label.options.noHide === true) {
|
|
marker.showLabel();
|
|
}
|
|
}
|
|
// Should we watch for every specific marker on the map?
|
|
var shouldWatch = !isDefined(attrs.watchMarkers) || attrs.watchMarkers === 'true';
|
|
if (shouldWatch) {
|
|
addMarkerWatcher(marker, newName, leafletScope, layers, map);
|
|
listenMarkerEvents(marker, markerData, leafletScope);
|
|
}
|
|
bindMarkerEvents(marker, newName, markerData, leafletScope);
|
|
}
|
|
}
|
|
}, true);
|
|
});
|
|
});
|
|
}
|
|
};
|
|
}
|
|
]);
|
|
angular.module('leaflet-directive').directive('paths', [
|
|
'$log',
|
|
'leafletData',
|
|
'leafletMapDefaults',
|
|
'leafletHelpers',
|
|
'leafletPathsHelpers',
|
|
'leafletEvents',
|
|
function ($log, leafletData, leafletMapDefaults, leafletHelpers, leafletPathsHelpers, leafletEvents) {
|
|
return {
|
|
restrict: 'A',
|
|
scope: false,
|
|
replace: false,
|
|
require: 'leaflet',
|
|
link: function (scope, element, attrs, controller) {
|
|
var isDefined = leafletHelpers.isDefined, leafletScope = controller.getLeafletScope(), paths = leafletScope.paths, createPath = leafletPathsHelpers.createPath, bindPathEvents = leafletEvents.bindPathEvents, setPathOptions = leafletPathsHelpers.setPathOptions;
|
|
controller.getMap().then(function (map) {
|
|
var defaults = leafletMapDefaults.getDefaults(attrs.id);
|
|
if (!isDefined(paths)) {
|
|
return;
|
|
}
|
|
var leafletPaths = {};
|
|
leafletData.setPaths(leafletPaths, attrs.id);
|
|
// Function for listening every single path once created
|
|
var watchPathFn = function (leafletPath, name) {
|
|
var clearWatch = leafletScope.$watch('paths.' + name, function (pathData) {
|
|
if (!isDefined(pathData)) {
|
|
map.removeLayer(leafletPath);
|
|
clearWatch();
|
|
return;
|
|
}
|
|
setPathOptions(leafletPath, pathData.type, pathData);
|
|
}, true);
|
|
};
|
|
leafletScope.$watch('paths', function (newPaths) {
|
|
// Create the new paths
|
|
for (var newName in newPaths) {
|
|
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);
|
|
}
|
|
// Show label if defined
|
|
if (leafletHelpers.LabelPlugin.isLoaded() && isDefined(pathData.label) && isDefined(pathData.label.message)) {
|
|
newPath.bindLabel(pathData.label.message, pathData.label.options);
|
|
}
|
|
// Listen for changes on the new path
|
|
if (isDefined(newPath)) {
|
|
leafletPaths[newName] = newPath;
|
|
map.addLayer(newPath);
|
|
watchPathFn(newPath, newName);
|
|
}
|
|
bindPathEvents(newPath, newName, pathData, leafletScope);
|
|
}
|
|
}
|
|
// Delete paths (by name) from the array
|
|
for (var name in leafletPaths) {
|
|
if (!isDefined(newPaths[name])) {
|
|
delete leafletPaths[name];
|
|
}
|
|
}
|
|
}, true);
|
|
});
|
|
}
|
|
};
|
|
}
|
|
]);
|
|
angular.module('leaflet-directive').directive('controls', [
|
|
'$log',
|
|
'leafletHelpers',
|
|
function ($log, leafletHelpers) {
|
|
return {
|
|
restrict: 'A',
|
|
scope: false,
|
|
replace: false,
|
|
require: '?^leaflet',
|
|
link: function (scope, element, attrs, controller) {
|
|
if (!controller) {
|
|
return;
|
|
}
|
|
var isDefined = leafletHelpers.isDefined, leafletScope = controller.getLeafletScope(), controls = leafletScope.controls;
|
|
controller.getMap().then(function (map) {
|
|
if (isDefined(L.Control.Draw) && isDefined(controls.draw)) {
|
|
var drawnItems = new L.FeatureGroup();
|
|
map.addLayer(drawnItems);
|
|
var options = { edit: { featureGroup: drawnItems } };
|
|
angular.extend(options, controls.draw.options);
|
|
var drawControl = new L.Control.Draw(options);
|
|
map.addControl(drawControl);
|
|
}
|
|
if (isDefined(controls.custom)) {
|
|
for (var i in controls.custom) {
|
|
map.addControl(controls.custom[i]);
|
|
}
|
|
}
|
|
});
|
|
}
|
|
};
|
|
}
|
|
]);
|
|
angular.module('leaflet-directive').directive('eventBroadcast', [
|
|
'$log',
|
|
'$rootScope',
|
|
'leafletHelpers',
|
|
'leafletEvents',
|
|
function ($log, $rootScope, leafletHelpers, leafletEvents) {
|
|
return {
|
|
restrict: 'A',
|
|
scope: false,
|
|
replace: false,
|
|
require: 'leaflet',
|
|
link: function (scope, element, attrs, controller) {
|
|
var isObject = leafletHelpers.isObject, leafletScope = controller.getLeafletScope(), eventBroadcast = leafletScope.eventBroadcast, availableMapEvents = leafletEvents.getAvailableMapEvents(), genDispatchMapEvent = leafletEvents.genDispatchMapEvent;
|
|
controller.getMap().then(function (map) {
|
|
var mapEvents = [];
|
|
var i;
|
|
var eventName;
|
|
var logic = 'broadcast';
|
|
if (isObject(eventBroadcast)) {
|
|
// We have a possible valid object
|
|
if (eventBroadcast.map === undefined || eventBroadcast.map === null) {
|
|
// We do not have events enable/disable do we do nothing (all enabled by default)
|
|
mapEvents = availableMapEvents;
|
|
} else if (typeof eventBroadcast.map !== 'object') {
|
|
// 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 !== undefined && eventBroadcast.map.logic !== null) {
|
|
// We take care of possible propagation 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 if (eventBroadcast.map.logic === 'emit') {
|
|
logic = 'emit';
|
|
}
|
|
}
|
|
// Enable / Disable
|
|
var mapEventsEnable = false, mapEventsDisable = false;
|
|
if (eventBroadcast.map.enable !== undefined && eventBroadcast.map.enable !== null) {
|
|
if (typeof eventBroadcast.map.enable === 'object') {
|
|
mapEventsEnable = true;
|
|
}
|
|
}
|
|
if (eventBroadcast.map.disable !== undefined && eventBroadcast.map.disable !== null) {
|
|
if (typeof eventBroadcast.map.disable === 'object') {
|
|
mapEventsDisable = true;
|
|
}
|
|
}
|
|
if (mapEventsEnable && mapEventsDisable) {
|
|
// Both are active, this is an error
|
|
$log.warn('[AngularJS - Leaflet] can not enable and disable events at the time');
|
|
} else if (!mapEventsEnable && !mapEventsDisable) {
|
|
// Both are inactive, this is an error
|
|
$log.warn('[AngularJS - Leaflet] must enable or disable events');
|
|
} else {
|
|
// At this point the map object is OK, lets enable or disable events
|
|
if (mapEventsEnable) {
|
|
// Enable events
|
|
for (i = 0; i < eventBroadcast.map.enable.length; i++) {
|
|
eventName = eventBroadcast.map.enable[i];
|
|
// Do we have already the event enabled?
|
|
if (mapEvents.indexOf(eventName) !== -1) {
|
|
// Repeated event, this is an error
|
|
$log.warn('[AngularJS - Leaflet] This event ' + eventName + ' is already enabled');
|
|
} else {
|
|
// Does the event exists?
|
|
if (availableMapEvents.indexOf(eventName) === -1) {
|
|
// The event does not exists, this is an error
|
|
$log.warn('[AngularJS - Leaflet] This event ' + eventName + ' does not exist');
|
|
} else {
|
|
// All ok enable the event
|
|
mapEvents.push(eventName);
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
// Disable events
|
|
mapEvents = availableMapEvents;
|
|
for (i = 0; i < eventBroadcast.map.disable.length; i++) {
|
|
eventName = eventBroadcast.map.disable[i];
|
|
var index = mapEvents.indexOf(eventName);
|
|
if (index === -1) {
|
|
// The event does not exist
|
|
$log.warn('[AngularJS - Leaflet] This event ' + eventName + ' does not exist or has been already disabled');
|
|
} else {
|
|
mapEvents.splice(index, 1);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
for (i = 0; i < mapEvents.length; i++) {
|
|
eventName = mapEvents[i];
|
|
map.on(eventName, genDispatchMapEvent(leafletScope, eventName, logic), { eventName: eventName });
|
|
}
|
|
} else {
|
|
// Not a valid object
|
|
$log.warn('[AngularJS - Leaflet] event-broadcast must be an object, check your model.');
|
|
}
|
|
});
|
|
}
|
|
};
|
|
}
|
|
]);
|
|
angular.module('leaflet-directive').directive('maxbounds', [
|
|
'$log',
|
|
'leafletMapDefaults',
|
|
'leafletBoundsHelpers',
|
|
function ($log, leafletMapDefaults, leafletBoundsHelpers) {
|
|
return {
|
|
restrict: 'A',
|
|
scope: false,
|
|
replace: false,
|
|
require: 'leaflet',
|
|
link: function (scope, element, attrs, controller) {
|
|
var leafletScope = controller.getLeafletScope(), isValidBounds = leafletBoundsHelpers.isValidBounds;
|
|
controller.getMap().then(function (map) {
|
|
leafletScope.$watch('maxbounds', function (maxbounds) {
|
|
if (!isValidBounds(maxbounds)) {
|
|
// Unset any previous maxbounds
|
|
map.setMaxBounds();
|
|
return;
|
|
}
|
|
var bounds = [
|
|
[
|
|
maxbounds.southWest.lat,
|
|
maxbounds.southWest.lng
|
|
],
|
|
[
|
|
maxbounds.northEast.lat,
|
|
maxbounds.northEast.lng
|
|
]
|
|
];
|
|
map.setMaxBounds(bounds);
|
|
map.fitBounds(bounds);
|
|
});
|
|
});
|
|
}
|
|
};
|
|
}
|
|
]);
|
|
angular.module('leaflet-directive').service('leafletData', [
|
|
'$log',
|
|
'$q',
|
|
'leafletHelpers',
|
|
function ($log, $q, leafletHelpers) {
|
|
var getDefer = leafletHelpers.getDefer, getUnresolvedDefer = leafletHelpers.getUnresolvedDefer, setResolvedDefer = leafletHelpers.setResolvedDefer;
|
|
var maps = {};
|
|
var tiles = {};
|
|
var layers = {};
|
|
var paths = {};
|
|
var markers = {};
|
|
var geoJSON = {};
|
|
this.setMap = function (leafletMap, scopeId) {
|
|
var defer = getUnresolvedDefer(maps, scopeId);
|
|
defer.resolve(leafletMap);
|
|
setResolvedDefer(maps, scopeId);
|
|
};
|
|
this.getMap = function (scopeId) {
|
|
var defer = getDefer(maps, scopeId);
|
|
return defer.promise;
|
|
};
|
|
this.unresolveMap = function (scopeId) {
|
|
var id = leafletHelpers.obtainEffectiveMapId(maps, scopeId);
|
|
maps[id] = undefined;
|
|
};
|
|
this.getPaths = function (scopeId) {
|
|
var defer = getDefer(paths, scopeId);
|
|
return defer.promise;
|
|
};
|
|
this.setPaths = function (leafletPaths, scopeId) {
|
|
var defer = getUnresolvedDefer(paths, scopeId);
|
|
defer.resolve(leafletPaths);
|
|
setResolvedDefer(paths, scopeId);
|
|
};
|
|
this.getMarkers = function (scopeId) {
|
|
var defer = getDefer(markers, scopeId);
|
|
return defer.promise;
|
|
};
|
|
this.setMarkers = function (leafletMarkers, scopeId) {
|
|
var defer = getUnresolvedDefer(markers, scopeId);
|
|
defer.resolve(leafletMarkers);
|
|
setResolvedDefer(markers, scopeId);
|
|
};
|
|
this.getLayers = function (scopeId) {
|
|
var defer = getDefer(layers, scopeId);
|
|
return defer.promise;
|
|
};
|
|
this.setLayers = function (leafletLayers, scopeId) {
|
|
var defer = getUnresolvedDefer(layers, scopeId);
|
|
defer.resolve(leafletLayers);
|
|
setResolvedDefer(layers, scopeId);
|
|
};
|
|
this.setTiles = function (leafletTiles, scopeId) {
|
|
var defer = getUnresolvedDefer(tiles, scopeId);
|
|
defer.resolve(leafletTiles);
|
|
setResolvedDefer(tiles, scopeId);
|
|
};
|
|
this.getTiles = function (scopeId) {
|
|
var defer = getDefer(tiles, scopeId);
|
|
return defer.promise;
|
|
};
|
|
this.setGeoJSON = function (leafletGeoJSON, scopeId) {
|
|
var defer = getUnresolvedDefer(geoJSON, scopeId);
|
|
defer.resolve(leafletGeoJSON);
|
|
setResolvedDefer(geoJSON, scopeId);
|
|
};
|
|
this.getGeoJSON = function (scopeId) {
|
|
var defer = getDefer(geoJSON, scopeId);
|
|
return defer.promise;
|
|
};
|
|
}
|
|
]);
|
|
angular.module('leaflet-directive').factory('leafletMapDefaults', [
|
|
'$q',
|
|
'leafletHelpers',
|
|
function ($q, leafletHelpers) {
|
|
function _getDefaults() {
|
|
return {
|
|
keyboard: true,
|
|
dragging: true,
|
|
worldCopyJump: false,
|
|
doubleClickZoom: true,
|
|
scrollWheelZoom: true,
|
|
zoomControl: true,
|
|
zoomsliderControl: false,
|
|
zoomControlPosition: 'topleft',
|
|
attributionControl: true,
|
|
controls: {
|
|
layers: {
|
|
visible: true,
|
|
position: 'topright',
|
|
collapsed: true
|
|
}
|
|
},
|
|
crs: L.CRS.EPSG3857,
|
|
tileLayer: 'http://{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, obtainEffectiveMapId = leafletHelpers.obtainEffectiveMapId, defaults = {};
|
|
// Get the _defaults dictionary, and override the properties defined by the user
|
|
return {
|
|
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,
|
|
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.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 (isDefined(userDefaults.crs) && isDefined(L.CRS[userDefaults.crs])) {
|
|
newDefaults.crs = L.CRS[userDefaults.crs];
|
|
}
|
|
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;
|
|
}
|
|
}
|
|
var mapId = obtainEffectiveMapId(defaults, scopeId);
|
|
defaults[mapId] = newDefaults;
|
|
return newDefaults;
|
|
}
|
|
};
|
|
}
|
|
]);
|
|
angular.module('leaflet-directive').factory('leafletEvents', [
|
|
'$rootScope',
|
|
'$q',
|
|
'$log',
|
|
'leafletHelpers',
|
|
function ($rootScope, $q, $log, leafletHelpers) {
|
|
var safeApply = leafletHelpers.safeApply, isDefined = leafletHelpers.isDefined, isObject = leafletHelpers.isObject, Helpers = leafletHelpers;
|
|
var _getAvailableLabelEvents = function () {
|
|
return [
|
|
'click',
|
|
'dblclick',
|
|
'mousedown',
|
|
'mouseover',
|
|
'mouseout',
|
|
'contextmenu'
|
|
];
|
|
};
|
|
var genLabelEvents = function (leafletScope, logic, marker, name) {
|
|
var labelEvents = _getAvailableLabelEvents();
|
|
var scopeWatchName = 'markers.' + name;
|
|
for (var i = 0; i < labelEvents.length; i++) {
|
|
var eventName = labelEvents[i];
|
|
marker.label.on(eventName, genDispatchLabelEvent(leafletScope, eventName, logic, marker.label, scopeWatchName));
|
|
}
|
|
};
|
|
var genDispatchMarkerEvent = function (eventName, logic, leafletScope, marker, name, markerData) {
|
|
return function (e) {
|
|
var broadcastName = 'leafletDirectiveMarker.' + eventName;
|
|
// 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 () {
|
|
markerData.lat = marker.getLatLng().lat;
|
|
markerData.lng = marker.getLatLng().lng;
|
|
});
|
|
if (markerData.message && markerData.focus === true) {
|
|
marker.openPopup();
|
|
}
|
|
}
|
|
safeApply(leafletScope, function (scope) {
|
|
if (logic === 'emit') {
|
|
scope.$emit(broadcastName, {
|
|
markerName: name,
|
|
leafletEvent: e
|
|
});
|
|
} else {
|
|
$rootScope.$broadcast(broadcastName, {
|
|
markerName: name,
|
|
leafletEvent: e
|
|
});
|
|
}
|
|
});
|
|
};
|
|
};
|
|
var genDispatchPathEvent = function (eventName, logic, leafletScope, marker, name) {
|
|
return function (e) {
|
|
var broadcastName = 'leafletDirectivePath.' + eventName;
|
|
safeApply(leafletScope, function (scope) {
|
|
if (logic === 'emit') {
|
|
scope.$emit(broadcastName, {
|
|
pathName: name,
|
|
leafletEvent: e
|
|
});
|
|
} else {
|
|
$rootScope.$broadcast(broadcastName, {
|
|
pathName: name,
|
|
leafletEvent: e
|
|
});
|
|
}
|
|
});
|
|
};
|
|
};
|
|
var genDispatchLabelEvent = function (scope, eventName, logic, label, scope_watch_name) {
|
|
return function (e) {
|
|
// Put together broadcast name
|
|
var broadcastName = 'leafletDirectiveLabel.' + eventName;
|
|
var markerName = scope_watch_name.replace('markers.', '');
|
|
// Safely broadcast the event
|
|
safeApply(scope, function (scope) {
|
|
if (logic === 'emit') {
|
|
scope.$emit(broadcastName, {
|
|
leafletEvent: e,
|
|
label: label,
|
|
markerName: markerName
|
|
});
|
|
} else if (logic === 'broadcast') {
|
|
$rootScope.$broadcast(broadcastName, {
|
|
leafletEvent: e,
|
|
label: label,
|
|
markerName: markerName
|
|
});
|
|
}
|
|
});
|
|
};
|
|
};
|
|
var _getAvailableMarkerEvents = function () {
|
|
return [
|
|
'click',
|
|
'dblclick',
|
|
'mousedown',
|
|
'mouseover',
|
|
'mouseout',
|
|
'contextmenu',
|
|
'dragstart',
|
|
'drag',
|
|
'dragend',
|
|
'move',
|
|
'remove',
|
|
'popupopen',
|
|
'popupclose'
|
|
];
|
|
};
|
|
var _getAvailablePathEvents = function () {
|
|
return [
|
|
'click',
|
|
'dblclick',
|
|
'mousedown',
|
|
'mouseover',
|
|
'mouseout',
|
|
'contextmenu',
|
|
'add',
|
|
'remove',
|
|
'popupopen',
|
|
'popupclose'
|
|
];
|
|
};
|
|
return {
|
|
getAvailableMapEvents: function () {
|
|
return [
|
|
'click',
|
|
'dblclick',
|
|
'mousedown',
|
|
'mouseup',
|
|
'mouseover',
|
|
'mouseout',
|
|
'mousemove',
|
|
'contextmenu',
|
|
'focus',
|
|
'blur',
|
|
'preclick',
|
|
'load',
|
|
'unload',
|
|
'viewreset',
|
|
'movestart',
|
|
'move',
|
|
'moveend',
|
|
'dragstart',
|
|
'drag',
|
|
'dragend',
|
|
'zoomstart',
|
|
'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'
|
|
];
|
|
},
|
|
genDispatchMapEvent: function (scope, eventName, logic) {
|
|
return function (e) {
|
|
// Put together broadcast name
|
|
var broadcastName = 'leafletDirectiveMap.' + eventName;
|
|
// Safely broadcast the event
|
|
safeApply(scope, function (scope) {
|
|
if (logic === 'emit') {
|
|
scope.$emit(broadcastName, { leafletEvent: e });
|
|
} else if (logic === 'broadcast') {
|
|
$rootScope.$broadcast(broadcastName, { leafletEvent: e });
|
|
}
|
|
});
|
|
};
|
|
},
|
|
getAvailableMarkerEvents: _getAvailableMarkerEvents,
|
|
getAvailablePathEvents: _getAvailablePathEvents,
|
|
notifyCenterChangedToBounds: function (scope) {
|
|
scope.$broadcast('boundsChanged');
|
|
},
|
|
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);
|
|
}
|
|
},
|
|
bindMarkerEvents: function (marker, name, markerData, leafletScope) {
|
|
var markerEvents = [];
|
|
var i;
|
|
var eventName;
|
|
var logic = 'broadcast';
|
|
if (!isDefined(leafletScope.eventBroadcast)) {
|
|
// Backward compatibility, if no event-broadcast attribute, all events are broadcasted
|
|
markerEvents = _getAvailableMarkerEvents();
|
|
} else if (!isObject(leafletScope.eventBroadcast)) {
|
|
// Not a valid object
|
|
$log.error('[AngularJS - Leaflet] event-broadcast must be an object check your model.');
|
|
} else {
|
|
// We have a possible valid object
|
|
if (!isDefined(leafletScope.eventBroadcast.marker)) {
|
|
// We do not have events enable/disable do we do nothing (all enabled by default)
|
|
markerEvents = _getAvailableMarkerEvents();
|
|
} else if (!isObject(leafletScope.eventBroadcast.marker)) {
|
|
// Not a valid object
|
|
$log.warn('[AngularJS - Leaflet] event-broadcast.marker must be an object check your model.');
|
|
} else {
|
|
// We have a possible valid map object
|
|
// Event propadation logic
|
|
if (leafletScope.eventBroadcast.marker.logic !== undefined && leafletScope.eventBroadcast.marker.logic !== null) {
|
|
// We take care of possible propagation logic
|
|
if (leafletScope.eventBroadcast.marker.logic !== 'emit' && leafletScope.eventBroadcast.marker.logic !== 'broadcast') {
|
|
// This is an error
|
|
$log.warn('[AngularJS - Leaflet] Available event propagation logic are: \'emit\' or \'broadcast\'.');
|
|
} else if (leafletScope.eventBroadcast.marker.logic === 'emit') {
|
|
logic = 'emit';
|
|
}
|
|
}
|
|
// Enable / Disable
|
|
var markerEventsEnable = false, markerEventsDisable = false;
|
|
if (leafletScope.eventBroadcast.marker.enable !== undefined && leafletScope.eventBroadcast.marker.enable !== null) {
|
|
if (typeof leafletScope.eventBroadcast.marker.enable === 'object') {
|
|
markerEventsEnable = true;
|
|
}
|
|
}
|
|
if (leafletScope.eventBroadcast.marker.disable !== undefined && leafletScope.eventBroadcast.marker.disable !== null) {
|
|
if (typeof leafletScope.eventBroadcast.marker.disable === 'object') {
|
|
markerEventsDisable = true;
|
|
}
|
|
}
|
|
if (markerEventsEnable && markerEventsDisable) {
|
|
// Both are active, this is an error
|
|
$log.warn('[AngularJS - Leaflet] can not enable and disable events at the same time');
|
|
} else if (!markerEventsEnable && !markerEventsDisable) {
|
|
// Both are inactive, this is an error
|
|
$log.warn('[AngularJS - Leaflet] must enable or disable events');
|
|
} else {
|
|
// At this point the marker object is OK, lets enable or disable events
|
|
if (markerEventsEnable) {
|
|
// Enable events
|
|
for (i = 0; i < leafletScope.eventBroadcast.marker.enable.length; i++) {
|
|
eventName = leafletScope.eventBroadcast.marker.enable[i];
|
|
// Do we have already the event enabled?
|
|
if (markerEvents.indexOf(eventName) !== -1) {
|
|
// Repeated event, this is an error
|
|
$log.warn('[AngularJS - Leaflet] This event ' + eventName + ' is already enabled');
|
|
} else {
|
|
// Does the event exists?
|
|
if (_getAvailableMarkerEvents().indexOf(eventName) === -1) {
|
|
// The event does not exists, this is an error
|
|
$log.warn('[AngularJS - Leaflet] This event ' + eventName + ' does not exist');
|
|
} else {
|
|
// All ok enable the event
|
|
markerEvents.push(eventName);
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
// Disable events
|
|
markerEvents = _getAvailableMarkerEvents();
|
|
for (i = 0; i < leafletScope.eventBroadcast.marker.disable.length; i++) {
|
|
eventName = leafletScope.eventBroadcast.marker.disable[i];
|
|
var index = markerEvents.indexOf(eventName);
|
|
if (index === -1) {
|
|
// The event does not exist
|
|
$log.warn('[AngularJS - Leaflet] This event ' + eventName + ' does not exist or has been already disabled');
|
|
} else {
|
|
markerEvents.splice(index, 1);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
for (i = 0; i < markerEvents.length; i++) {
|
|
eventName = markerEvents[i];
|
|
marker.on(eventName, genDispatchMarkerEvent(eventName, logic, leafletScope, marker, name, markerData));
|
|
}
|
|
if (Helpers.LabelPlugin.isLoaded() && isDefined(marker.label)) {
|
|
genLabelEvents(leafletScope, logic, marker, name);
|
|
}
|
|
},
|
|
bindPathEvents: function (path, name, pathData, leafletScope) {
|
|
var pathEvents = [];
|
|
var i;
|
|
var eventName;
|
|
var logic = 'broadcast';
|
|
window.lls = leafletScope;
|
|
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('[AngularJS - Leaflet] 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('[AngularJS - Leaflet] 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('[AngularJS - Leaflet] Available event propagation logic are: \'emit\' or \'broadcast\'.');
|
|
} else if (leafletScope.eventBroadcast.path.logic === 'emit') {
|
|
logic = 'emit';
|
|
}
|
|
}
|
|
// Enable / Disable
|
|
var pathEventsEnable = false, 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('[AngularJS - Leaflet] can not enable and disable events at the same time');
|
|
} else if (!pathEventsEnable && !pathEventsDisable) {
|
|
// Both are inactive, this is an error
|
|
$log.warn('[AngularJS - Leaflet] 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('[AngularJS - Leaflet] 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('[AngularJS - Leaflet] 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('[AngularJS - Leaflet] 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];
|
|
path.on(eventName, genDispatchPathEvent(eventName, logic, leafletScope, pathEvents, name));
|
|
}
|
|
if (Helpers.LabelPlugin.isLoaded() && isDefined(path.label)) {
|
|
genLabelEvents(leafletScope, logic, path, name);
|
|
}
|
|
}
|
|
};
|
|
}
|
|
]);
|
|
angular.module('leaflet-directive').factory('leafletLayerHelpers', [
|
|
'$rootScope',
|
|
'$log',
|
|
'leafletHelpers',
|
|
function ($rootScope, $log, leafletHelpers) {
|
|
var Helpers = leafletHelpers, isString = leafletHelpers.isString, isObject = leafletHelpers.isObject, isDefined = leafletHelpers.isDefined;
|
|
var layerTypes = {
|
|
xyz: {
|
|
mustHaveUrl: true,
|
|
createLayer: function (params) {
|
|
return L.tileLayer(params.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);
|
|
}
|
|
},
|
|
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 && 'string' === typeof options.crs) {
|
|
/*jshint -W061 */
|
|
options.crs = eval(options.crs);
|
|
}
|
|
return new L.GeoJSON.WFS(params.url, params.layer, options);
|
|
}
|
|
},
|
|
group: {
|
|
mustHaveUrl: false,
|
|
createLayer: function () {
|
|
return L.layerGroup();
|
|
}
|
|
},
|
|
google: {
|
|
mustHaveUrl: false,
|
|
createLayer: function (params) {
|
|
var type = params.type || 'SATELLITE';
|
|
if (!Helpers.GoogleLayerPlugin.isLoaded()) {
|
|
return;
|
|
}
|
|
return new L.Google(type, params.options);
|
|
}
|
|
},
|
|
china: {
|
|
mustHaveUrl: false,
|
|
createLayer: function (params) {
|
|
var type = params.type || '';
|
|
if (!Helpers.ChinaLayerPlugin.isLoaded()) {
|
|
return;
|
|
}
|
|
return L.tileLayer.chinaProvider(type, 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;
|
|
}
|
|
},
|
|
dynamic: {
|
|
mustHaveUrl: true,
|
|
createLayer: function (params) {
|
|
if (!Helpers.DynamicMapLayerPlugin.isLoaded()) {
|
|
return;
|
|
}
|
|
return L.esri.dynamicMapLayer(params.url, params.options);
|
|
}
|
|
},
|
|
markercluster: {
|
|
mustHaveUrl: false,
|
|
createLayer: function (params) {
|
|
if (!Helpers.MarkerClusterPlugin.isLoaded()) {
|
|
$log.error('[AngularJS - Leaflet] 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);
|
|
}
|
|
},
|
|
heatmap: {
|
|
mustHaveUrl: false,
|
|
mustHaveData: true,
|
|
createLayer: function (params) {
|
|
if (!Helpers.HeatMapLayerPlugin.isLoaded()) {
|
|
return;
|
|
}
|
|
var layer = new L.TileLayer.WebGLHeatMap(params.options);
|
|
if (isDefined(params.data)) {
|
|
layer.setData(params.data);
|
|
}
|
|
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);
|
|
}
|
|
}
|
|
};
|
|
function isValidLayerType(layerDefinition) {
|
|
// Check if the baselayer has a valid type
|
|
if (!isString(layerDefinition.type)) {
|
|
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;
|
|
}
|
|
return true;
|
|
}
|
|
return {
|
|
createLayer: function (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,
|
|
type: layerDefinition.layerType,
|
|
bounds: layerDefinition.bounds,
|
|
key: layerDefinition.key,
|
|
pluginOptions: layerDefinition.pluginOptions
|
|
};
|
|
//TODO Add $watch to the layer properties
|
|
return layerTypes[layerDefinition.type].createLayer(params);
|
|
}
|
|
};
|
|
}
|
|
]);
|
|
angular.module('leaflet-directive').factory('leafletControlHelpers', [
|
|
'$rootScope',
|
|
'$log',
|
|
'leafletHelpers',
|
|
'leafletMapDefaults',
|
|
function ($rootScope, $log, leafletHelpers, leafletMapDefaults) {
|
|
var isObject = leafletHelpers.isObject, isDefined = leafletHelpers.isDefined;
|
|
var _layersControl;
|
|
var _controlLayersMustBeVisible = function (baselayers, overlays) {
|
|
var numberOfLayers = 0;
|
|
if (isObject(baselayers)) {
|
|
numberOfLayers += Object.keys(baselayers).length;
|
|
}
|
|
if (isObject(overlays)) {
|
|
numberOfLayers += Object.keys(overlays).length;
|
|
}
|
|
return numberOfLayers > 1;
|
|
};
|
|
var _createLayersControl = function (mapId) {
|
|
var defaults = leafletMapDefaults.getDefaults(mapId);
|
|
var controlOptions = {
|
|
collapsed: defaults.controls.layers.collapsed,
|
|
position: defaults.controls.layers.position
|
|
};
|
|
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;
|
|
};
|
|
return {
|
|
layersControlMustBeVisible: _controlLayersMustBeVisible,
|
|
updateLayersControl: function (map, mapId, loaded, baselayers, overlays, leafletLayers) {
|
|
var i;
|
|
var mustBeLoaded = _controlLayersMustBeVisible(baselayers, overlays);
|
|
if (isDefined(_layersControl) && loaded) {
|
|
for (i in leafletLayers.baselayers) {
|
|
_layersControl.removeLayer(leafletLayers.baselayers[i]);
|
|
}
|
|
for (i in leafletLayers.overlays) {
|
|
_layersControl.removeLayer(leafletLayers.overlays[i]);
|
|
}
|
|
_layersControl.removeFrom(map);
|
|
}
|
|
if (mustBeLoaded) {
|
|
_layersControl = _createLayersControl(mapId);
|
|
for (i in baselayers) {
|
|
if (isDefined(leafletLayers.baselayers[i])) {
|
|
_layersControl.addBaseLayer(leafletLayers.baselayers[i], baselayers[i].name);
|
|
}
|
|
}
|
|
for (i in overlays) {
|
|
if (isDefined(leafletLayers.overlays[i])) {
|
|
_layersControl.addOverlay(leafletLayers.overlays[i], overlays[i].name);
|
|
}
|
|
}
|
|
_layersControl.addTo(map);
|
|
}
|
|
return mustBeLoaded;
|
|
}
|
|
};
|
|
}
|
|
]);
|
|
angular.module('leaflet-directive').factory('leafletLegendHelpers', function () {
|
|
var _updateArcGISLegend = function (div, legendData) {
|
|
div.innerHTML = '';
|
|
if (legendData.error) {
|
|
div.innerHTML += '<div class="info-title alert alert-danger">' + legendData.error.message + '</div>';
|
|
} else {
|
|
for (var i = 0; i < legendData.layers.length; i++) {
|
|
var layer = legendData.layers[i];
|
|
div.innerHTML += '<div class="info-title">' + layer.layerName + '</div>';
|
|
for (var j = 0; j < layer.legend.length; j++) {
|
|
var leg = layer.legend[j];
|
|
div.innerHTML += '<div class="inline"><img src="data:' + leg.contentType + ';base64,' + leg.imageData + '" /></div>' + '<div class="info-label">' + leg.label + '</div>';
|
|
}
|
|
}
|
|
}
|
|
};
|
|
var _getOnAddArcGISLegend = function (legendData, legendClass) {
|
|
return function () {
|
|
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);
|
|
}
|
|
_updateArcGISLegend(div, legendData);
|
|
return div;
|
|
};
|
|
};
|
|
var _getOnAddArrayLegend = function (legend, legendClass) {
|
|
return function () {
|
|
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 {
|
|
getOnAddArcGISLegend: _getOnAddArcGISLegend,
|
|
getOnAddArrayLegend: _getOnAddArrayLegend,
|
|
updateArcGISLegend: _updateArcGISLegend
|
|
};
|
|
});
|
|
angular.module('leaflet-directive').factory('leafletPathsHelpers', [
|
|
'$rootScope',
|
|
'$log',
|
|
'leafletHelpers',
|
|
function ($rootScope, $log, leafletHelpers) {
|
|
var isDefined = leafletHelpers.isDefined, isArray = leafletHelpers.isArray, isNumber = leafletHelpers.isNumber, isValidPoint = leafletHelpers.isValidPoint;
|
|
function _convertToLeafletLatLngs(latlngs) {
|
|
return latlngs.filter(function (latlng) {
|
|
return isValidPoint(latlng);
|
|
}).map(function (latlng) {
|
|
return new L.LatLng(latlng.lat, latlng.lng);
|
|
});
|
|
}
|
|
function _convertToLeafletLatLng(latlng) {
|
|
return new L.LatLng(latlng.lat, latlng.lng);
|
|
}
|
|
function _convertToLeafletMultiLatLngs(paths) {
|
|
return paths.map(function (latlngs) {
|
|
return _convertToLeafletLatLngs(latlngs);
|
|
});
|
|
}
|
|
function _getOptions(path, defaults) {
|
|
var availableOptions = [
|
|
'stroke',
|
|
'weight',
|
|
'color',
|
|
'opacity',
|
|
'fill',
|
|
'fillColor',
|
|
'fillOpacity',
|
|
'dashArray',
|
|
'lineCap',
|
|
'lineJoin',
|
|
'clickable',
|
|
'pointerEvents',
|
|
'className',
|
|
'smoothFactor',
|
|
'noClip'
|
|
];
|
|
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) {
|
|
if (isDefined(data.weight)) {
|
|
path.setStyle({ weight: data.weight });
|
|
}
|
|
if (isDefined(data.color)) {
|
|
path.setStyle({ color: data.color });
|
|
}
|
|
if (isDefined(data.opacity)) {
|
|
path.setStyle({ opacity: data.opacity });
|
|
}
|
|
};
|
|
var _isValidPolyline = function (latlngs) {
|
|
if (!isArray(latlngs)) {
|
|
return false;
|
|
}
|
|
for (var i in latlngs) {
|
|
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').factory('leafletBoundsHelpers', [
|
|
'$log',
|
|
'leafletHelpers',
|
|
function ($log, leafletHelpers) {
|
|
var isArray = leafletHelpers.isArray, isNumber = leafletHelpers.isNumber;
|
|
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]
|
|
}
|
|
};
|
|
}
|
|
};
|
|
}
|
|
]);
|
|
angular.module('leaflet-directive').factory('leafletMarkersHelpers', [
|
|
'$rootScope',
|
|
'leafletHelpers',
|
|
'$log',
|
|
function ($rootScope, leafletHelpers, $log) {
|
|
var isDefined = leafletHelpers.isDefined, MarkerClusterPlugin = leafletHelpers.MarkerClusterPlugin, AwesomeMarkersPlugin = leafletHelpers.AwesomeMarkersPlugin, safeApply = leafletHelpers.safeApply, Helpers = leafletHelpers, isString = leafletHelpers.isString, isNumber = leafletHelpers.isNumber, isObject = leafletHelpers.isObject, groups = {};
|
|
var createLeafletIcon = function (iconData) {
|
|
if (isDefined(iconData) && isDefined(iconData.type) && iconData.type === 'awesomeMarker') {
|
|
if (!AwesomeMarkersPlugin.isLoaded()) {
|
|
$log.error('[AngularJS - Leaflet] The AwesomeMarkers Plugin is not loaded.');
|
|
}
|
|
return new L.AwesomeMarkers.icon(iconData);
|
|
}
|
|
if (isDefined(iconData) && isDefined(iconData.type) && iconData.type === 'div') {
|
|
return new L.divIcon(iconData);
|
|
}
|
|
var base64icon = '';
|
|
var base64shadow = '';
|
|
if (!isDefined(iconData)) {
|
|
return new L.Icon.Default({
|
|
iconUrl: base64icon,
|
|
shadowUrl: base64shadow
|
|
});
|
|
}
|
|
if (!isDefined(iconData.iconUrl)) {
|
|
iconData.iconUrl = base64icon;
|
|
iconData.shadowUrl = base64shadow;
|
|
}
|
|
return new L.Icon.Default(iconData);
|
|
};
|
|
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) {
|
|
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);
|
|
}
|
|
};
|
|
return {
|
|
deleteMarker: _deleteMarker,
|
|
createMarker: function (markerData) {
|
|
if (!isDefined(markerData)) {
|
|
$log.error('[AngularJS - Leaflet] The marker definition is not valid.');
|
|
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
|
|
};
|
|
return new L.marker(markerData, markerOptions);
|
|
},
|
|
addMarkerToGroup: function (marker, groupName, map) {
|
|
if (!isString(groupName)) {
|
|
$log.error('[AngularJS - Leaflet] The marker group you have specified is invalid.');
|
|
return;
|
|
}
|
|
if (!MarkerClusterPlugin.isLoaded()) {
|
|
$log.error('[AngularJS - Leaflet] The MarkerCluster plugin is not loaded.');
|
|
return;
|
|
}
|
|
if (!isDefined(groups[groupName])) {
|
|
groups[groupName] = new L.MarkerClusterGroup();
|
|
map.addLayer(groups[groupName]);
|
|
}
|
|
groups[groupName].addLayer(marker);
|
|
},
|
|
listenMarkerEvents: function (marker, markerData, leafletScope) {
|
|
marker.on('popupopen', function () {
|
|
safeApply(leafletScope, function () {
|
|
markerData.focus = true;
|
|
});
|
|
});
|
|
marker.on('popupclose', function () {
|
|
safeApply(leafletScope, function () {
|
|
markerData.focus = false;
|
|
});
|
|
});
|
|
},
|
|
addMarkerWatcher: function (marker, name, leafletScope, layers, map) {
|
|
var clearWatch = leafletScope.$watch('markers.' + name, function (markerData, oldMarkerData) {
|
|
if (!isDefined(markerData)) {
|
|
_deleteMarker(marker, map, layers);
|
|
clearWatch();
|
|
return;
|
|
}
|
|
if (!isDefined(oldMarkerData)) {
|
|
return;
|
|
}
|
|
// Update the lat-lng property (always present in marker properties)
|
|
if (!(isNumber(markerData.lat) && isNumber(markerData.lng))) {
|
|
$log.warn('There are problems with lat-lng data, please verify your marker model');
|
|
_deleteMarker(marker, map, layers);
|
|
return;
|
|
}
|
|
// 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 (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('[AngularJS - Leaflet] 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)) {
|
|
$log.error('[AngularJS - Leaflet] A marker can only be added to a layer of type "group"');
|
|
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);
|
|
}
|
|
}
|
|
}
|
|
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);
|
|
}
|
|
}
|
|
// Update the Popup message property
|
|
if (!isString(markerData.message) && isString(oldMarkerData.message)) {
|
|
marker.closePopup();
|
|
marker.unbindPopup();
|
|
}
|
|
// Update the label content
|
|
if (Helpers.LabelPlugin.isLoaded() && isDefined(markerData.label) && isDefined(markerData.label.message) && !angular.equals(markerData.label.message, oldMarkerData.label.message)) {
|
|
marker.updateLabelContent(markerData.label.message);
|
|
}
|
|
// 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);
|
|
if (markerData.focus === true) {
|
|
// If the focus is set, we must open the popup, because we do not know if it was opened before
|
|
marker.openPopup();
|
|
}
|
|
}
|
|
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 && oldMarkerData.focus !== true) {
|
|
marker.openPopup();
|
|
updatedFocus = true;
|
|
}
|
|
if (oldMarkerData.focus === true && markerData.focus === true) {
|
|
// Reopen the popup when focus is still true
|
|
marker.openPopup();
|
|
updatedFocus = true;
|
|
}
|
|
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 (markerLatLng.lat !== markerData.lat || markerLatLng.lng !== markerData.lng) {
|
|
marker.setLatLng([
|
|
markerData.lat,
|
|
markerData.lng
|
|
]);
|
|
}
|
|
}, true);
|
|
}
|
|
};
|
|
}
|
|
]);
|
|
angular.module('leaflet-directive').factory('leafletHelpers', [
|
|
'$q',
|
|
'$log',
|
|
function ($q, $log) {
|
|
function _obtainEffectiveMapId(d, mapId) {
|
|
var id, i;
|
|
if (!angular.isDefined(mapId)) {
|
|
if (Object.keys(d).length === 1) {
|
|
for (i in d) {
|
|
if (d.hasOwnProperty(i)) {
|
|
id = i;
|
|
}
|
|
}
|
|
} else if (Object.keys(d).length === 0) {
|
|
id = 'main';
|
|
} else {
|
|
$log.error('[AngularJS - Leaflet] - 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), 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;
|
|
}
|
|
return {
|
|
isEmpty: function (value) {
|
|
return Object.keys(value).length === 0;
|
|
},
|
|
isUndefinedOrEmpty: function (value) {
|
|
return angular.isUndefined(value) || value === null || Object.keys(value).length === 0;
|
|
},
|
|
isDefined: function (value) {
|
|
return angular.isDefined(value) && value !== null;
|
|
},
|
|
isNumber: function (value) {
|
|
return angular.isNumber(value);
|
|
},
|
|
isString: function (value) {
|
|
return angular.isString(value);
|
|
},
|
|
isArray: function (value) {
|
|
return angular.isArray(value);
|
|
},
|
|
isObject: function (value) {
|
|
return angular.isObject(value);
|
|
},
|
|
isFunction: function (value) {
|
|
return angular.isFunction(value);
|
|
},
|
|
equals: function (o1, o2) {
|
|
return angular.equals(o1, o2);
|
|
},
|
|
isValidCenter: function (center) {
|
|
return angular.isDefined(center) && angular.isNumber(center.lat) && angular.isNumber(center.lng) && angular.isNumber(center.zoom);
|
|
},
|
|
isValidPoint: function (point) {
|
|
return angular.isDefined(point) && angular.isNumber(point.lat) && angular.isNumber(point.lng);
|
|
},
|
|
isSameCenterOnMap: function (centerModel, map) {
|
|
var mapCenter = map.getCenter();
|
|
var zoom = map.getZoom();
|
|
if (mapCenter.lat === centerModel.lat && mapCenter.lng === centerModel.lng && 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.$apply(fn);
|
|
}
|
|
},
|
|
obtainEffectiveMapId: _obtainEffectiveMapId,
|
|
getDefer: function (d, mapId) {
|
|
var id = _obtainEffectiveMapId(d, mapId), 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;
|
|
},
|
|
AwesomeMarkersPlugin: {
|
|
isLoaded: function () {
|
|
if (angular.isDefined(L.AwesomeMarkers) && angular.isDefined(L.AwesomeMarkers.Icon)) {
|
|
return true;
|
|
} else {
|
|
return false;
|
|
}
|
|
},
|
|
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;
|
|
}
|
|
}
|
|
},
|
|
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;
|
|
}
|
|
}
|
|
},
|
|
ChinaLayerPlugin: {
|
|
isLoaded: function () {
|
|
return angular.isDefined(L.tileLayer.chinaProvider);
|
|
}
|
|
},
|
|
HeatMapLayerPlugin: {
|
|
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;
|
|
}
|
|
}
|
|
},
|
|
AGSLayerPlugin: {
|
|
isLoaded: function () {
|
|
return lvector !== undefined && lvector.AGS !== undefined;
|
|
},
|
|
is: function (layer) {
|
|
if (this.isLoaded()) {
|
|
return layer instanceof lvector.AGS;
|
|
} 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;
|
|
}
|
|
}
|
|
},
|
|
DynamicMapLayerPlugin: {
|
|
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;
|
|
}
|
|
}
|
|
},
|
|
GeoJSONPlugin: {
|
|
isLoaded: function () {
|
|
return angular.isDefined(L.TileLayer.GeoJSON);
|
|
},
|
|
is: function (layer) {
|
|
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;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
};
|
|
}
|
|
]);
|
|
}()); |