//= require templates/notes/show
//= require templates/notes/new
-$(document).ready(function () {
- var params = OSM.mapParams();
- var newNotes;
-
- function saveNewNotes(o) {
- var layer = o.object;
- newNotes = layer.getFeaturesByAttribute("status", "new")
- layer.removeFeatures(newNotes, { silent: true });
- }
+function initializeNotes(map) {
+ var params = OSM.mapParams(),
+ noteLayer = map.noteLayer,
+ notes = {},
+ newNote;
+
+ var noteIcons = {
+ "new": L.icon({
+ iconUrl: "<%= image_path 'new_note_marker.png' %>",
+ iconSize: [25, 40],
+ iconAnchor: [12, 40]
+ }),
+ "open": L.icon({
+ iconUrl: "<%= image_path 'open_note_marker.png' %>",
+ iconSize: [25, 40],
+ iconAnchor: [12, 40]
+ }),
+ "closed": L.icon({
+ iconUrl: "<%= image_path 'closed_note_marker.png' %>",
+ iconSize: [25, 40],
+ iconAnchor: [12, 40]
+ })
+ };
+
+ map.on("layeradd", function (e) {
+ if (e.layer == noteLayer) {
+ loadNotes();
+ map.on("moveend", loadNotes);
+ }
+ }).on("layerremove", function (e) {
+ if (e.layer == noteLayer) {
+ map.off("moveend", loadNotes);
+ noteLayer.clearLayers();
+ notes = {};
+ }
+ }).on("popupclose", function (e) {
+ if (newNote && e.popup == newNote._popup) {
+ $(newNote).oneTime(10, "removenote", function () {
+ map.removeLayer(newNote);
+ newNote = null;
+ });
+ }
+ }).on("popupopen", function (e) {
+ if (!('ontouchstart' in document.documentElement)) {
+ $(e.popup._container).find(".comment").focus();
+ }
+ });
- function restoreNewNotes(o) {
- var layer = o.object;
- layer.addFeatures(newNotes);
- newNotes = undefined;
+ if (OSM.STATUS != 'api_offline' && OSM.STATUS != 'database_offline') {
+ if (params.notes || params.layers.indexOf('N') >= 0) map.addLayer(noteLayer);
+ if (params.note) {
+ $.ajax({
+ url: "/api/" + OSM.API_VERSION + "/notes/" + params.note + ".json",
+ success: function (feature) {
+ var marker = updateMarker(notes[feature.properties.id], feature);
+ notes[feature.properties.id] = marker;
+ map.addLayer(noteLayer);
+ marker.openPopup();
+ }
+ });
+ }
}
- function createNote(feature, form) {
- var location = unproj(feature.geometry.getBounds().getCenterLonLat());
-
- $(form).find("input[type=submit]").prop("disabled", true);
+ function updateMarker(marker, feature) {
+ if (marker) {
+ marker.setIcon(noteIcons[feature.properties.status]);
+ marker.setPopupContent(createPopupContent(
+ marker, feature.properties,
+ $(marker._popup._content).find("textarea").val()
+ ));
+ } else {
+ marker = L.marker(feature.geometry.coordinates.reverse(), {
+ icon: noteIcons[feature.properties.status],
+ opacity: 0.9
+ });
+ marker.addTo(noteLayer).bindPopup(
+ createPopupContent(marker, feature.properties),
+ popupOptions()
+ );
+ }
+ return marker;
+ }
- $.ajax($("#createnoteanchor").attr("href"), {
- type: "POST",
- data: {
- lon: location.lon,
- lat: location.lat,
- text: $(form.text).val()
- },
- success: function (data) {
- map.noteSelector.unselect(feature);
+ var noteLoader;
- feature.attributes = data.properties;
+ function loadNotes() {
+ var bounds = map.getBounds();
+ var size = bounds.getSize();
- map.noteLayer.drawFeature(feature);
+ if (size <= OSM.MAX_NOTE_REQUEST_AREA) {
+ var url = "/api/" + OSM.API_VERSION + "/notes.json?bbox=" + bounds.toBBoxString();
- map.noteMover.deactivate();
- }
- });
- }
+ if (noteLoader) noteLoader.abort();
- function updateNote(feature, form, close) {
- var url = close ? feature.attributes.close_url : feature.attributes.comment_url;
+ noteLoader = $.ajax({
+ url: url,
+ success: success
+ });
+ }
- $(form).find("input[type=submit]").prop("disabled", true);
+ function success(json) {
+ var oldNotes = notes;
+ notes = {};
+ json.features.forEach(updateMarkers);
- $.ajax(url, {
- type: "POST",
- data: {
- text: $(form.text).val()
- },
- success: function (data) {
- map.noteSelector.unselect(feature)
-
- feature.attributes = data.properties;
-
- map.noteSelector.select(feature)
+ function updateMarkers(feature) {
+ var marker = oldNotes[feature.properties.id];
+ delete oldNotes[feature.properties.id];
+ notes[feature.properties.id] = updateMarker(marker, feature);
}
- });
- }
-
- function noteSelected(o) {
- var feature = o.feature;
- var location = feature.geometry.getBounds().getCenterLonLat();
- var content;
- var onClose;
-
- if (feature.attributes.status === "new") {
- content = JST["templates/notes/new"]();
- onClose = function (e) {
- feature.attributes.status = "cancelled";
-
- map.noteSelector.unselect(feature);
- map.noteLayer.removeFeatures(feature);
+ for (id in oldNotes) {
+ noteLayer.removeLayer(oldNotes[id]);
+ }
- feature.destroy();
+ noteLoader = null;
+ }
+ };
- map.noteMover.deactivate();
- };
- } else {
- content = JST["templates/notes/show"]({ note: feature.attributes });
+ function popupOptions() {
+ var mapSize = map.getSize();
- onClose = function (e) {
- map.noteSelector.unselect(feature)
- };
+ return {
+ minWidth: 320,
+ maxWidth: mapSize.y * 1 / 3,
+ maxHeight: mapSize.y * 2 / 3,
+ offset: new L.Point(0, -40),
+ autoPanPadding: new L.Point(60, 40)
};
+ }
- feature.popup = new OpenLayers.Popup.FramedCloud(
- feature.attributes.id, location, null, content, null, true, onClose
- );
-
- map.addPopup(feature.popup);
- // feature.popup.show();
-
- $(feature.popup.contentDiv).find("textarea").autoGrow();
+ function createPopupContent(marker, properties, comment) {
+ var content = $(JST["templates/notes/show"]({ note: properties }));
- $(feature.popup.contentDiv).find("textarea").on("input", function (e) {
+ content.find("textarea").on("input", function (e) {
var form = e.target.form;
if ($(e.target).val() == "") {
- $(form.close).val(I18n.t("javascripts.notes.show.close"));
+ $(form.close).val(I18n.t("javascripts.notes.show.resolve"));
+ $(form.comment).prop("disabled", true);
} else {
- $(form.close).val(I18n.t("javascripts.notes.show.comment_and_close"));
+ $(form.close).val(I18n.t("javascripts.notes.show.comment_and_resolve"));
+ $(form.comment).prop("disabled", false);
}
});
- $(feature.popup.contentDiv).find("input#note-add").click(function (e) {
+ content.find("input[type=submit]").on("click", function (e) {
e.preventDefault();
-
- createNote(feature, e.target.form);
+ var data = $(e.target).data();
+ updateNote(marker, e.target.form, data.method, data.url);
});
- $(feature.popup.contentDiv).find("input#note-comment").click(function (e) {
- e.preventDefault();
+ if (comment) {
+ content.find("textarea").val(comment).trigger("input");
+ }
- updateNote(feature, e.target.form, false);
- });
+ return content[0];
+ }
- $(feature.popup.contentDiv).find("input#note-close").click(function (e) {
- e.preventDefault();
+ function createNote(marker, form, url) {
+ var location = marker.getLatLng();
+
+ marker.options.draggable = false;
+ marker.dragging.disable();
- updateNote(feature, e.target.form, true);
+ $(form).find("input[type=submit]").prop("disabled", true);
+
+ $.ajax({
+ url: url,
+ type: "POST",
+ oauth: true,
+ data: {
+ lat: location.lat,
+ lon: location.lng,
+ text: $(form.text).val()
+ },
+ success: noteCreated
});
- feature.popup.updateSize();
- }
+ function noteCreated(feature) {
+ $(marker._popup._content).find("textarea").val("");
- function noteUnselected(o) {
- var feature = o.feature;
+ notes[feature.properties.id] = updateMarker(marker, feature);
+ newNote = null;
- map.removePopup(feature.popup);
+ $("#createnoteanchor").removeClass("disabled").addClass("geolink");
+ }
}
- function addNote() {
- var lonlat = map.getCenter();
- var layer = map.noteLayer;
- var geometry = new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat);
- var feature = new OpenLayers.Feature.Vector(geometry, {
- status: "new"
- });
+ function updateNote(marker, form, method, url) {
+ $(form).find("input[type=submit]").prop("disabled", true);
- layer.addFeatures(feature);
- map.noteSelector.unselectAll();
- map.noteSelector.select(feature);
- map.noteMover.activate();
- map.noteLayer.setVisibility(true);
- }
+ $.ajax({
+ url: url,
+ type: method,
+ oauth: true,
+ data: {
+ text: $(form.text).val()
+ },
+ success: function (feature) {
+ if (feature.properties.status == "hidden") {
+ noteLayer.removeLayer(marker);
- $("#map").on("initialised", function () {
- map.noteLayer = new OpenLayers.Layer.Vector("Notes", {
- visibility: params.notes,
- displayInLayerSwitcher: false,
- projection: new OpenLayers.Projection("EPSG:4326"),
- styleMap: new OpenLayers.StyleMap(new OpenLayers.Style({
- graphicWidth: 22,
- graphicHeight: 22,
- graphicOpacity: 0.7,
- graphicXOffset: -11,
- graphicYOffset: -11
- }, {
- rules: [
- new OpenLayers.Rule({
- filter: new OpenLayers.Filter.Comparison({
- type: OpenLayers.Filter.Comparison.EQUAL_TO,
- property: "status",
- value: "new"
- }),
- symbolizer: {
- externalGraphic: "<%= image_path 'new_note_marker.png' %>"
- }
- }),
- new OpenLayers.Rule({
- filter: new OpenLayers.Filter.Comparison({
- type: OpenLayers.Filter.Comparison.EQUAL_TO,
- property: "status",
- value: "open"
- }),
- symbolizer: {
- externalGraphic: "<%= image_path 'open_note_marker.png' %>"
- }
- }),
- new OpenLayers.Rule({
- filter: new OpenLayers.Filter.Comparison({
- type: OpenLayers.Filter.Comparison.EQUAL_TO,
- property: "status",
- value: "closed"
- }),
- symbolizer: {
- externalGraphic: "<%= image_path 'closed_note_marker.png' %>"
- }
- })
- ]
- })),
- strategies: [
- new OpenLayers.Strategy.BBOX()
- ],
- protocol: new OpenLayers.Protocol.HTTP({
- url: $("#show_notes").attr("href"),
- format: new OpenLayers.Format.GeoJSON()
- })
+ delete notes[feature.properties.id];
+ } else {
+ var popupContent = createPopupContent(marker, feature.properties);
+
+ marker.setIcon(noteIcons[feature.properties.status]);
+ marker.setPopupContent(popupContent);
+ }
+ }
});
+ }
- map.noteLayer.events.register("beforefeaturesremoved", map, saveNewNotes);
- map.noteLayer.events.register("featuresremoved", map, restoreNewNotes);
- map.noteLayer.events.register("featureselected", map, noteSelected);
- map.noteLayer.events.register("featureunselected", map, noteUnselected);
+ $(".leaflet-control-attribution").on("click", "#createnoteanchor", function (e) {
+ e.preventDefault();
- map.addLayer(map.noteLayer);
+ if ($(e.target).hasClass("disabled")) return;
- map.noteSelector = new OpenLayers.Control.SelectFeature(map.noteLayer, {
- autoActivate: true
- });
+ $(e.target).removeClass("geolink").addClass("disabled");
- map.addControl(map.noteSelector);
+ map.addLayer(noteLayer);
- map.noteMover = new OpenLayers.Control.DragFeature(map.noteLayer, {
- onDrag: function (feature, pixel) {
- feature.popup.lonlat = feature.geometry.getBounds().getCenterLonLat();
- feature.popup.updatePosition();
- },
- featureCallbacks: {
- over: function (feature) {
- if (feature.attributes.status === "new") {
- map.noteMover.overFeature.apply(map.noteMover, [feature]);
- }
- }
- }
+ var mapSize = map.getSize();
+ var markerPosition;
+
+ if (mapSize.y > 800) {
+ markerPosition = [mapSize.x / 2, mapSize.y / 2];
+ } else if (mapSize.y > 400) {
+ markerPosition = [mapSize.x / 2, 400];
+ } else {
+ markerPosition = [mapSize.x / 2, mapSize.y];
+ }
+
+ newNote = L.marker(map.containerPointToLatLng(markerPosition), {
+ icon: noteIcons["new"],
+ opacity: 0.9,
+ draggable: true
});
- map.addControl(map.noteMover);
+ var popupContent = $(JST["templates/notes/new"]({
+ create_url: $(e.target).attr("href")
+ }));
+
+ popupContent.find("textarea").on("input", disableWhenBlank);
- $("#show_notes").click(function (e) {
- map.noteLayer.setVisibility(true);
+ function disableWhenBlank(e) {
+ $(e.target.form).prop("disabled", $(e.target).val() === "");
+ }
+ popupContent.find("input[type=submit]").on("click", function (e) {
e.preventDefault();
+ createNote(newNote, e.target.form, $(e.target).data("url"));
});
- $("#createnoteanchor").click(function (e) {
- map.noteLayer.setVisibility(true);
+ newNote.addTo(noteLayer).bindPopup(popupContent[0], popupOptions()).openPopup();
- addNote();
-
- e.preventDefault();
+ newNote.on("remove", function (e) {
+ $("#createnoteanchor").removeClass("disabled").addClass("geolink");
+ }).on("dragstart", function (e) {
+ $(newNote).stopTime("removenote");
+ }).on("dragend", function (e) {
+ e.target.openPopup();
});
});
-});
+}