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