4264e56161f1c921b94163f3d8cf5a629da67963
[rails.git] / app / assets / javascripts / leaflet.share.js
1 L.OSM.share = function (options) {
2   var control = L.control(options),
3     marker = L.marker([0, 0], {draggable: true}),
4     locationFilter = new L.LocationFilter({
5       enableButton: false,
6       adjustButton: false
7     });
8
9   control.onAdd = function (map) {
10     var $container = $('<div>')
11       .attr('class', 'control-share');
12
13     $('<a>')
14       .attr('class', 'control-button')
15       .attr('href', '#')
16       .attr('title', 'Share')
17       .html('<span class="icon share"></span>')
18       .on('click', toggle)
19       .appendTo($container);
20
21     var $ui = $('<div>')
22       .attr('class', 'share-ui');
23
24     $('<div>')
25       .attr('class', 'sidebar_heading')
26       .appendTo($ui)
27       .append(
28         $('<a>')
29           .text(I18n.t('javascripts.close'))
30           .attr('class', 'sidebar_close')
31           .attr('href', '#')
32           .bind('click', toggle))
33       .append(
34         $('<h4>')
35           .text(I18n.t('javascripts.share.title')));
36
37     // Link / Embed
38
39     var $linkSection = $('<div>')
40       .attr('class', 'section share-link')
41       .appendTo($ui);
42
43     $('<h4>')
44       .text(I18n.t('javascripts.share.link'))
45       .appendTo($linkSection);
46
47     var $form = $('<form>')
48       .attr('class', 'standard-form')
49       .appendTo($linkSection);
50
51     $('<div>')
52       .attr('class', 'form-row')
53       .appendTo($form)
54       .append(
55         $('<label>')
56           .attr('for', 'link_marker')
57           .append(
58             $('<input>')
59               .attr('id', 'link_marker')
60               .attr('type', 'checkbox')
61               .bind('change', toggleMarker))
62           .append(I18n.t('javascripts.share.include_marker')));
63
64     $('<div>')
65       .attr('class', 'form-row')
66       .appendTo($form)
67       .append(
68         $('<label>')
69           .attr('for', 'center_marker')
70           .append(
71             $('<input>')
72               .attr('id', 'center_marker')
73               .attr('type', 'checkbox')
74               .prop('checked', true)
75               .bind('change', update))
76           .append(I18n.t('javascripts.share.center_marker')));
77
78     $('<div>')
79       .attr('class', 'form-row')
80       .appendTo($form)
81       .append($('<label>')
82         .attr('for', 'long_input')
83         .text(I18n.t('javascripts.share.long_link')))
84       .append($('<a>')
85         .attr('id', 'long_link')
86         .append($('<span>')
87           .attr('class', 'icon link')))
88       .append($('<input>')
89         .attr('id', 'long_input')
90         .attr('type', 'text')
91         .on('click', select));
92
93     $('<div>')
94       .attr('class', 'form-row')
95       .appendTo($form)
96       .append($('<label>')
97         .attr('for', 'short_input')
98         .text(I18n.t('javascripts.share.short_link')))
99       .append($('<a>')
100         .attr('id', 'short_link')
101         .append($('<span>')
102           .attr('class', 'icon link')))
103       .append($('<input>')
104         .attr('id', 'short_input')
105         .attr('type', 'text')
106         .on('click', select));
107
108     $('<div>')
109       .attr('class', 'form-row')
110       .appendTo($form)
111       .append($('<label>')
112         .attr('for', 'embed_html')
113         .text(I18n.t('javascripts.share.embed')))
114       .append(
115         $('<textarea>')
116           .attr('id', 'embed_html')
117           .on('click', select));
118
119     $('<p>')
120       .attr('class', 'deemphasize')
121       .text(I18n.t('javascripts.share.paste_html'))
122       .appendTo($linkSection);
123
124     // Image
125
126     var $imageSection = $('<div>')
127       .attr('class', 'section share-image')
128       .appendTo($ui);
129
130     $('<h4>')
131       .text(I18n.t('javascripts.share.image'))
132       .appendTo($imageSection);
133
134     $form = $('<form>')
135       .attr('class', 'standard-form')
136       .attr('action', '/export/finish')
137       .attr('method', 'post')
138       .appendTo($imageSection);
139
140     $('<div>')
141       .attr('class', 'form-row')
142       .appendTo($form)
143       .append(
144         $('<label>')
145           .attr('for', 'image_filter')
146           .append(
147             $('<input>')
148               .attr('id', 'image_filter')
149               .attr('type', 'checkbox')
150               .bind('change', toggleFilter))
151           .append(I18n.t('javascripts.share.custom_dimensions')));
152
153     $('<div>')
154       .attr('class', 'form-row')
155       .appendTo($form)
156       .append(
157         $('<label>')
158           .attr('class', 'standard-label')
159           .attr('for', 'mapnik_format')
160           .text(I18n.t('javascripts.share.format')))
161       .append($('<select>')
162         .attr('name', 'mapnik_format')
163         .attr('id', 'mapnik_format')
164         .append($('<option>').val('png').text('PNG').prop('selected', true))
165         .append($('<option>').val('jpeg').text('JPEG'))
166         .append($('<option>').val('svg').text('SVG'))
167         .append($('<option>').val('pdf').text('PDF')));
168
169     $('<div>')
170       .attr('class', 'form-row')
171       .appendTo($form)
172       .append($('<label>')
173         .attr('class', 'standard-label')
174         .attr('for', 'mapnik_scale')
175         .text(I18n.t('javascripts.share.scale')))
176       .append('1 : ')
177       .append($('<input>')
178         .attr('name', 'mapnik_scale')
179         .attr('id', 'mapnik_scale')
180         .attr('type', 'text')
181         .on('change', update));
182
183     ['minlon', 'minlat', 'maxlon', 'maxlat'].forEach(function(name) {
184       $('<input>')
185         .attr('id', 'mapnik_' + name)
186         .attr('name', name)
187         .attr('type', 'hidden')
188         .appendTo($form);
189     });
190
191     $('<input>')
192       .attr('name', 'format')
193       .attr('value', 'mapnik')
194       .attr('type', 'hidden')
195       .appendTo($form);
196
197     $('<p>')
198       .attr('class', 'deemphasize')
199       .html(I18n.t('javascripts.share.image_size') + ' <span id="mapnik_image_width"></span> x <span id="mapnik_image_height"></span>')
200       .appendTo($form);
201
202     $('<input>')
203       .attr('type', 'submit')
204       .attr('value', I18n.t('javascripts.share.download'))
205       .appendTo($form);
206
207     locationFilter
208       .on('change', update)
209       .addTo(map);
210
211     map.on('moveend layeradd layerremove', update);
212     marker.on('dragend', update);
213
214     options.sidebar.addPane($ui);
215
216     $ui
217       .on('hide', hidden);
218
219     function hidden() {
220       map.removeLayer(marker);
221       locationFilter.disable();
222       update();
223     }
224
225     function toggle(e) {
226       e.stopPropagation();
227       e.preventDefault();
228
229       $('#mapnik_scale').val(getScale());
230       marker.setLatLng(map.getCenter());
231
232       update();
233       options.sidebar.togglePane($ui);
234     }
235
236     function toggleMarker() {
237       if ($(this).is(':checked')) {
238         marker.setLatLng(map.getCenter());
239         map.addLayer(marker);
240       } else {
241         map.removeLayer(marker);
242       }
243       update();
244     }
245
246     function toggleFilter() {
247       if ($(this).is(':checked')) {
248         if (!locationFilter.getBounds().isValid()) {
249           locationFilter.setBounds(map.getBounds().pad(-0.2));
250         }
251
252         locationFilter.enable();
253       } else {
254         locationFilter.disable();
255       }
256       update();
257     }
258
259     function update() {
260       if (map.hasLayer(marker) && $('#center_marker').is(':checked')) {
261         map.panTo(marker.getLatLng());
262       }
263
264       var bounds = map.getBounds();
265
266       $('#link_marker')
267         .prop('checked', map.hasLayer(marker));
268
269       $('#center_marker')
270         .prop('disabled', !map.hasLayer(marker));
271
272       $('#image_filter')
273         .prop('checked', locationFilter.isEnabled());
274
275       // Link / Embed
276
277       $('#short_input').val(map.getShortUrl(marker));
278       $('#long_input').val(map.getUrl(marker));
279       $('#short_link').attr('href', map.getShortUrl(marker));
280       $('#long_link').attr('href', map.getUrl(marker));
281
282       var params = {
283         bbox: bounds.toBBoxString(),
284         layer: map.getMapBaseLayerId()
285       };
286
287       if (map.hasLayer(marker)) {
288         params.marker = marker.getLatLng().lat + ',' + marker.getLatLng().lng;
289       }
290
291       $('#embed_html').val(
292         '<iframe width="425" height="350" frameborder="0" scrolling="no" marginheight="0" marginwidth="0" src="' +
293           'http://' + OSM.SERVER_URL + '/export/embed.html?' + $.param(params) +
294           '" style="border: 1px solid black"></iframe><br/>' +
295           '<small><a href="' + map.getUrl(marker) + '</a></small>');
296
297       // Image
298
299       if (locationFilter.isEnabled()) {
300         bounds = locationFilter.getBounds();
301       }
302
303       var scale = $("#mapnik_scale").val(),
304         size = L.bounds(L.CRS.EPSG3857.project(bounds.getSouthWest()),
305                         L.CRS.EPSG3857.project(bounds.getNorthEast())).getSize(),
306         maxScale = Math.floor(Math.sqrt(size.x * size.y / 0.3136));
307
308       $('#mapnik_minlon').val(bounds.getWest());
309       $('#mapnik_minlat').val(bounds.getSouth());
310       $('#mapnik_maxlon').val(bounds.getEast());
311       $('#mapnik_maxlat').val(bounds.getNorth());
312
313       if (scale < maxScale) {
314         scale = roundScale(maxScale);
315         $("#mapnik_scale").val(scale);
316       }
317
318       $("#mapnik_image_width").text(Math.round(size.x / scale / 0.00028));
319       $("#mapnik_image_height").text(Math.round(size.y / scale / 0.00028));
320     }
321
322     function select() {
323       $(this).select();
324     }
325
326     function getScale() {
327       var bounds = map.getBounds(),
328         centerLat = bounds.getCenter().lat,
329         halfWorldMeters = 6378137 * Math.PI * Math.cos(centerLat * Math.PI / 180),
330         meters = halfWorldMeters * (bounds.getEast() - bounds.getWest()) / 180,
331         pixelsPerMeter = map.getSize().x / meters,
332         metersPerPixel = 1 / (92 * 39.3701);
333       return Math.round(1 / (pixelsPerMeter * metersPerPixel));
334     }
335
336     function roundScale(scale) {
337       var precision = 5 * Math.pow(10, Math.floor(Math.LOG10E * Math.log(scale)) - 2);
338       return precision * Math.ceil(scale / precision);
339     }
340
341     return $container[0];
342   };
343
344   return control;
345 };