page.replace_html :sidebar_title, t('browse.start_rjs.data_frame_title') page.replace_html :sidebar_content, :partial => 'start' page << <= 15) { useMap(); } else { setStatus("#{I18n.t('browse.start_rjs.zoom_or_select')}"); } } } function stopBrowse() { if (browseActive) { browseActive = false; if (browseSelectControl) { browseSelectControl.destroy(); browseSelectControl = null; } if (browseBoxControl) { browseBoxControl.destroy(); browseBoxControl = null; } if (browseActiveFeature) { browseActiveFeature.destroy(); browseActiveFeature = null; } if (browseDataLayer) { browseDataLayer.destroy(); browseDataLayer = null; } map.dataLayer.setVisibility(false); map.events.unregister("moveend", map, showData); } } function startDrag() { $("browse_select_box").innerHTML='#{I18n.t('browse.start_rjs.drag_a_box')}'; browseBoxControl.activate(); return false; }; $("browse_select_box").onclick = startDrag; function useMap() { var bounds = map.getExtent(); var projected = bounds.clone().transform(map.getProjectionObject(), epsg4326); if (!browseBounds || !browseBounds.containsBounds(projected)) { var center = bounds.getCenterLonLat(); var tileWidth = bounds.getWidth() * 1.2; var tileHeight = bounds.getHeight() * 1.2; var tileBounds = new OpenLayers.Bounds(center.lon - (tileWidth / 2), center.lat - (tileHeight / 2), center.lon + (tileWidth / 2), center.lat + (tileHeight / 2)); browseBounds = tileBounds; getData(tileBounds); browseMode = "auto"; $("browse_select_view").style.display = "none"; } return false; } $("browse_select_view").onclick = useMap; function endDrag(bbox) { var bounds = bbox.getBounds(); var projected = bounds.clone().transform(map.getProjectionObject(), epsg4326); browseBoxControl.deactivate(); browseBounds = projected; getData(bounds); browseMode = "manual"; $("browse_select_box").innerHTML = "#{I18n.t('browse.start_rjs.manually_select')}"; $("browse_select_view").style.display = "inline"; } function displayFeatureWarning() { clearStatus(); var div = document.createElement("div"); var p = document.createElement("p"); p.appendChild(document.createTextNode(i18n_js("#{I18n.t('browse.start_rjs.loaded_an_area_with_num_features', :num_features => '{{num_features}}')}", { num_features: browseFeatureList.length }))); div.appendChild(p); var input = document.createElement("input"); input.type = "submit"; input.value = "#{I18n.t('browse.start_rjs.load_data')}"; input.onclick = loadFeatureList; div.appendChild(input); $("browse_content").innerHTML = ""; $("browse_content").appendChild(div); } function loadFeatureList() { browseDataLayer.addFeatures(browseFeatureList); browseDataLayer.events.triggerEvent("loadend"); browseFeatureList = []; return false; } function customDataLoader(request) { if (browseActive) { var doc = request.responseXML; if (!doc || !doc.documentElement) { doc = request.responseText; } var options = {}; OpenLayers.Util.extend(options, this.formatOptions); if (this.map && !this.projection.equals(this.map.getProjectionObject())) { options.externalProjection = this.projection; options.internalProjection = this.map.getProjectionObject(); } var gml = this.format ? new this.format(options) : new OpenLayers.Format.GML(options); browseFeatureList = gml.read(doc); if (!this.maxFeatures || browseFeatureList.length <= this.maxFeatures) { loadFeatureList(); } else { displayFeatureWarning(); } } } function getData(bounds) { var projected = bounds.clone().transform(new OpenLayers.Projection("EPSG:900913"), new OpenLayers.Projection("EPSG:4326")); var size = projected.getWidth() * projected.getHeight(); if (size > #{APP_CONFIG['max_request_area']}) { setStatus(i18n_js("#{I18n.t('browse.start_rjs.unable_to_load_size', :bbox_size => '{{bbox_size}}', :max_bbox_size => APP_CONFIG['max_request_area'])}", { bbox_size: size })); } else { loadGML("/api/#{API_VERSION}/map?bbox=" + projected.toBBOX()); } } function loadGML(url) { setStatus("#{I18n.t('browse.start_rjs.loading')}"); $("browse_content").innerHTML = ""; if (!browseDataLayer) { var style = new OpenLayers.Style(); style.addRules([new OpenLayers.Rule({ symbolizer: { Polygon: { fillColor: '#ff0000', strokeColor: '#ff0000' }, Line: { fillColor: '#ffff00', strokeColor: '#000000', strokeOpacity: '0.4' }, Point: { fillColor: '#00ff00', strokeColor: '#00ff00' } } })]); browseDataLayer = new OpenLayers.Layer.GML("Data", url, { format: OpenLayers.Format.OSM, formatOptions: { checkTags: true, interestingTagsExclude: ['source','source_ref','source:ref','history','attribution','created_by','tiger:county','tiger:tlid','tiger:upload_uuid'] }, maxFeatures: 100, requestSuccess: customDataLoader, displayInLayerSwitcher: false, styleMap: new OpenLayers.StyleMap({ 'default': style, 'select': { strokeColor: '#0000ff', strokeWidth: 8 } }) }); browseDataLayer.events.register("loadend", browseDataLayer, dataLoaded ); map.addLayer(browseDataLayer); browseSelectControl = new OpenLayers.Control.SelectFeature(browseDataLayer, { onSelect: onFeatureSelect }); browseSelectControl.handlers.feature.stopDown = false; browseSelectControl.handlers.feature.stopUp = false; map.addControl(browseSelectControl); browseSelectControl.activate(); } else { browseDataLayer.setUrl(url); } browseActiveFeature = null; } function dataLoaded() { if (browseActive) { clearStatus(); browseObjectList = document.createElement("div") var heading = document.createElement("p"); heading.className = "browse_heading"; heading.appendChild(document.createTextNode("#{I18n.t('browse.start_rjs.object_list.heading')}")); browseObjectList.appendChild(heading); var list = document.createElement("ul"); for (var i = 0; i < this.features.length; i++) { var feature = this.features[i]; // Type, for linking var type = featureType(feature); var typeName = featureTypeName(feature); var li = document.createElement("li"); li.appendChild(document.createTextNode(typeName + " ")); // Link, for viewing in the tab var link = document.createElement("a"); link.href = "/browse/" + type + "/" + feature.osm_id; var name = feature.attributes.name || feature.osm_id; link.appendChild(document.createTextNode(name)); link.feature = feature; link.onclick = OpenLayers.Function.bind(viewFeatureLink, link); li.appendChild(link); list.appendChild(li); } browseObjectList.appendChild(list) var link = document.createElement("a"); link.href = this.url; link.appendChild(document.createTextNode("#{I18n.t('browse.start_rjs.object_list.api')}")); browseObjectList.appendChild(link); $("browse_content").innerHTML = ""; $("browse_content").appendChild(browseObjectList); } } function viewFeatureLink() { var layer = this.feature.layer; for (var i = 0; i < layer.selectedFeatures.length; i++) { var f = layer.selectedFeatures[i]; layer.drawFeature(f, layer.styleMap.createSymbolizer(f, "default")); } onFeatureSelect(this.feature); if (browseMode != "auto") { map.setCenter(this.feature.geometry.getBounds().getCenterLonLat()); } return false; } function loadObjectList() { $("browse_content").innerHTML=""; $("browse_content").appendChild(browseObjectList); return false; } function onFeatureSelect(feature) { // Unselect previously selected feature if (browseActiveFeature) { browseActiveFeature.layer.drawFeature( browseActiveFeature, browseActiveFeature.layer.styleMap.createSymbolizer(browseActiveFeature, "default") ); } // Redraw in selected style feature.layer.drawFeature( feature, feature.layer.styleMap.createSymbolizer(feature, "select") ); // If the current object is the list, don't innerHTML="", since that could clear it. if ($("browse_content").firstChild == browseObjectList) { $("browse_content").removeChild(browseObjectList); } else { $("browse_content").innerHTML = ""; } // Create a link back to the object list var div = document.createElement("div"); div.style.textAlign = "center"; div.style.marginBottom = "20px"; $("browse_content").appendChild(div); var link = document.createElement("a"); link.href = "#"; link.onclick = loadObjectList; link.appendChild(document.createTextNode("#{I18n.t('browse.start_rjs.object_list.back')}")); div.appendChild(link); var table = document.createElement("table"); table.width = "100%"; table.className = "browse_heading"; $("browse_content").appendChild(table); var tr = document.createElement("tr"); table.appendChild(tr); var heading = document.createElement("td"); heading.appendChild(document.createTextNode(featureNameSelect(feature))); tr.appendChild(heading); var td = document.createElement("td"); td.align = "right"; tr.appendChild(td); var type = featureType(feature); var link = document.createElement("a"); link.href = "/browse/" + type + "/" + feature.osm_id; link.appendChild(document.createTextNode("#{I18n.t('browse.start_rjs.object_list.details')}")); td.appendChild(link); var div = document.createElement("div"); div.className = "browse_details"; $("browse_content").appendChild(div); // Now the list of attributes var ul = document.createElement("ul"); for (var key in feature.attributes) { var li = document.createElement("li"); var b = document.createElement("b"); b.appendChild(document.createTextNode(key)); li.appendChild(b); li.appendChild(document.createTextNode(": " + feature.attributes[key])); ul.appendChild(li); } div.appendChild(ul); var link = document.createElement("a"); link.href = "/browse/" + type + "/" + feature.osm_id + "/history"; link.appendChild(document.createTextNode("#{I18n.t('browse.start_rjs.show_history')}")); link.onclick = OpenLayers.Function.bind(loadHistory, { type: type, feature: feature, link: link }); div.appendChild(link); // Stash the currently drawn feature browseActiveFeature = feature; } function loadHistory() { this.link.href = ""; this.link.innerHTML = "#{I18n.t('browse.start_rjs.wait')}"; new Ajax.Request("/api/#{API_VERSION}/" + this.type + "/" + this.feature.osm_id + "/history", { onComplete: OpenLayers.Function.bind(displayHistory, this) }); return false; } function displayHistory(request) { if (browseActiveFeature.osm_id != this.feature.osm_id || $("browse_content").firstChild == browseObjectList) { return false; } this.link.parentNode.removeChild(this.link); var doc = request.responseXML; var table = document.createElement("table"); table.width = "100%"; table.className = "browse_heading"; $("browse_content").appendChild(table); var tr = document.createElement("tr"); table.appendChild(tr); var heading = document.createElement("td"); heading.appendChild(document.createTextNode(i18n_js("#{I18n.t('browse.start_rjs.history_for_feature', :feature => '{{feature}}')}", { feature: featureNameHistory(this.feature) }))); tr.appendChild(heading); var td = document.createElement("td"); td.align = "right"; tr.appendChild(td); var link = document.createElement("a"); link.href = "/browse/" + this.type + "/" + this.feature.osm_id + "/history"; link.appendChild(document.createTextNode("#{I18n.t('browse.start_rjs.details')}")); td.appendChild(link); var div = document.createElement("div"); div.className = "browse_details"; var nodes = doc.getElementsByTagName(this.type); var history = document.createElement("ul"); for (var i = nodes.length - 1; i >= 0; i--) { var user = nodes[i].getAttribute("user") || "#{I18n.t('browse.start_rjs.private_user')}"; var timestamp = nodes[i].getAttribute("timestamp"); var item = document.createElement("li"); item.appendChild(document.createTextNode(i18n_js("#{I18n.t('browse.start_rjs.edited_by_user_at_timestamp', :user => '{{user}}', :timestamp => '{{timestamp}}')}", { user: user, timestamp: timestamp }))); history.appendChild(item); } div.appendChild(history); $("browse_content").appendChild(div); } function featureType(feature) { if (feature.geometry.CLASS_NAME == "OpenLayers.Geometry.Point") { return "node"; } else { return "way"; } } function featureTypeName(feature) { if (featureType(feature) == "node") { return "#{I18n.t('browse.start_rjs.object_list.type.node')}"; } else if (featureType(feature) == "way") { return "#{I18n.t('browse.start_rjs.object_list.type.way')}"; } } function featureNameSelect(feature) { if (feature.attributes.name) { return feature.attributes.name; } else if (featureType(feature) == "node") { return i18n_js("#{I18n.t('browse.start_rjs.object_list.selected.type.node', :id => '{{id}}')}", { id: feature.osm_id }); } else if (featureType(feature) == "way") { return i18n_js("#{I18n.t('browse.start_rjs.object_list.selected.type.way', :id => '{{id}}')}", { id: feature.osm_id }); } } function featureNameHistory(feature) { if (feature.attributes.name) { return feature.attributes.name; } else if (featureType(feature) == "node") { return i18n_js("#{I18n.t('browse.start_rjs.object_list.history.type.node', :id => '{{id}}')}", { id: feature.osm_id }); } else if (featureType(feature) == "way") { return i18n_js("#{I18n.t('browse.start_rjs.object_list.history.type.way', :id => '{{id}}')}", { id: feature.osm_id }); } } /* This is a hack so that we can interpolate JavaScript variables in translation strings from Ruby, doing the string replacement client-side instead of on the server. */ function i18n_js(string, keys) { for (var key in keys) { var re_key = '\{\{' + key + '\}\}'; var re = new RegExp(re_key, "g"); string = string.replace(re, keys[key]); } return string; } function setStatus(status) { $("browse_status").innerHTML = status; $("browse_status").style.display = "block"; } function clearStatus() { $("browse_status").innerHTML = ""; $("browse_status").style.display = "none"; } startBrowse(); EOJ