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