var map; var draw_control; var layers = {}; var recorded_obj; var marker_store = {}; var marker_store_sync_id; function eventmap_send_update() { var update_doc = { 'sync-id': marker_store_sync_id, 'markers': {} }; $.each(marker_store, function(marker_name, marker) { update_doc.markers[marker_name] = {}; var marker_info = update_doc.markers[marker_name]; marker_info.lat = marker.getLatLng().lat; marker_info.lng = marker.getLatLng().lng; marker_info.layer = marker.options.layer_name; }); $.ajax({ url: 'api/markers/post', type: 'POST', contentType: 'application/json', data: JSON.stringify(update_doc), processData: false, dataType: 'json' }); } function eventmap_process_update(data) { if (typeof data == "string") data = JSON.parse(data); marker_store_sync_id = data['sync-id']; $.each(data['markers'], function(marker_name, marker_info) { if (marker_name in marker_store) { var marker = marker_store[marker_name]; var marker_pos = marker.getLatLng(); if (marker_pos.lat != marker_info.lat || marker_pos.lng != marker_info.lng) marker.setLatLng([marker_info.lat, marker_info.lng]); if (marker.options.layer_name != marker_info.layer) { var old_lg = layers[marker.options.layer_name]; var new_lg = layers[marker_info.layer]; var old_draw = old_lg.getLayers()[1]; var new_draw = new_lg.getLayers()[1]; old_draw.removeLayer(marker); new_draw.addLayer(marker); marker.options.layer_name = marker_info.layer; } marker.options.sync_id = marker_store_sync_id; console.log("Kept marker '" + marker_name + "'."); } else { var marker = L.marker([marker_info.lat, marker_info.lng]); marker.bindLabel('', { noHide: marker_labels_no_hide }); add_contextmenu(marker); marker.options.label_text = marker_name; marker.updateLabelContent(marker.options.label_text); marker_store[marker_name] = marker; marker.options.layer_name = marker_info.layer layers[marker_info.layer].getLayers()[1].addLayer(marker); marker.options.sync_id = marker_store_sync_id; console.log("Added marker '" + marker_name + "'."); } }); for (var marker_name in marker_store) { if (marker_store[marker_name].options.sync_id == marker_store_sync_id) continue; var marker = marker_store[marker_name]; layers[marker.options.layer_name].getLayers()[1].removeLayer(marker); delete marker_store[marker_name]; console.log("Removed marker '" + marker_name + "'."); } (function longpoll() { $.ajax({ url: 'api/markers/poll/' + marker_store_sync_id, timeout: 600000 }).done(eventmap_process_update).fail(function() { setTimeout(longpoll, 10000); }); })(); } function delete_marker(marker) { layers[marker.options.layer_name].getLayers()[1].removeLayer(marker); delete marker_store[marker.options.label_text]; eventmap_send_update(); } function add_contextmenu(marker) { marker.options.contextmenu = true; marker.options.contextmenuItems = [ { text: 'Move', callback: function() { move_marker(marker); } }, { text: 'Rename', callback: function() { rename_marker(marker); } }, { text: 'Delete', callback: function() { delete_marker(marker); } }, ]; $.each(layers, function(layer_name, layer_object) { marker.options.contextmenuItems.push({ text: 'Send to ' + layer_name, callback: function() { $.each(layers, function(key, value) { map.removeLayer(value); drawing_layer = value.getLayers()[1]; if (drawing_layer.hasLayer(marker)) drawing_layer.removeLayer(marker); }); map.addLayer(layer_object); layer_object.getLayers()[1].addLayer(marker); marker.options.layer_name = layer_name; eventmap_send_update(); } }) }); marker._initContextMenu(); } /* Functionality of (re)naming a marker - if I understood how objects worked * in javascript, this should probably be one. :/ */ function rename_marker(marker) { var label_text; var new_label_text; if (marker.options.label_text === undefined) label_text = ''; else label_text = marker.options.label_text; do { new_label_text = prompt("Please enter name", label_text); if (new_label_text in marker_store && marker_store[new_label_text] !== marker) { alert("This name is not unique!"); } else { break; } } while (1); if (marker.options.label_text !== undefined) delete marker_store[label_text] marker.options.label_text = new_label_text; marker.updateLabelContent(marker.options.label_text); marker_store[new_label_text] = marker eventmap_send_update(); } /* Functionality of moving a marker - if I understood how objects worked * in javascript, this should probably be one. :/ */ var move_marker_orig_latlng; var move_marker_marker; function move_marker_mousemove(e) { move_marker_marker.setLatLng(e.latlng); } function move_marker_enable_events() { L.DomEvent.addListener(map._container, 'keyup', move_marker_keyup); map.on('mousemove', move_marker_mousemove) map.on('click', move_marker_commit); move_marker_marker.on('click', move_marker_commit); } function move_marker_disable_events() { L.DomEvent.removeListener(map._container, 'keyup', move_marker_keyup); map.off('mousemove', move_marker_mousemove); map.off('click', move_marker_commit); move_marker_marker.off('click', move_marker_commit); } function move_marker_commit(e) { move_marker_disable_events(); eventmap_send_update(); } function move_marker_keyup(e) { if (e.keyCode === 27) { move_marker_marker.setLatLng(move_marker_orig_latlng); move_marker_disable_events(); } } function move_marker(marker) { move_marker_marker = marker; move_marker_orig_latlng = marker.getLatLng(); move_marker_enable_events(); } /* other functionality */ var marker_labels_no_hide = false; function marker_labels_calc_nohide(e) { marker_labels_no_hide = (map.getZoom() >= 3); $.each(layers, function(layer_name, layer_group) { var drawing_layer; drawing_layer = layer_group.getLayers()[1] $.each(drawing_layer.getLayers(), function(marker_index, marker) { marker.setLabelNoHide(marker_labels_no_hide); }); }); } $(function() { map = L.map('map', { center: new L.LatLng(70,-50), contextmenu: true, zoom: 2 }); map.on('zoomend', marker_labels_calc_nohide); draw_control = new L.Control.Draw({ }); map.addControl(draw_control); map.on('draw:created', function (e) { var created_object_type = e.layerType; var created_object = e.layer; if (created_object_type !== 'marker') { return; } created_object.bindLabel('', { noHide: marker_labels_no_hide }); $.each(layers, function(layer_name, layer_object) { if (!map.hasLayer(layer_object)) return true; add_contextmenu(created_object); layer_object.getLayers()[1].addLayer(created_object); created_object.options.layer_name = layer_name; return false; }); rename_marker(created_object); /* update will be sent by "rename_marker" */ }); $.getJSON('js/layers.json', function(data) { var first_layer = true; $.each(data, function(layer_index, layer_info) { var layer_path; var base_layer; var drawing_layer; var layer; layer_path = 'images/tiles/' + layer_info.name + '/{z}/{x}/{y}.png'; base_layer = L.tileLayer(layer_path, { noWrap: true, continuousWorld: true, maxZoom: layer_info.max_zoom }); drawing_layer = new L.FeatureGroup(); layer = L.layerGroup([base_layer, drawing_layer]); layers[layer_info.name] = layer; if (first_layer) { map.addLayer(layer); first_layer = false; } }); L.control.layers(layers, {}).addTo(map); $.ajax({ url: 'api/markers/get' }).done(eventmap_process_update).fail(function() { alert("Couldn't load marker info from server!"); }); }); });