I needed a way to control the appearance for my Leaflet map of New Zealand, especially when embedding a map with an iframe. Leaflet have no native support for URL parameters for, like the Permalink control for OpenLayers (there is a plugin + Leaflet-hash). This blog post will show you how you can add support for URL parameters for map position (latitude/longitude), zoom level and active layers.
First, I'm defining my layers in an JavaScript object:
var layers = {
'Seafloor': L.tileLayer('tiles/nz-seafloor/{z}/{x}/{y}.png'),
'Topographic': L.tileLayer('tiles/nz-topo/{z}/{x}/{y}.png'),
'Tour': L.cartoDB('http://thematicmapping.cartodb.com/api/v2/sql?q=SELECT * FROM new_zealand_tour')
};
You can use the same object to populate the Leaflet layers control. Next, I'm extracting the URL parameters with a regular expression (from Papermashup):
var params = {};
window.location.href.replace(/[?&]+([^=&]+)=([^&]*)/gi, function(m, key, value) {
params[key] = value;
});
All URL parameters are now stored as key/value pairs in the params object. We need to create an array of active map layers:
if (params.layers) {
var activeLayers = params.layers.split(',').map(function(item) {
return layers[item];
});
}
The array map function will create a new array with a reference to the visible layers on the map. Please note that the map function is not supported in Internet Explorer < 9 (you can easily rewrite the function with a for loop to support this browser).
We can now create a map with our URL parameters:
var map = new L.Map('map', {
center: [params.lat || -41.2728, params.lng || 173.2996],
minZoom: 4,
maxZoom: 12,
zoom: params.zoom || 6,
layers: activeLayers || [layers.Seafloor, layers.Topographic]
});
I'm using the Logical OR (||) operator to add default values. Lastly, I'm adding my map layers to the layers control, allowing users to toggle the visibility:
L.control.layers({}, layers).addTo(map);
This is the full example:
We can now control the map med URL parameters like (the code in this example is changed to support base layers and overlays):
http://earthatlas.info/nz/?lat=-42.4&lng=168&zoom=8
http://earthatlas.info/nz/?lat=-39.296&lng=174.065&zoom=11&layers=Springs
Easy peasy!
First, I'm defining my layers in an JavaScript object:
var layers = {
'Seafloor': L.tileLayer('tiles/nz-seafloor/{z}/{x}/{y}.png'),
'Topographic': L.tileLayer('tiles/nz-topo/{z}/{x}/{y}.png'),
'Tour': L.cartoDB('http://thematicmapping.cartodb.com/api/v2/sql?q=SELECT * FROM new_zealand_tour')
};
You can use the same object to populate the Leaflet layers control. Next, I'm extracting the URL parameters with a regular expression (from Papermashup):
var params = {};
window.location.href.replace(/[?&]+([^=&]+)=([^&]*)/gi, function(m, key, value) {
params[key] = value;
});
All URL parameters are now stored as key/value pairs in the params object. We need to create an array of active map layers:
if (params.layers) {
var activeLayers = params.layers.split(',').map(function(item) {
return layers[item];
});
}
The array map function will create a new array with a reference to the visible layers on the map. Please note that the map function is not supported in Internet Explorer < 9 (you can easily rewrite the function with a for loop to support this browser).
We can now create a map with our URL parameters:
var map = new L.Map('map', {
center: [params.lat || -41.2728, params.lng || 173.2996],
minZoom: 4,
maxZoom: 12,
zoom: params.zoom || 6,
layers: activeLayers || [layers.Seafloor, layers.Topographic]
});
I'm using the Logical OR (||) operator to add default values. Lastly, I'm adding my map layers to the layers control, allowing users to toggle the visibility:
L.control.layers({}, layers).addTo(map);
This is the full example:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
// Create map layers | |
var layers = { | |
'Seafloor': L.tileLayer('tiles/nz-seafloor/{z}/{x}/{y}.png', { | |
maxZoom: 9, | |
attribution: '<a href="http://www.niwa.co.nz/our-science/oceans/bathymetry">NIWA</a>' | |
}), | |
'Topographic': L.tileLayer('tiles/nz-topo/{z}/{x}/{y}.png', { | |
attribution: '<a href="http://data.linz.govt.nz/">LINZ</a>, <a href="http://lris.scinfo.org.nz/">LRIS</a>' | |
}), | |
'Tour': L.cartoDB('http://thematicmapping.cartodb.com/api/v2/sql?q=SELECT * FROM new_zealand_tour', { | |
pointToLayer: function(feature, latlng) { | |
return L.marker(latlng, { | |
icon: L.icon({ | |
iconSize: [27, 27], | |
iconAnchor: [13, 27], | |
popupAnchor: [1, -24], | |
iconUrl: 'icons/' + feature.properties.icon + '.png' | |
}) | |
}) | |
}, | |
onEachFeature: function(feature, layer) { | |
layer.bindPopup('<strong>' + feature.properties.name + '</strong><br>' + (feature.properties.description ? feature.properties.description : '')); | |
} | |
}) | |
}; | |
// Get url parameters | |
var params = {}; | |
window.location.href.replace(/[?&]+([^=&]+)=([^&]*)/gi, function(m, key, value) { | |
params[key] = value; | |
}); | |
if (params.layers) { | |
var activeLayers = params.layers.split(',').map(function(item) { // map function not supported in IE < 9 | |
return layers[item]; | |
}); | |
} | |
// Create map | |
var map = new L.Map('map', { | |
center: [params.lat || -41.2728, params.lng || 173.2996], | |
minZoom: 4, | |
maxZoom: 12, | |
zoom: params.zoom || 6, | |
layers: activeLayers || [layers.Seafloor, layers.Topographic] | |
}); | |
map.attributionControl.setPrefix('<a href="http://blog.thematicmapping.org/">thematic mapping blog</a>'); | |
// Add controls | |
L.control.scale().addTo(map); | |
L.control.layers({}, layers).addTo(map); |
http://earthatlas.info/nz/?lat=-42.4&lng=168&zoom=8
http://earthatlas.info/nz/?lat=-39.296&lng=174.065&zoom=11&layers=Springs
Easy peasy!
Thanks so much. That was the missing link.. for me, getting the overlays in a url hash. happy travels..
ReplyDeleteThere is an OL style Permalink plugin for Leaflet at https://github.com/shramov/leaflet-plugins.
ReplyDeleteit would be quite useful if
ReplyDeletehttps://github.com/mlevans/leaflet-hash
handled baselayers & overlays
Is there still no native leaflet-support for this? I felt all the solutions that are around seemed very fragile …
ReplyDelete