A Leaflet extension to distort images -- "rubbersheeting" -- for the MapKnitter.org (src) image georectification service by Public Lab. Leaflet.DistortableImage allows for perspectival distortions of images, client-side, using CSS3 transformations in the DOM.
Advantages include:
- It can handle over 100 images smoothly, even on a smartphone.
- Images can be right-clicked and downloaded individually in their original state
- CSS3 transforms are GPU-accelerated in most (all?) browsers, for a very smooth UI
- No need to server-side generate raster GeoTiffs, tilesets, etc. in order to view distorted imagery layers
- Images use DOM event handling for real-time distortion
- Full resolution download option for large images, using WebGL acceleration
Download as zip or clone the repo to get a local copy.
This plugin has basic functionality, and is in production as part of MapKnitter, but there are plenty of outstanding issues to resolve. Please consider helping out!
The recommended Google satellite base layer can be integrated using this Leaflet plugin: https://gitlab.com/IvanSanchez/Leaflet.GridLayer.GoogleMutant
Here's a screenshot:
Check out this simple demo.
And watch this GIF demo:
To test the code, open index.html
in your browser and click and drag the markers on the edges of the image. The image will show perspectival distortions.
The most simple implementation to get started:
// basic Leaflet map setup
map = L.map('map').setView([51.505, -0.09], 13);
L.tileLayer('https://{s}.tiles.mapbox.com/v3/anishshah101.ipm9j6em/{z}/{x}/{y}.png', {
maxZoom: 18,
attribution: 'Map data © <a href="http://openstreetmap.org">OpenStreetMap</a> contributors, ' +
'<a href="http://creativecommons.org/licenses/by-sa/2.0/">CC-BY-SA</a>, ' +
'Imagery © <a href="http://mapbox.com">Mapbox</a>',
id: 'examples.map-i86knfo3'
}).addTo(map);
// create an image
img = new L.DistortableImageOverlay(
'example.png', {
corners: [
new L.latLng(51.52,-0.10),
new L.latLng(51.52,-0.14),
new L.latLng(51.50,-0.10),
new L.latLng(51.50,-0.14)
],
}
).addTo(map);
// enable editing
L.DomEvent.on(img._image, 'load', img.editing.enable, img.editing);
We've added a GPU-accelerated means to generate a full resolution version of the distorted image; it requires two additional dependencies to enable; see how we've included them in the demo:
<script src="../node_modules/webgl-distort/dist/webgl-distort.js"></script>
<script src="../node_modules/glfx/glfx.js"></script>
When instantiating a Distortable Image, pass in a fullResolutionSrc
option set to the url of the higher resolution image. This image will be used in full-res exporting.
// basic Leaflet map setup
map = L.map('map').setView([51.505, -0.09], 13);
L.tileLayer('https://{s}.tiles.mapbox.com/v3/anishshah101.ipm9j6em/{z}/{x}/{y}.png', {
maxZoom: 18,
attribution: 'Map data © <a href="http://openstreetmap.org">OpenStreetMap</a> contributors, ' +
'<a href="http://creativecommons.org/licenses/by-sa/2.0/">CC-BY-SA</a>, ' +
'Imagery © <a href="http://mapbox.com">Mapbox</a>',
id: 'examples.map-i86knfo3'
}).addTo(map);
```js
// create an image
img = new L.DistortableImageOverlay(
'example.png', {
corners: [
new L.latLng(51.52,-0.10),
new L.latLng(51.52,-0.14),
new L.latLng(51.50,-0.10),
new L.latLng(51.50,-0.14)
],
fullResolutionSrc: 'large.jpg'
}
).addTo(map);
L.DomEvent.on(img._image, 'load', img.editing.enable, img.editing);
To test the multi-image interface, open select.html
. Currently it supports multiple image selection and translations; image distortions still use the single-image interface.
- Multiple images can be selected using cmd +
click
to toggle individual image selection. - Click on the map or hit the esc key to quickly deselect all images.
Our DistortableCollection
class allows working with multiple images simultaneously. Say we instantiated 3 images, saved them to the variables img
, img2
, and img3
, and enabled editing on all of them. To access the UI and functionalities available in the multiple image interface, pass them to the collection class:
// OPTION 1: Pass in images immediately
new L.DistortableCollection([img, img2, img3]).addTo(map);
// OPTION 2: Instantiate an empty collection and pass in images later
var imageFeatureGroup = new L.DistortableCollection().addTo(map);
imageFeatureGroup.addLayer(img);
imageFeatureGroup.addLayer(img2);
imageFeatureGroup.addLayer(img3);
For multiple images, we've also added a ToggleOrder
action, that switches overlapping images back and forth into view by employing bringToFront()
and bringToBack()
from the Leaflet API.
ToggleOrder = EditOverlayAction.extend({
options: {
toolbarIcon: {
html: '<span class="fa fa-sort"></span>',
tooltip: 'Change order',
title: 'Toggle order'
}
},
addHooks: function ()
{
var editing = this._overlay.editing;
editing._toggleOrder(); // toggles images into view
this.disable();
}
});
The corners are stored in img._corners
as L.latLng
objects, so after instantiating the image and moving it around, you can always access them like this:
img = new L.DistortableImageOverlay(...);
img.addTo(map);
// move the image around
JSON.stringify(img._corners)
=> "[{"lat":51.52,"lng":-0.1},{"lat":51.52,"lng":-0.14},{"lat":51.5,"lng":-0.1},{"lat":51.5,"lng":-0.14}]"
- From the root directory, run
npm install
orsudo npm install
- Open examples/index.html in a browser
- This project uses
grunt
to do a lot of things, including concatenate source files from /src/ to /DistortableImageOverlay.js. But you may need to install grunt-cli:npm install -g grunt-cli
first. - Run
grunt
in the root directory, and it will watch for changes and concatenate them on the fly.
To build all files from /src/
into the /dist/
folder, run grunt concat:dist
.
- Anish Shah, @anishshah101
- Justin Manley, @manleyjster
- Jeff Warren @jywarren
Many more at https://github.com/publiclab/Leaflet.DistortableImage/graphs/contributors